ncac / php-cognitive-complexity
CLI tool for measuring PHP cognitive complexity (ISO SonarQube) — integrates with CI/CD pipelines and Husky pre-commit hooks
Requires
- php: ^8.2
- nikic/php-parser: ^5.0
- revolt/event-loop: >=1.0.0 <1.0.7
- symfony/console: ^6.0 || ^7.0
- symfony/finder: ^6.0 || ^7.0
- symfony/yaml: ^6.0 || ^7.0
Requires (Dev)
- friendsofphp/php-cs-fixer: 3.64.*
- ncac/phpcs-standard: ^4.1
- phing/phing: ^3.1
- phpunit/phpunit: ^10.0 || ^11.0
- squizlabs/php_codesniffer: ^3.7 || ^4.0
- vimeo/psalm: ^6.0
This package is auto-updated.
Last update: 2026-05-12 19:17:31 UTC
README
A modern PHP CLI tool for measuring cognitive complexity of PHP code — ISO SonarQube (G. Ann Campbell's specification).
Two commands under one binary: check for CI/hooks (strict gate, exit 1 on violation), analyse for interactive exploration (coloured output, severity levels, always exit 0).
Why cognitive complexity?
Cyclomatic complexity counts paths. Cognitive complexity measures how hard code is to read. It penalises deep nesting and rewards early returns — much closer to human perception of code quality.
This tool implements the SonarSource specification for PHP.
Installation
composer require --dev ncac/php-cognitive-complexity
Usage
check — CI gate
Strict mode: exits 1 on any violation. Designed for pre-commit hooks and CI pipelines.
# Basic check vendor/bin/cognitive-complexity check src/ --max=15 # With a config file vendor/bin/cognitive-complexity check src/ --config=cognitive.yaml # Only git-modified files (pre-commit) vendor/bin/cognitive-complexity check src/ --max=15 --diff # JSON output vendor/bin/cognitive-complexity check src/ --format=json # GitLab Code Quality artifact vendor/bin/cognitive-complexity check src/ --format=gitlab > gl-code-quality-report.json
analyse — interactive exploration
Developer-experience mode: coloured output grouped by file, severity labels, always exits 0.
# Show only violations (default) vendor/bin/cognitive-complexity analyse src/ # Show all functions including those below threshold vendor/bin/cognitive-complexity analyse src/ --all # Sort by score descending (prioritise refactoring targets) vendor/bin/cognitive-complexity analyse src/ --sort=score # Custom threshold vendor/bin/cognitive-complexity analyse src/ --max=20 # Drupal/multi-extension projects vendor/bin/cognitive-complexity analyse web/ --ext=php,module,inc,theme,install
Example output:
── src/Service/OrderService.php
[ 18] processOrder:42 — minor
[ 4] validateItems:91
[ 0] formatResult:112
── src/Gateway/PaymentGateway.php
[ 32] validate:87 — moderate
✗ 2 violation(s) / 4 function(s) in 2 file(s) — threshold=15
Severity levels (multiples of threshold, default 15):
| Level | Score range |
|---|---|
minor |
threshold+1 → 2×threshold |
moderate |
2×threshold+1 → 3×threshold |
high |
3×threshold+1 → 50 |
critical |
> 50 |
Generate a baseline (ignore pre-existing violations)
vendor/bin/cognitive-complexity baseline src/ --output=baseline.json vendor/bin/cognitive-complexity check src/ --baseline=baseline.json
Configuration file
Create a cognitive.yaml at the root of your project:
# Default threshold max_complexity: 15 # Per-path overrides paths: src/Controller/: 10 src/Service/: 12 tests/: 20 # Excluded paths exclude: - vendor/ - cache/ - legacy/
Console output example
✗ src/Service/OrderService.php::processOrder → 18 (max: 15) [line 42]
✗ src/Gateway/PaymentGateway.php::validate → 16 (max: 15) [line 87]
2 violation(s) found. Cognitive complexity threshold exceeded.
This is the output of check. For richer output, use analyse.
Integration in your project
Add convenience scripts to your composer.json for fixed targets:
{
"scripts": {
"cc": "cognitive-complexity check src/ --max=15",
"cc:analyse": "cognitive-complexity analyse src/ --all --sort=score",
"cc:baseline": "cognitive-complexity baseline src/ --max=15 --output=cognitive-baseline.json"
}
}
composer cc # CI gate — exit 1 on violations composer cc:analyse # interactive report sorted by score composer cc:baseline # regenerate the baseline after a refactoring
For ad-hoc analysis on a specific path, call cc — the short wrapper installed alongside the main binary:
# cc defaults to 'check' when no sub-command is given vendor/bin/cc src/ vendor/bin/cc src/ --max=20 vendor/bin/cc src/ --diff # pre-commit: git-modified files only # Explicit sub-commands vendor/bin/cc analyse web/modules/custom/ --all vendor/bin/cc analyse web/modules/custom/ --sort=score --ext=php,module,inc vendor/bin/cc baseline src/ --output=cognitive-baseline.json
To ignore pre-existing violations during onboarding, generate a baseline first:
composer cc:baseline # then add --baseline to your cc script: # "cc": "cognitive-complexity check src/ --max=15 --baseline=cognitive-baseline.json"
# .husky/pre-commit
vendor/bin/cognitive-complexity check src/ --max=15
GitLab CI/CD integration
# .gitlab-ci.yml cognitive_complexity: stage: quality script: - php vendor/bin/cognitive-complexity check src/ --format=gitlab > gl-code-quality-report.json artifacts: reports: codequality: gl-code-quality-report.json
Algorithm
Based on G. Ann Campbell's Cognitive Complexity specification:
| Structure | Increment |
|---|---|
if, else if, else |
+1 |
for, foreach, while, do-while |
+1 |
switch |
+1 |
catch |
+1 |
Ternary ?: |
+1 |
&&, || (logical sequence changes) |
+1 |
| Each nested level | +level bonus |
Development
git clone https://github.com/ncac/php-cognitive-complexity.git cd php-cognitive-complexity composer install # Run tests composer test # Run all checks composer check # Coverage composer test-coverage