1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-10 15:09:04 +01:00
psalm/docs/manipulating_code/fixing.md

616 lines
8.4 KiB
Markdown
Raw Normal View History

2019-03-29 05:06:40 +01:00
# Fixing Code
2018-02-27 23:25:35 +01:00
Psalm is good at finding potential issues in large codebases, but once found, it can be something of a gargantuan task to fix all the issues.
2019-03-20 00:47:49 +01:00
It comes with a tool, called Psalter, that helps you fix code.
You can either run it via its binary
```
vendor/bin/psalter [args]
```
or via Psalm's binary:
```
vendor/bin/psalm --alter [args]
```
2018-02-27 23:27:02 +01:00
## Safety features
Updating code is inherently risky, doing so automatically is even more so. I've added a few features to make it a little more reassuring:
- To see what changes Psalter will make ahead of time, you can run it with `--dry-run`.
- You can target particular versions of PHP via `--php-version`, so that (for example) you don't add nullable typehints to PHP 7.0 code, or any typehints at all to PHP 5.6 code. `--php-version` defaults to your current version.
- it has a `--safe-types` mode that will only update PHP 7 return typehints with information Psalm has gathered from non-docblock sources of type information (e.g. typehinted params, `instanceof` checks, other return typehints etc.)
- using `--allow-backwards-incompatible-changes=false` you can make sure to not create backwards incompatible changes
2018-02-27 23:27:02 +01:00
2018-02-27 23:25:35 +01:00
## Plugins
You can pass in your own manipulation plugins e.g.
```bash
2019-06-20 15:05:22 +02:00
vendor/bin/psalter --plugin=vendor/vimeo/psalm/examples/plugins/ClassUnqualifier.php --dry-run
2018-02-27 23:25:35 +01:00
```
The above example plugin converts all unnecessarily qualified classnames in your code to shorter aliased versions.
## Supported fixes
This initial release provides support for the following alterations, corresponding to the names of issues Psalm finds.
To fix all of these at once, run `vendor/bin/psalter --issues=all`
2018-02-27 23:25:35 +01:00
### MissingReturnType
Running `vendor/bin/psalter --issues=MissingReturnType --php-version=7.0` on
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() {
return "hello";
}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() : string {
return "hello";
}
```
and running `vendor/bin/psalter --issues=MissingReturnType --php-version=5.6` on
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() {
return "hello";
}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
/**
* @return string
*/
function foo() {
return "hello";
}
```
### MissingClosureReturnType
As above, except for closures
### InvalidReturnType
Running `vendor/bin/psalter --issues=InvalidReturnType` on
```php
<?php
2018-02-27 23:25:35 +01:00
/**
* @return int
*/
function foo() {
return "hello";
}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
/**
* @return string
*/
function foo() {
return "hello";
}
```
There's also support for return typehints, so running `vendor/bin/psalter --issues=InvalidReturnType` on
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() : int {
return "hello";
}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() : string {
return "hello";
}
```
### InvalidNullableReturnType
Running `vendor/bin/psalter --issues=InvalidNullableReturnType --php-version=7.1` on
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() : string {
return rand(0, 1) ? "hello" : null;
}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() : ?string {
return rand(0, 1) ? "hello" : null;
}
```
and running `vendor/bin/psalter --issues=InvalidNullableReturnType --php-version=7.0` on
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() : string {
return rand(0, 1) ? "hello" : null;
}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
/**
* @return string|null
*/
function foo() {
return rand(0, 1) ? "hello" : null;
}
```
### InvalidFalsableReturnType
Running `vendor/bin/psalter --issues=InvalidFalsableReturnType` on
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() : string {
return rand(0, 1) ? "hello" : false;
}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
/**
* @return string|false
*/
function foo() {
return rand(0, 1) ? "hello" : false;
}
```
2019-06-04 14:45:01 +02:00
### MissingParamType
2019-06-04 14:50:50 +02:00
Running `vendor/bin/psalter --issues=MissingParamType` on
2019-06-04 14:45:01 +02:00
```php
<?php
2019-06-04 14:45:01 +02:00
class C {
public static function foo($s) : void {
echo $s;
}
}
C::foo("hello");
```
gives
```php
<?php
2019-06-04 14:45:01 +02:00
class C {
/**
* @param string $s
*/
public static function foo($s) : void {
echo $s;
}
}
C::foo("hello");
```
2020-07-30 18:11:28 +02:00
### MissingPropertyType
2020-07-29 05:56:32 +02:00
2020-07-30 18:11:28 +02:00
Running `vendor/bin/psalter --issues=MissingPropertyType` on
2020-07-29 05:56:32 +02:00
```php
<?php
class A {
public $foo;
public $bar;
2020-08-09 14:36:20 +02:00
public $baz;
2020-07-30 18:11:28 +02:00
2020-07-29 05:56:32 +02:00
public function __construct()
{
if (rand(0, 1)) {
$this->foo = 5;
} else {
$this->foo = "hello";
}
2020-07-30 18:11:28 +02:00
2020-07-29 05:56:32 +02:00
$this->bar = "baz";
}
2020-08-09 14:36:20 +02:00
public function setBaz() {
$this->baz = [1, 2, 3];
}
2020-07-29 05:56:32 +02:00
}
```
gives
```php
<?php
class A {
/**
* @var string|int
*/
public $foo;
public string $bar;
2020-07-30 18:11:28 +02:00
2020-08-09 14:36:20 +02:00
/**
* @var array<int, int>|null
* @psalm-var non-empty-list<int>|null
*/
public $baz;
2020-07-29 05:56:32 +02:00
public function __construct()
{
if (rand(0, 1)) {
$this->foo = 5;
} else {
$this->foo = "hello";
}
2020-07-30 18:11:28 +02:00
2020-07-29 05:56:32 +02:00
$this->bar = "baz";
}
2020-08-09 14:36:20 +02:00
public function setBaz() {
$this->baz = [1, 2, 3];
}
2020-07-29 05:56:32 +02:00
}
```
2018-02-27 23:25:35 +01:00
### MismatchingDocblockParamType
Given
```php
<?php
2018-02-27 23:25:35 +01:00
class A {}
class B extends A {}
class C extends A {}
class D {}
```
running `vendor/bin/psalter --issues=MismatchingDocblockParamType` on
```php
<?php
2018-02-27 23:25:35 +01:00
/**
* @param B|C $first
* @param D $second
*/
function foo(A $first, A $second) : void {}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
/**
* @param B|C $first
* @param A $second
*/
function foo(A $first, A $second) : void {}
```
### MismatchingDocblockReturnType
Running `vendor/bin/psalter --issues=MismatchingDocblockReturnType` on
```php
<?php
2018-02-27 23:25:35 +01:00
/**
* @return int
*/
function foo() : string {
return "hello";
}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
/**
* @return string
*/
function foo() : string {
return "hello";
}
```
### LessSpecificReturnType
Running `vendor/bin/psalter --issues=LessSpecificReturnType` on
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() : ?string {
return "hello";
}
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
function foo() : string {
return "hello";
}
```
### PossiblyUndefinedVariable
Running `vendor/bin/psalter --issues=PossiblyUndefinedVariable` on
```php
<?php
function foo()
{
if (rand(0, 1)) {
$a = 5;
}
echo $a;
}
```
gives
```php
<?php
function foo()
{
$a = null;
if (rand(0, 1)) {
$a = 5;
}
echo $a;
}
```
### PossiblyUndefinedGlobalVariable
Running `vendor/bin/psalter --issues=PossiblyUndefinedGlobalVariable` on
2018-02-27 23:25:35 +01:00
```php
<?php
2018-02-27 23:25:35 +01:00
if (rand(0, 1)) {
$a = 5;
}
echo $a;
```
gives
```php
<?php
2018-02-27 23:25:35 +01:00
$a = null;
if (rand(0, 1)) {
$a = 5;
}
echo $a;
```
### UnusedMethod
This removes private unused methods.
Running `vendor/bin/psalter --issues=UnusedMethod` on
```php
<?php
class A {
private function foo() : void {}
}
new A();
```
gives
```php
<?php
class A {
}
new A();
```
### PossiblyUnusedMethod
This removes protected/public unused methods.
Running `vendor/bin/psalter --issues=PossiblyUnusedMethod` on
```php
<?php
class A {
protected function foo() : void {}
public function bar() : void {}
}
new A();
```
gives
```php
<?php
class A {
}
new A();
```
### UnusedProperty
This removes private unused properties.
Running `vendor/bin/psalter --issues=UnusedProperty` on
```php
<?php
class A {
/** @var string */
private $foo;
}
new A();
```
gives
```php
<?php
class A {
}
new A();
```
### PossiblyUnusedProperty
This removes protected/public unused properties.
Running `vendor/bin/psalter --issues=PossiblyUnusedProperty` on
```php
<?php
class A {
/** @var string */
public $foo;
/** @var string */
protected $bar;
}
new A();
```
gives
```php
<?php
class A {
}
new A();
```
### UnusedVariable
This removes unused variables.
Running `vendor/bin/psalter --issues=UnusedVariable` on
```php
<?php
function foo(string $s) : void {
$a = 5;
$b = 6;
$c = $b += $a -= intval($s);
echo "foo";
}
```
gives
```php
<?php
function foo(string $s) : void {
echo "foo";
}
```
### UnnecessaryVarAnnotation
This removes unused `@var` annotations
Running `vendor/bin/psalter --issues=UnnecessaryVarAnnotation` on
```php
<?php
function foo() : string {
return "hello";
}
/** @var string */
$a = foo();
```
gives
```php
<?php
function foo() : string {
return "hello";
}
$a = foo();
```
### ParamNameMismatch
This aligns child class param names with their parent.
Running `vendor/bin/psalter --issues=ParamNameMismatch` on
```php
<?php
class A {
public function foo(string $str, bool $b = false) : void {}
}
class AChild extends A {
public function foo(string $string, bool $b = false) : void {
echo $string;
}
}
```
gives
```php
<?php
class A {
public function foo(string $str, bool $b = false) : void {}
}
class AChild extends A {
public function foo(string $str, bool $b = false) : void {
echo $str;
}
}
```