Skip to content

πŸ“¦ Fast detection of composer dependency issues (unused dependencies, shadow dependencies, misplaced dependencies)

License

Notifications You must be signed in to change notification settings

reinfi/composer-dependency-analyser

Β 
Β 

Repository files navigation

Composer dependency analyser

  • πŸ’ͺ Powerful: Detects unused, shadow and misplaced composer dependencies
  • ⚑ Performant: Scans 15 000 files in 2s!
  • βš™οΈ Configurable: Fine-grained ignores via PHP config
  • πŸ•ΈοΈ Lightweight: No composer dependencies
  • 🍰 Easy-to-use: No config needed for first try
  • ✨ Compatible: PHP 7.2 - 8.3

Comparison:

Project Dead
dependency
Shadow
dependency
Misplaced
in require
Misplaced
in require-dev
Time*
maglnet/
composer-require-checker
❌ βœ… ❌ ❌ 124 secs
icanhazstring/
composer-unused
βœ… ❌ ❌ ❌ 72 secs
shipmonk/
composer-dependency-analyser
βœ… βœ… βœ… βœ… 2 secs

*Time measured on codebase with ~15 000 files

Installation:

composer require --dev shipmonk/composer-dependency-analyser

Note that this package itself has zero composer dependencies.

Usage:

vendor/bin/composer-dependency-analyser

Example output:

Found shadow dependencies!
(those are used, but not listed as dependency in composer.json)

  β€’ nette/utils
    e.g. Nette\Utils\Strings in app/Controller/ProductController.php:24 (+ 6 more)

Found unused dependencies!
(those are listed in composer.json, but no usage was found in scanned paths)

  β€’ nette/utils

(scanned 13970 files in 2.297 s)

Detected issues:

This tool reads your composer.json and scans all paths listed in autoload & autoload-dev sections while analysing:

Shadowed dependencies

  • Those are dependencies of your dependencies, which are not listed in composer.json
  • Your code can break when your direct dependency gets updated to newer version which does not require that shadowed dependency anymore
  • You should list all those packages within your dependencies

Unused dependencies

  • Any non-dev dependency is expected to have at least single usage within the scanned paths
  • To avoid false positives here, you might need to adjust scanned paths or ignore some packages by --config

Dev dependencies in production code

  • For libraries, this is risky as your users might not have those installed
  • For applications, it can break once you run it with composer install --no-dev
  • You should move those from require-dev to require

Prod dependencies used only in dev paths

  • For libraries, this miscategorization can lead to uselessly required dependencies for your users
  • You should move those from require to require-dev

Unknown classes

  • Any class that cannot be autoloaded gets reported as we cannot say if that one is shadowed or not

Cli options:

  • --composer-json path/to/composer.json for custom path to composer.json
  • --dump-usages symfony/console to show usages of certain package(s), * placeholder is supported
  • --config path/to/config.php for custom path to config file
  • --help display usage & cli options
  • --verbose to see more example classes & usages
  • --show-all-usages to see all usages
  • --ignore-unknown-classes to globally ignore unknown classes
  • --ignore-shadow-deps to globally ignore shadow dependencies
  • --ignore-unused-deps to globally ignore unused dependencies
  • --ignore-dev-in-prod-deps to globally ignore dev dependencies in prod code
  • --ignore-prod-only-in-dev-deps to globally ignore prod dependencies used only in dev paths

Configuration:

When a file named composer-dependency-analyser.php is located in cwd, it gets loaded automatically. The file must return ShipMonk\ComposerDependencyAnalyser\Config\Configuration object. You can use custom path and filename via --config cli option. Here is example of what you can do:

<?php

use ShipMonk\ComposerDependencyAnalyser\Config\Configuration;
use ShipMonk\ComposerDependencyAnalyser\Config\ErrorType;

$config = new Configuration();

return $config
    // disable scanning autoload & autoload-dev paths from composer.json
    // with such option, you should add custom paths by addPathToScan() or addPathsToScan()
    ->disableComposerAutoloadPathScan()

    // report unused dependencies even for dev packages
    // dev packages are often used only in CI, so this is not enabled by default
    // but you may want to ignore those packages manually to be sure
    ->enableAnalysisOfUnusedDevDependencies()

    // do not report ignores that never matched any error
    ->disableReportingUnmatchedIgnores()

    // globally disable specific error type
    ->ignoreErrors([ErrorType::DEV_DEPENDENCY_IN_PROD])

    // overwrite file extensions to scan, defaults to 'php'
    // applies only to directory scanning, not directly listed files
    ->setFileExtensions(['php'])

    // add extra path to scan
    // for multiple paths at once, use addPathsToScan()
    ->addPathToScan(__DIR__ . '/build', isDev: false)

    // exclude path from scanning
    // for multiple paths at once, use addPathsToExclude()
    ->addPathToExclude(__DIR__ . '/samples')

    // ignore errors on specific paths
    // this can be handy when DIC container file was passed as extra path, but you want to ignore shadow dependencies there
    // for multiple paths at once, use ignoreErrorsOnPaths()
    ->ignoreErrorsOnPath(__DIR__ . '/cache/DIC.php', [ErrorType::SHADOW_DEPENDENCY])

    // ignore errors on specific packages
    // you might have various reasons to ignore certain errors
    // e.g. polyfills are often used in libraries, but those are obviously unused when running with latest PHP
    // for multiple packages at once, use ignoreErrorsOnPackages()
    ->ignoreErrorsOnPackage('symfony/polyfill-php73', [ErrorType::UNUSED_DEPENDENCY])

    // ignore errors on specific packages and paths
    // for multiple, use ignoreErrorsOnPackagesAndPaths() or ignoreErrorsOnPackageAndPaths()
    ->ignoreErrorsOnPackageAndPath('symfony/console', __DIR__ . '/src/OptionalCommand.php', [ErrorType::SHADOW_DEPENDENCY])

    // allow using classes not present in composer's autoloader
    // e.g. a library may conditionally support some feature only when Memcached is available
    ->ignoreUnknownClasses(['Memcached'])

    // allow using classes not present in composer's autoloader by regex
    // e.g. when you want to ignore whole namespace of classes
    ->ignoreUnknownClassesRegex('~^PHPStan\\.*?~')

    // force certain classes to be treated as used
    // handy when dealing with dependencies in non-php files (e.g. DIC config), see example below
    // beware that those are not validated and do not even trigger unknown class error
    ->addForceUsedSymbols($classesExtractedFromNeonJsonYamlXmlEtc)
;

All paths are expected to exist. If you need some glob functionality, you can do it in your config file and pass the expanded list to e.g. ignoreErrorsOnPaths.

Detecting classes from non-php files:

Simplest fuzzy search for classnames within your yaml/neon/xml/json files might look like this:

$classNameRegex = '[a-zA-Z_\x80-\xff][a-zA-Z0-9_\x80-\xff]*'; // https://www.php.net/manual/en/language.oop5.basic.php
$dicFileContents = file_get_contents(__DIR__ . '/config/services.yaml');

preg_match_all(
    "~$classNameRegex(?:\\\\$classNameRegex)+~", // at least one backslash
    $dicFileContents,
    $matches
); // or parse the yaml properly

$config->addForceUsedSymbols($matches[1]); // possibly filter by class_exists || interface_exists

Similar approach should help you to avoid false positives in unused dependencies due to the usages being present in e.g. DIC config files only. Another approach for DIC-only usages is to scan the generated php file, but that gave us worse results.

Limitations:

  • Extension dependencies are not analysed (e.g. ext-json)
  • Files without namespace has limited support
    • Only classes with use statements and FQNs are detected
  • Function and constant usages are not analysed
    • Therefore, if some package contains only functions, it will be reported as unused

Despite those limitations, our experience is that this composer-dependency-analyser works much better than composer-unused and composer-require-checker.

Contributing:

  • Check your code by composer check
  • Autofix coding-style by composer fix:cs
  • All functionality must be tested

Supported PHP versions

  • Runtime requires PHP 7.2 - 8.3
  • Scanned codebase should use PHP >= 5.3
    • This can be done by pointing --composer-json to codebase located elsewhere

About

πŸ“¦ Fast detection of composer dependency issues (unused dependencies, shadow dependencies, misplaced dependencies)

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • PHP 100.0%