Since Rector 0.12 a new RectorConfig
is available with simpler and easier to use config methods.
When it comes to completing type declaration from docblocks, we rely on trust and hopes in commented code. One way out of is dynamic analysis that works with real data that enter the method. But we have to log it, wait for it, and update our codebase based on logged data.
Is there a faster, simpler solution we can just plugin?
Let's say we have a Person
object:
final class Person
{
/**
* @var string
*/
public $name;
/**
* @param string $name
*/
public function __construct($name)
{
$this->name = $name;
}
/**
* @return string
*/
public function getName()
{
return $this->name;
}
}
How sure are you about the name being a string
? 80-95 %? Every percent under 100 % is a probability of a bug behind the corner.
We can do this:
$person = new Person(1000);
Or even this (don't try to imagine it):
$person = new Person(1000);
$anotherPerson = new Person($person);
See 3v4l.org.
Rector has a TYPE_DECLARATION
set to autocomplete types based PHPStan types that rely heavily on docblocks even in strict mode. This set is useful for keeping high code quality but might break some older code.
The way out of legacy is to completely type declaration right in PHP to every single place it can appear:
Such work has enormous benefits, as we can rely 100 % on the types and move our focus on more critical parts. But it is tedious and prolonged work.
When it comes to a single type of declaration, there is more than meets the eye. More encoded knowledge is not visible to the human eye, but it is there.
Let's say we add a single type we are sure off:
final class Person
{
/**
* @var string
*/
public $name;
- /**
- * @param string $name
- */
- public function __construct($name)
+ public function __construct(string $name)
{
$this->name = $name;
}
/**
* @return string
*/
public function getName()
{
return $this->name;
}
}
The param $name
in a constructor is always a string. What does it mean for the rest of the code? Assign in the constructor to property means that property uses identical type:
final class Person
{
- /**
- * @var string
- */
- public $name;
+ public string $name;
public function __construct(string $name)
{
$this->name = $name;
}
/**
* @return string
*/
public function getName()
{
return $this->name;
}
}
The $name
property is now the string
type right from the object construction. In effect, any getter inherits the same type:
class Person
{
public string $name;
public function __construct(string $name)
{
$this->name = $name;
}
- /**
- * @return string
- */
- public function getName()
+ public function getName(): string
{
return $this->name;
}
}
Now we have an object fully typed, and all we had to do is complete a single type in constructor.
What about places that are using the Person
object?
final class PersonScanner
{
- public function getPersonName(Person $person)
+ public function getPersonName(Person $person): string
{
return $person->getName();
}
}
And all methods using PersonScanner->getPersonName()
? They know the string
too. This healthy immunity is now spreading through our code base with every single type of declaration we add.
From single manually added type declaration Rector can autocomplete:
Rector watch will save you so much detailed detective work on types that are already in the code but hard to spot.
Add TYPE_DECLARATION_STICT
set yourself of pick rule by rule, so you can see how your code base becomes strict for each new rule you add:
use Rector\TypeDeclaration\Rector\Param\ParamTypeFromStrictTypedPropertyRector;
use Rector\TypeDeclaration\Rector\ClassMethod\ReturnTypeFromReturnNewRector;
use Rector\TypeDeclaration\Rector\ClassMethod\ReturnTypeFromStrictTypedCallRector;
use Rector\TypeDeclaration\Rector\ClassMethod\ReturnTypeFromStrictTypedPropertyRector;
use Rector\TypeDeclaration\Rector\Property\TypedPropertyFromStrictConstructorRector;
use Rector\Config\RectorConfig;
return function (RectorConfig $rectorConfig): void {
$rectorConfig->rule(ParamTypeFromStrictTypedPropertyRector::class);
$rectorConfig->rule(ReturnTypeFromReturnNewRector::class);
$rectorConfig->rule(ReturnTypeFromStrictTypedPropertyRector::class);
$rectorConfig->rule(ReturnTypeFromStrictTypedCallRector::class);
$rectorConfig->rule(TypedPropertyFromStrictConstructorRector::class);
};
Happy coding!