Slevomat Coding Standard for PHP_CodeSniffer extends Consistence Coding Standard by providing sniffs with additional checks.
- Unused private class properties and methods
- Trailing array comma
- Yoda conditions
- Alphabetically sorted uses
- Unused uses
- Other namespace-related sniffs
- Empty lines around opening and closing type braces
- Type name matches file name
- Installation
- Using the standard as a whole
- Using individual sniffs
- Fixing errors automatically
- Contributing
🔧 = Automatic errors fixing
Although PHP_CodeSniffer is not suitable for static analysis because it is limited to analysing one file at a time, it is possible to use it to perform certain checks. UnusedPrivateElementsSniff
checks for unused methods and unused or write-only properties in a class. Reported unused elements are safe to remove.
This is very useful during refactoring to clean up dead code and injected dependencies.
This sniff supports alwaysUsedPropertiesAnnotations
setting to mark certain properties as always used, for example the ones with @ORM\Column
annotations. Also, alwaysUsedPropertiesSuffixes
can be set to mark properties with name ending with a certain string to be always marked as used.
Commas after last element in an array make adding a new element easier and result in a cleaner versioning diff.
This sniff enforces trailing commas in multi-line arrays and requires short array syntax []
.
Yoda conditions decrease code comprehensibility and readability by switching operands around comparison operators forcing the reader to read the code in an unnatural way.
YodaComparisonSniff
looks for and fixes such comparisons not only in if
statements but in the whole code.
Checks whether uses at the top of a file are alphabetically sorted. Follows natural sorting and takes edge cases with special symbols into consideration. The following code snippet is an example of correctly sorted uses:
use Baz;
use Foo;
use Foo\Bar;
use Foo_Baz;
use Foo_bar;
use Foo1;
use Foo2;
use Foo11;
use Foo22;
use FooBaz;
use Foobar;
use Foobarz;
Looks for unused imports from other namespaces. Provides a property setting searchAnnotations
(default false
) that enables searching for mentions in annotations, which is especially useful for projects using Doctrine Annotations.
Enforces fully qualified type references after configurable set of language keywords.
For example with the following setting, extended or implemented type must always be referenced with a fully qualified name:
<rule ref="SlevomatCodingStandard.Namespaces.FullyQualifiedClassNameAfterKeyword">
<properties>
<property name="keywordsToCheck" type="array" value="T_EXTENDS,T_IMPLEMENTS"/>
</properties>
</rule>
This sniff reduces confusion in the following code snippet:
try {
$this->foo();
} catch (Exception $e) {
// Is this the general exception all exceptions must extend from? Or Exception from the current namespace?
}
All references to types named Exception
or ending with Exception
must be referenced via a fully qualified name:
try {
$this->foo();
} catch (\FooCurrentNamespace\Exception $e) {
} catch (\Exception $e) {
}
Exceptions with different names can be configured in specialExceptionNames
property.
Prohibits multiple uses separated by commas:
use Foo, Bar;
Enforces to use all referenced names with configurable omissions:
fullyQualifiedKeywords
- allows fully qualified names after certain keywords. Useful in tandem with FullyQualifiedClassNameAfterKeyword sniff.
allowFullyQualifiedExceptions
& specialExceptionNames
- allows fully qualified exceptions. Useful in tandem with FullyQualifiedExceptions sniff.
allowPartialUses
- allows using and referencing whole namespaces:
use Foo;
//...
new Foo\Bar();
namespacesRequiredToUse
- if not set, all namespaces are required to be used. When set, only mentioned namespaces are required to be used. Useful in tandem with UseOnlyWhitelistedNamespaces sniff.
Prohibits uses from the same namespace:
namespace Foo;
use Foo\Bar;
Disallows uses of other than configured namespaces.
namespacesRequiredToUse
- namespaces in this array are the only ones allowed to be used. E. g. root project namespace.
allowUseFromRootNamespace
- also allow using top-level namespace:
use DateTimeImmutable;
Disallows leading backslash in use statement:
use \Foo\Bar;
Enforces one empty line after opening class/interface/trait brace and one empty line before the closing brace.
For projects not following the PSR-0 or PSR-4 autoloading standards, this sniff checks whether a namespace and a name of a class/interface/trait follows agreed-on way to organize code into directories and files.
Other than enforcing that the type name must match the name of the file it's contained in, this sniff is very configurable. Consider the following sample configuration:
<rule ref="SlevomatCodingStandard.Files.TypeNameMatchesFileName">
<properties>
<property name="rootNamespaces" type="array" value="app/ui=>Slevomat\UI,app=>Slevomat,build/SlevomatSniffs/Sniffs=>SlevomatSniffs\Sniffs,tests/ui=>Slevomat\UI,tests=>Slevomat"/>
<property name="skipDirs" type="array" value="components,forms,model,models,services,stubs,data,new"/>
<property name="ignoredNamespaces" type="array" value="Slevomat\Services"/>
</properties>
</rule>
rootNamespaces
property expects configuration similar to PSR-4 - project directories mapped to certain namespaces.
skipDirs
are not taken into consideration when comparing a path to a namespace. For example, with the above settings, file at path app/services/Product/Product.php
is expected to contain Slevomat\Product\Product
, not Slevomat\services\Product\Product
.
Sniff is not performed on types in ignoredNamespaces
.
The recommended way to install Slevomat Coding Standard is through Composer.
{
"require-dev": {
"slevomat/coding-standard": "^1.0"
}
}
This package also installs jakub-onderka/php-parallel-lint which checks source code for syntax errors. Sniffs count on the processed code to be syntatically valid (no parse errors), otherwise they can behave unexpectedly. It is advised to run PHP-Parallel-Lint
in your build tool before running PHP_CodeSniffer
and exiting the build process early if PHP-Parallel-Lint
fails.
If you want to use the whole coding standard, besides requiring slevomat/coding-standard
in composer.json, require also Consistence Coding Standard:
{
"require-dev": {
"consistence/coding-standard": "^0.10"
}
}
Then mention both standards in ruleset.xml
:
<?xml version="1.0"?>
<ruleset name="AcmeProject">
<rule ref="vendor/consistence/coding-standard/Consistence/ruleset.xml" />
<rule ref="vendor/slevomat/coding-standard/SlevomatCodingStandard/ruleset.xml" />
<!-- additional settings -->
</ruleset>
To check your code base for violations, run PHP-Parallel-Lint
and PHP_CodeSniffer
from the command line:
vendor/bin/parallel-lint src tests
vendor/bin/phpcs --standard=ruleset.xml --extensions=php --encoding=utf-8 --tab-width=4 -sp src tests
If you don't want to follow the whole standard, but find a handful of included sniffs useful, you can use them selectively.
You can choose one of two ways to run only selected sniffs from the standard on your codebase:
Mention Slevomat Conding Standard in your project's ruleset.xml
:
<?xml version="1.0"?>
<ruleset name="AcmeProject">
<rule ref="vendor/slevomat/coding-standard/SlevomatCodingStandard/ruleset.xml" />
</ruleset>
When running phpcs
on the command line, use the --sniffs
option to list all the sniffs you want to use separated by a comma:
vendor/bin/phpcs --standard=ruleset.xml \
--sniffs=SlevomatCodingStandard.ControlStructures.YodaComparison,SlevomatCodingStandard.Namespaces.AlphabeticallySortedUses \
--extensions=php --encoding=utf-8 --tab-width=4 -sp src tests
Mention Slevomat Conding Standard in your project's ruleset.xml
and list all the excluded sniffs:
<?xml version="1.0"?>
<ruleset name="AcmeProject">
<rule ref="vendor/slevomat/coding-standard/SlevomatCodingStandard/ruleset.xml">
<exclude name="SlevomatCodingStandard.Namespaces.FullyQualifiedClassNameAfterKeyword"/>
<exclude name="SlevomatCodingStandard.Namespaces.FullyQualifiedExceptions"/>
<exclude name="SlevomatCodingStandard.Namespaces.ReferenceUsedNamesOnly"/>
<exclude name="SlevomatCodingStandard.Namespaces.UseOnlyWhitelistedNamespaces"/>
<exclude name="SlevomatCodingStandard.Types.EmptyLinesAroundTypeBraces"/>
<exclude name="SlevomatCodingStandard.Files.TypeNameMatchesFileName"/>
</rule>
</ruleset>
Then run the remaining sniffs in the usual way:
vendor/bin/phpcs --standard=ruleset.xml --extensions=php --encoding=utf-8 --tab-width=4 -sp src tests
Sniffs in this standard marked by the 🔧 symbol support automatic fixing of coding standard violations. To fix your code automatically, run phpcbf insteand of phpcs:
vendor/bin/phpcbf --standard=ruleset.xml --extensions=php --encoding=utf-8 --tab-width=4 -sp src tests
Always remember to back up your code before performing automatic fixes and check the results with your own eyes as the automatic fixer can sometimes produce unwanted results.
To make this repository work on your machine, clone it and run these two commands in the root directory of the repository:
composer install
vendor/bin/phing
After writing some code and editing or adding unit tests, run phing again to check that everything is OK:
vendor/bin/phing
We are always looking forward for your bugreports, feature requests and pull requests. Thank you.