bin | ||
examples | ||
src/Psalm | ||
tests | ||
.gitignore | ||
composer.json | ||
composer.lock | ||
phpcs.xml | ||
psalm.xml | ||
PsalmLogo.png | ||
README.md |
- Introduction
- Installation
- Configuration
- Running Psalm
- Dealing with code issues
- Typing in Psalm
- Plugins
- Checking non-PHP files
Introduction
Psalm is a static analysis tool for finding errors in PHP applications, and runs in PHP 5.4+ and PHP 7.0.
While some tools (like PHP Codesniffer and PHP Mess Detector) are designed to make your code adhere to style guides and make code easier to maintain, Psalm is designed to find errors that may prevent your application from running properly.
It can discover over 50 different types of issues that could break your application in both obvious and subtle ways. For example, here's what happens when we run Psalm on a small code snippet with a simple bug:
// somefile.php
<?php
$a = ['foo', 'bar'];
echo implode($a, ' ');
> ./vendor/bin/psalm somefile.php
ERROR: InvalidArgument - somefile.php:3 - Argument 1 of implode expects `string`, `array` provided
Why Psalm?
There are two main inspirations for Psalm:
- Etsy's Phan, which uses nikic's
php-ast
extension to create an abstract syntax tree - Facebook's Hack, a PHP-like language that supports many advanced typing features natively, so docblocks aren't necessary.
Psalm's built-in function argument map is stolen wholesale from Phan, and its treatment of object-like arrays borrows heavily from Hack's.
So why should you use Psalm, and not those other tools? It comes down to coding style, and also your environment. If you have complete control over your stack, then you may well benefit from Hack's comprehensive typing support. If you're running on PHP7 and able to install custom extensions, Phan may be good for you.
Phan has one key drawback, as the php-ast
extension's generated tree is designed only to provide information that PHP uses at runtime. It therefore omits some docblock comments that provide useful hints to the developer, and also to Psalm.
Nikic's php-parser
AST generator, however, creates a more complete picture of your code, and Psalm uses that (PHP-native) package in its representation of your code.
That means you can use typehints like
/** @var string **/
$a = some_function();
and Psalm will treat $a
as a string.
Installation
Psalm Requires PHP >= 5.4 and Composer.
> composer require --dev "vimeo/psalm:dev-master"
> composer install
Configuration
Psalm uses an XML config file. A barebones example looks like this:
<?xml version="1.0"?>
<psalm name="Barebones config" stopOnFirstError="false" useDocblockTypes="true">
<inspectFiles>
<directory name="src" />
</inspectFiles>
</psalm>
and a more complete example (with recommended default values) can be found here.
Options
stopOnFirstError
whether or not to stop when the first error is encountereduseDocblockTypes
whether or not to use types as defined in docblocksautoloader
(optional) if your script that registers a custom autoloader and/or universal constants/functions, register them here
Parameters
<inspectFiles>
Contains a list of all the directories that Psalm should inspect<fileExtensions>
(optional)
A list of extensions to search over. See Checking non-PHP files to understand how to extend this.<plugins>
(optional)
A list of<plugin filename="path_to_plugin.php" />
entries. See the Plugins section for more information.<issueHandler>
(optional)
If you don't want Psalm to complain about every single issue it finds, the issueHandler tag allows you to configure that. Dealing with code issues tells you more.<includeHandler>
(optional)
If there are files that your scripts include that you don't want Psalm to traverse, include them here with<file name="path_to_file.php" />
.<mockClasses>
(optional)
Do you use mock classes in your tests? If you want Psalm to ignore them when checking files, include a fully-qualified path to the class with<class name="Your\Namespace\ClassName" />
Running Psalm
Once you've set up your config file, you can run Psalm from your project's root directory with
./vendor/bin/psalm
and Psalm will scan all files in the project referenced by <inspectFiles>
.
If you want to run on specific files, use
./vendor/bin/psalm file1.php [file2.php...]
Command-line options
--help
Display the list of help options--debug
With this flag, Psalm will list the files it's scanning, and provide a summary of memory usage--config
Path to a configuration file, if not ./psalm.xml--monochrome
Disables colored output--show-info=[BOOLEAN]
Show non-error parser findings.--diff
Only check files that have changed (and their dependents) since the last successful run--self-check
Make Psalm check itself (useful when making updates to Psalm)
Dealing with code issues
Code issues in Psalm fall into three categories:
- error
- this will cause Psalm to print a message, and to ultimately terminate with a non-zero exist status
- info
- this will cause Psalm to print a message
- suppress
- this will cause Psalm to ignore the code issue entirely
The third category, suppress
, is the one you will probably be most interested in, especially when introducing Psalm to a large codebase.
Suppressing issues
There are two ways to suppress an issue – via the Psalm config or via a function docblock.
Config suppression
You can use the <issueHandler>
tag in the config file to influence how issues are treated.
<issueHandler>
<MissingPropertyType errorLevel="suppress" />
<InvalidReturnType>
<excludeFiles>
<directory name="some_bad_directory" /> <!-- all InvalidReturnType issues in this directory are suppressed -->
<file name="some_bad_file.php" /> <!-- all InvalidReturnType issues in this file are suppressed -->
</excludeFiles>
</InvalidReturnType>
</issueHandler>
Docblock suppression
You can also use @psalm-suppress IssueName
on a function's docblock to suppress Psalm issues e.g.
/**
* @psalm-suppress InvalidReturnType
*/
function (int $a) : string {
return $a;
}
Typing in Psalm
Psalm is able to interpret all PHPDoc type annotations, and use them to further understand the codebase.
Union Types
PHP and other dynamically-typed languages allow expressions to resolved to conflicting types – for example, after this statement
$rabbit = rand(0, 10) === 4 ? 'rabbit' : ['rabbit'];
$rabbit
will be either a string
or an array
. We can represent that idea with Union Types – so $rabbit
is typed as string|array
. Union types represent all the possible types a given variable can have.
Use of false
in Union Types
This also extends to builtin PHP methods, many of which can return false
to denote some sort of failure. For example, strpos
has the return type int|false
. This is a more specific version of int|bool
, and allows us to evaluate logic like
function str_index_of(string $haystack, string $needle) : int {
$pos = strpos($haystack, $needle);
if ($pos === false) {
return -1;
}
return $pos;
}
and verify that str_index_of
always returns an integer. If we instead typed the return of strpos
as int|bool
, then according to Psalm the last statement return $pos
could return either an integer or true
(the solution would be to turn if ($pos === false)
into if (is_bool($pos))
.
Property declaration types vs Assignment typehints
You can use the /** @var Type */
docblock to annotate both property declarations and to help Psalm understand variable assignment.
Property declaration types
You can specify a particular type for a class property declarion in Psalm by using the @var
declaration:
/** @var string|null */
public $foo;
When checking $this->foo = $some_variable;
, Psalm will check to see whether $some_variable
is either string
or null
and, if neither, emit an issue.
If you leave off the property type docblock, Psalm will emit a MissingPropertyType
issue.
Assignment typehints
Consider the following code:
$a = null;
foreach ([1, 2, 3] as $i) {
if ($a) {
return $a;
}
else {
$a = $i;
}
}
Because Psalm scans a file progressively, it cannot tell that return $a
produces an integer. Instead it knows only that $a
is not empty
. We can fix this by adding a type hint docblock:
/** @var int|null */
$a = null;
foreach ([1, 2, 3] as $i) {
if ($a) {
return $a;
}
else {
$a = $i;
}
}
This tells Psalm that int
is a possible type for $a
, and allows it to infer that return $a;
produces an integer.
Unlike property types, however, assignment typehints are not binding – they can be overridden by a new assignment without Psalm emitting an issue e.g.
/** @var string|null */
$a = foo();
$a = 6; // $a is now typed as an int
You can also use typehints on specific variables e.g.
/** @var string $a */
echo strpos($a, 'hello');
This tells Psalm to assume that $a
is a string (though it will still throw an error if $a
is undefined).
Typing arrays
In PHP, the array
type is commonly used to represent three different data structures:
-
a List
$a = [1, 2, 3, 4, 5];
-
$a = [0 => 'hello', 5 => 'goodbye']; $b = ['a' => 'AA', 'b' => 'BB', 'c' => 'CC']
-
makeshift Structs
$a = ['name' => 'Psalm', 'type' => 'tool'];
PHP treats all these arrays the same, essentially (though there are some optimisations under the hood for the first case).
PHPDoc allows you to specify the type of values the array holds with the annotation:
/** @return TValue[] */
where TValue
is a union type, but it does not allow you to specify the type of keys.
Psalm uses a syntax borrowed from Java to denote the types of both keys and values:
/** @return array<TKey, TValue> */
Makeshift Structs
Ideally (in the author's opinion), all data would either be encoded as lists, associative arrays, or as well-defined objects. However, PHP arrays are often used as makeshift structs.
Hack supports this usage by way of the Shape datastructure, but there is no agreed-upon documentation format for such arrays in regular PHP-land.
Psalm solves this by adding another way annotate array types, by using an object-like syntax when describing them.
So, for instance, the method below returns an array of arrays, both of which have the same keys:
/** @return array<int, array<string, string|bool>> */
function getToolsData() : array {
return [
['name' => 'Psalm', 'type' => 'tool', 'active' => true],
['name' => 'PhpParser', 'type' => 'tool', 'active' => true]
];
}
Using the type annotation for associative arrays, we could evaluate the expression
getToolsData()[0]['name']
and Psalm would know that it was had the type string|bool
.
However, we can provide a more-specific return type by using a brace annotation:
/** @return array<int, array{name: string, type: string, active: bool}> */
function getToolsData() : array {
return [
['name' => 'Psalm', 'type' => 'tool', 'active' => true],
['name' => 'PhpParser', 'type' => 'tool', 'active' => true]
];
}
This time, Psalm can evaluate getToolsData()[0]['name']
and it knows that the expression evaluates to a string.
Backwards compatibility
Psalm fully supports PHPDoc's array typing syntax, such that any array typed with TValue[]
will be typed in Psalm as array<mixed, TValue>
. That also extends to generic type definitions with only one param e.g. array<TValue>
, which is equivalent to array<mixed, TValue>
.
Plugins
@todo add this
Checking non-PHP files
Psalm supports the ability to check various PHPish files by extending the FileChecker
class. For example, if you have a template where the variables are set elsewhere, Psalm can scrape those variables and check the template with those variables pre-populated.
An example TemplateChecker is provided here.
To ensure your custom FileChecker
is used, you must update the Psalm fileExtensions
config in psalm.xml:
<fileExtensions>
<extension name=".php" />
<extension name=".phpt" filetypeHandler="path/to/TemplateChecker.php" />
</fileExtensions>