From b6553928207cfecb704e25c85d0117f8c1ecf3eb Mon Sep 17 00:00:00 2001 From: Jan Nedbal Date: Thu, 31 Oct 2024 12:53:08 +0100 Subject: [PATCH] Analyse even php extension dependencies (`ext-*`) (#118) --- .github/workflows/e2e.yml | 12 +- README.md | 8 +- composer-dependency-analyser.php | 8 +- src/Analyser.php | 217 +++++++++++++----- src/Cli.php | 5 + src/CliOptions.php | 5 + src/ComposerJson.php | 45 +++- src/Config/Configuration.php | 146 +++++++++++- src/Config/Ignore/IgnoreList.php | 44 ++-- src/Initializer.php | 7 + src/UsedSymbolExtractor.php | 144 +++++++++--- tests/AnalyserTest.php | 73 ++++++ tests/BinTest.php | 2 +- tests/CliTest.php | 3 +- tests/ConfigurationTest.php | 8 + tests/InitializerTest.php | 1 + tests/UsedSymbolExtractorTest.php | 85 ++++++- .../extensions/ext-dev-usages.php | 4 + .../extensions/ext-prod-usages.php | 6 + .../used-symbols/extensions-global.php | 22 ++ .../used-symbols/extensions.php | 22 ++ .../used-symbols/t-string-issues.php | 24 ++ 22 files changed, 760 insertions(+), 131 deletions(-) create mode 100644 tests/data/not-autoloaded/extensions/ext-dev-usages.php create mode 100644 tests/data/not-autoloaded/extensions/ext-prod-usages.php create mode 100644 tests/data/not-autoloaded/used-symbols/extensions-global.php create mode 100644 tests/data/not-autoloaded/used-symbols/extensions.php create mode 100644 tests/data/not-autoloaded/used-symbols/t-string-issues.php diff --git a/.github/workflows/e2e.yml b/.github/workflows/e2e.yml index 3758f12..696102a 100644 --- a/.github/workflows/e2e.yml +++ b/.github/workflows/e2e.yml @@ -119,6 +119,11 @@ jobs: uses: shivammathur/setup-php@v2 with: php-version: 8.3 + ini-file: development + + - + name: List enabled extensions + run: php -m - name: Install analyser dependencies @@ -136,7 +141,12 @@ jobs: working-directory: ${{ matrix.repo }} run: composer install --no-progress --no-interaction --ignore-platform-reqs ${{ matrix.composerArgs }} + - + name: Run analyser (--disable-ext-analysis) + working-directory: ${{ matrix.repo }} + run: php ../../analyser/bin/composer-dependency-analyser --show-all-usages --disable-ext-analysis ${{ matrix.cdaArgs }} + - name: Run analyser working-directory: ${{ matrix.repo }} - run: php ../../analyser/bin/composer-dependency-analyser ${{ matrix.cdaArgs }} + run: php ../../analyser/bin/composer-dependency-analyser --show-all-usages ${{ matrix.cdaArgs }} diff --git a/README.md b/README.md index 67eea35..3f19236 100644 --- a/README.md +++ b/README.md @@ -49,7 +49,7 @@ Found unused dependencies! ``` ## Detected issues: -This tool reads your `composer.json` and scans all paths listed in `autoload` & `autoload-dev` sections while analysing: +This tool reads your `composer.json` and scans all paths listed in `autoload` & `autoload-dev` sections while analysing you dependencies (both **packages and PHP extensions**). ### Shadowed dependencies - Those are dependencies of your dependencies, which are not listed in `composer.json` @@ -84,6 +84,7 @@ This tool reads your `composer.json` and scans all paths listed in `autoload` & - `--verbose` to see more example classes & usages - `--show-all-usages` to see all usages - `--format` to use different output format, available are: `console` (default), `junit` +- `--disable-ext-analysis` to disable php extensions analysis (e.g. `ext-xml`) - `--ignore-unknown-classes` to globally ignore unknown classes - `--ignore-unknown-functions` to globally ignore unknown functions - `--ignore-shadow-deps` to globally ignore shadow dependencies @@ -128,6 +129,7 @@ return $config //// Adjust analysis ->enableAnalysisOfUnusedDevDependencies() // dev packages are often used only in CI, so this is not enabled by default ->disableReportingUnmatchedIgnores() // do not report ignores that never matched any error + ->disableExtensionsAnalysis() // do not analyse ext-* dependencies //// Use symbols from yaml/xml/neon files // - designed for DIC config files (see below) @@ -166,8 +168,8 @@ Another approach for DIC-only usages is to scan the generated php file, but that NO_COLOR=1 vendor/bin/composer-dependency-analyser ``` -## Limitations: -- Extension dependencies are not analysed (e.g. `ext-json`) +## Recommendations: +- For precise `ext-*` analysis, your enabled extensions of your php runtime should be superset of those used in the scanned project ## Contributing: - Check your code by `composer check` diff --git a/composer-dependency-analyser.php b/composer-dependency-analyser.php index bb1ce73..108ca2f 100644 --- a/composer-dependency-analyser.php +++ b/composer-dependency-analyser.php @@ -1,8 +1,14 @@ addPathToScan(__FILE__, true) ->addPathToScan(__DIR__ . '/bin', false) - ->addPathToExclude(__DIR__ . '/tests/data'); + ->addPathToExclude(__DIR__ . '/tests/data') + ->ignoreErrorsOnExtensionsAndPaths( + ['ext-dom', 'ext-libxml'], + [__DIR__ . '/src/Result/JunitFormatter.php'], // optional usages guarded with extension_loaded() + [ErrorType::DEV_DEPENDENCY_IN_PROD] + ); diff --git a/src/Analyser.php b/src/Analyser.php index 2c984bd..d77416c 100644 --- a/src/Analyser.php +++ b/src/Analyser.php @@ -45,6 +45,24 @@ class Analyser { + /** + * Those are core PHP extensions, that can never be disabled + * There are more PHP "core" extensions, that are bundled by default, but PHP can be compiled without them + * You can check which are added conditionally in https://github.com/php/php-src/tree/master/ext (see config.w32 files) + */ + private const CORE_EXTENSIONS = [ + 'ext-core', + 'ext-date', + 'ext-json', + 'ext-hash', + 'ext-pcre', + 'ext-phar', + 'ext-reflection', + 'ext-spl', + 'ext-random', + 'ext-standard', + ]; + /** * @var Stopwatch */ @@ -73,7 +91,7 @@ class Analyser private $classmap = []; /** - * package name => is dev dependency + * package or ext-* => is dev dependency * * @var array */ @@ -87,15 +105,29 @@ class Analyser private $ignoredSymbols; /** - * function name => path + * custom function name => path * * @var array */ private $definedFunctions = []; + /** + * kind => [symbol name => ext-*] + * + * @var array> + */ + private $extensionSymbols = []; + + /** + * lowercase symbol name => kind + * + * @var array + */ + private $extensionSymbolKinds = []; + /** * @param array $classLoaders vendorDir => ClassLoader (e.g. result of \Composer\Autoload\ClassLoader::getRegisteredLoaders()) - * @param array $composerJsonDependencies package name => is dev dependency + * @param array $composerJsonDependencies package or ext-* => is dev dependency */ public function __construct( Stopwatch $stopwatch, @@ -107,11 +139,11 @@ public function __construct( { $this->stopwatch = $stopwatch; $this->config = $config; - $this->composerJsonDependencies = $composerJsonDependencies; + $this->composerJsonDependencies = $this->filterDependencies($composerJsonDependencies, $config); $this->vendorDirs = array_keys($classLoaders + [$defaultVendorDir => null]); $this->classLoaders = array_values($classLoaders); - $this->initExistingSymbols(); + $this->initExistingSymbols($config); } /** @@ -129,8 +161,8 @@ public function run(): AnalysisResult $prodOnlyInDevErrors = []; $unusedErrors = []; - $usedPackages = []; - $prodPackagesUsedInProdPath = []; + $usedDependencies = []; + $prodDependenciesUsedInProdPath = []; $usages = []; @@ -143,66 +175,71 @@ public function run(): AnalysisResult foreach ($usedSymbolsByKind as $kind => $usedSymbols) { foreach ($usedSymbols as $usedSymbol => $lineNumbers) { - $usedSymbolNameForIgnoreCheck = $kind === SymbolKind::FUNCTION ? strtolower($usedSymbol) : $usedSymbol; + $normalizedUsedSymbolName = $kind === SymbolKind::FUNCTION ? strtolower($usedSymbol) : $usedSymbol; - if (isset($this->ignoredSymbols[$usedSymbolNameForIgnoreCheck])) { + if (isset($this->ignoredSymbols[$normalizedUsedSymbolName])) { continue; } - $symbolPath = $this->getSymbolPath($usedSymbol, $kind); + if (isset($this->extensionSymbols[$kind][$normalizedUsedSymbolName])) { + $dependencyName = $this->extensionSymbols[$kind][$normalizedUsedSymbolName]; + + } else { + $symbolPath = $this->getSymbolPath($usedSymbol, $kind); - if ($symbolPath === null) { - if ($kind === SymbolKind::CLASSLIKE && !$ignoreList->shouldIgnoreUnknownClass($usedSymbol, $filePath)) { - foreach ($lineNumbers as $lineNumber) { - $unknownClassErrors[$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); + if ($symbolPath === null) { + if ($kind === SymbolKind::CLASSLIKE && !$ignoreList->shouldIgnoreUnknownClass($usedSymbol, $filePath)) { + foreach ($lineNumbers as $lineNumber) { + $unknownClassErrors[$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); + } } - } - if ($kind === SymbolKind::FUNCTION && !$ignoreList->shouldIgnoreUnknownFunction($usedSymbol, $filePath)) { - foreach ($lineNumbers as $lineNumber) { - $unknownFunctionErrors[$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); + if ($kind === SymbolKind::FUNCTION && !$ignoreList->shouldIgnoreUnknownFunction($usedSymbol, $filePath)) { + foreach ($lineNumbers as $lineNumber) { + $unknownFunctionErrors[$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); + } } + + continue; } - continue; - } + if (!$this->isVendorPath($symbolPath)) { + continue; // local class + } - if (!$this->isVendorPath($symbolPath)) { - continue; // local class + $dependencyName = $this->getPackageNameFromVendorPath($symbolPath); } - $packageName = $this->getPackageNameFromVendorPath($symbolPath); - if ( - $this->isShadowDependency($packageName) - && !$ignoreList->shouldIgnoreError(ErrorType::SHADOW_DEPENDENCY, $filePath, $packageName) + $this->isShadowDependency($dependencyName) + && !$ignoreList->shouldIgnoreError(ErrorType::SHADOW_DEPENDENCY, $filePath, $dependencyName) ) { foreach ($lineNumbers as $lineNumber) { - $shadowErrors[$packageName][$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); + $shadowErrors[$dependencyName][$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); } } if ( !$isDevFilePath - && $this->isDevDependency($packageName) - && !$ignoreList->shouldIgnoreError(ErrorType::DEV_DEPENDENCY_IN_PROD, $filePath, $packageName) + && $this->isDevDependency($dependencyName) + && !$ignoreList->shouldIgnoreError(ErrorType::DEV_DEPENDENCY_IN_PROD, $filePath, $dependencyName) ) { foreach ($lineNumbers as $lineNumber) { - $devInProdErrors[$packageName][$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); + $devInProdErrors[$dependencyName][$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); } } if ( !$isDevFilePath - && !$this->isDevDependency($packageName) + && !$this->isDevDependency($dependencyName) ) { - $prodPackagesUsedInProdPath[$packageName] = true; + $prodDependenciesUsedInProdPath[$dependencyName] = true; } - $usedPackages[$packageName] = true; + $usedDependencies[$dependencyName] = true; foreach ($lineNumbers as $lineNumber) { - $usages[$packageName][$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); + $usages[$dependencyName][$usedSymbol][] = new SymbolUsage($filePath, $lineNumber, $kind); } } } @@ -215,19 +252,31 @@ public function run(): AnalysisResult continue; } - $symbolPath = $this->getSymbolPath($forceUsedSymbol, null); + if ( + isset($this->extensionSymbols[SymbolKind::FUNCTION][$forceUsedSymbol]) + || isset($this->extensionSymbols[SymbolKind::CONSTANT][$forceUsedSymbol]) + || isset($this->extensionSymbols[SymbolKind::CLASSLIKE][$forceUsedSymbol]) + ) { + $forceUsedDependency = $this->extensionSymbols[SymbolKind::FUNCTION][$forceUsedSymbol] + ?? $this->extensionSymbols[SymbolKind::CONSTANT][$forceUsedSymbol] + ?? $this->extensionSymbols[SymbolKind::CLASSLIKE][$forceUsedSymbol]; + } else { + $symbolPath = $this->getSymbolPath($forceUsedSymbol, null); + + if ($symbolPath === null || !$this->isVendorPath($symbolPath)) { + continue; + } - if ($symbolPath === null || !$this->isVendorPath($symbolPath)) { - continue; + $forceUsedDependency = $this->getPackageNameFromVendorPath($symbolPath); } - $forceUsedPackage = $this->getPackageNameFromVendorPath($symbolPath); - $usedPackages[$forceUsedPackage] = true; - $forceUsedPackages[$forceUsedPackage] = true; + $usedDependencies[$forceUsedDependency] = true; + $forceUsedPackages[$forceUsedDependency] = true; } if ($this->config->shouldReportUnusedDevDependencies()) { $dependenciesForUnusedAnalysis = array_keys($this->composerJsonDependencies); + } else { $dependenciesForUnusedAnalysis = array_keys(array_filter($this->composerJsonDependencies, static function (bool $devDependency) { return !$devDependency; // dev deps are typically used only in CI @@ -236,7 +285,8 @@ public function run(): AnalysisResult $unusedDependencies = array_diff( $dependenciesForUnusedAnalysis, - array_keys($usedPackages) + array_keys($usedDependencies), + self::CORE_EXTENSIONS ); foreach ($unusedDependencies as $unusedDependency) { @@ -250,9 +300,10 @@ public function run(): AnalysisResult })); $prodPackagesUsedOnlyInDev = array_diff( $prodDependencies, - array_keys($prodPackagesUsedInProdPath), + array_keys($prodDependenciesUsedInProdPath), array_keys($forceUsedPackages), // we dont know where are those used, lets not report them - $unusedDependencies + $unusedDependencies, + self::CORE_EXTENSIONS ); foreach ($prodPackagesUsedOnlyInDev as $prodPackageUsedOnlyInDev) { @@ -340,7 +391,9 @@ private function getUsedSymbolsInFile(string $filePath): array throw new InvalidPathException("Unable to get contents of '$filePath'"); } - return (new UsedSymbolExtractor($code))->parseUsedSymbols(); + return (new UsedSymbolExtractor($code))->parseUsedSymbols( + $this->extensionSymbolKinds + ); } /** @@ -450,7 +503,7 @@ private function normalizePath(string $filePath): string return Path::normalize($filePath); } - private function initExistingSymbols(): void + private function initExistingSymbols(Configuration $config): void { $this->ignoredSymbols = [ // built-in types @@ -485,9 +538,25 @@ private function initExistingSymbols(): void 'Composer\\Autoload\\ClassLoader' => true, ]; - /** @var string $constantName */ - foreach (get_defined_constants() as $constantName => $constantValue) { - $this->ignoredSymbols[$constantName] = true; + /** @var array> $definedConstants */ + $definedConstants = get_defined_constants(true); + + foreach ($definedConstants as $constantExtension => $constants) { + foreach ($constants as $constantName => $_) { + if ($constantExtension === 'user' || !$config->shouldAnalyseExtensions()) { + $this->ignoredSymbols[$constantName] = true; + + } else { + $extensionName = $this->getNormalizedExtensionName($constantExtension); + + if (in_array($extensionName, self::CORE_EXTENSIONS, true)) { + $this->ignoredSymbols[$constantName] = true; + } else { + $this->extensionSymbols[SymbolKind::CONSTANT][$constantName] = $extensionName; + $this->extensionSymbolKinds[strtolower($constantName)] = SymbolKind::CONSTANT; + } + } + } } foreach (get_defined_functions() as $functionNames) { @@ -495,10 +564,19 @@ private function initExistingSymbols(): void $reflectionFunction = new ReflectionFunction($functionName); $functionFilePath = $reflectionFunction->getFileName(); - if ($reflectionFunction->getExtension() === null && is_string($functionFilePath)) { - $this->definedFunctions[$functionName] = Path::normalize($functionFilePath); + if ($reflectionFunction->getExtension() === null) { + if (is_string($functionFilePath)) { + $this->definedFunctions[$functionName] = Path::normalize($functionFilePath); + } } else { - $this->ignoredSymbols[$functionName] = true; + $extensionName = $this->getNormalizedExtensionName($reflectionFunction->getExtension()->name); + + if (in_array($extensionName, self::CORE_EXTENSIONS, true) || !$config->shouldAnalyseExtensions()) { + $this->ignoredSymbols[$functionName] = true; + } else { + $this->extensionSymbols[SymbolKind::FUNCTION][$functionName] = $extensionName; + $this->extensionSymbolKinds[$functionName] = SymbolKind::FUNCTION; + } } } } @@ -511,11 +589,44 @@ private function initExistingSymbols(): void foreach ($classLikes as $classLikeNames) { foreach ($classLikeNames as $classLikeName) { - if ((new ReflectionClass($classLikeName))->getExtension() !== null) { - $this->ignoredSymbols[$classLikeName] = true; + $classReflection = new ReflectionClass($classLikeName); + + if ($classReflection->getExtension() !== null) { + $extensionName = $this->getNormalizedExtensionName($classReflection->getExtension()->name); + + if (in_array($extensionName, self::CORE_EXTENSIONS, true) || !$config->shouldAnalyseExtensions()) { + $this->ignoredSymbols[$classLikeName] = true; + } else { + $this->extensionSymbols[SymbolKind::CLASSLIKE][$classLikeName] = $extensionName; + $this->extensionSymbolKinds[strtolower($classLikeName)] = SymbolKind::CLASSLIKE; + } } } } } + private function getNormalizedExtensionName(string $extension): string + { + return 'ext-' . ComposerJson::normalizeExtensionName($extension); + } + + /** + * @param array $dependencies + * @return array + */ + private function filterDependencies(array $dependencies, Configuration $config): array + { + $filtered = []; + + foreach ($dependencies as $dependency => $isDevDependency) { + if (!$config->shouldAnalyseExtensions() && strpos($dependency, 'ext-') === 0) { + continue; + } + + $filtered[$dependency] = $isDevDependency; + } + + return $filtered; + } + } diff --git a/src/Cli.php b/src/Cli.php index 64df045..ceb7f26 100644 --- a/src/Cli.php +++ b/src/Cli.php @@ -20,6 +20,7 @@ class Cli 'version' => false, 'help' => false, 'verbose' => false, + 'disable-ext-analysis' => false, 'ignore-shadow-deps' => false, 'ignore-unused-deps' => false, 'ignore-dev-in-prod-deps' => false, @@ -153,6 +154,10 @@ public function getProvidedOptions(): CliOptions $options->verbose = true; } + if (isset($this->providedOptions['disable-ext-analysis'])) { + $options->disableExtAnalysis = true; + } + if (isset($this->providedOptions['ignore-shadow-deps'])) { $options->ignoreShadowDeps = true; } diff --git a/src/CliOptions.php b/src/CliOptions.php index 3656225..3eeaf15 100644 --- a/src/CliOptions.php +++ b/src/CliOptions.php @@ -20,6 +20,11 @@ class CliOptions */ public $verbose = null; + /** + * @var true|null + */ + public $disableExtAnalysis = null; + /** * @var true|null */ diff --git a/src/ComposerJson.php b/src/ComposerJson.php index 3631304..a351499 100644 --- a/src/ComposerJson.php +++ b/src/ComposerJson.php @@ -23,6 +23,7 @@ use function realpath; use function str_replace; use function strpos; +use function strtolower; use function strtr; use function trim; use const ARRAY_FILTER_USE_KEY; @@ -44,7 +45,7 @@ class ComposerJson public $composerAutoloadPath; /** - * Package => isDev + * Package or ext-* => isDev * * @readonly * @var array @@ -99,18 +100,52 @@ public function __construct( $this->extractAutoloadExcludeRegexes($basePath, $composerJsonData['autoload-dev']['exclude-from-classmap'] ?? [], true) ); + $filterExtensions = static function (string $dependency): bool { + return strpos($dependency, 'ext-') === 0; + }; $filterPackages = static function (string $package): bool { return strpos($package, '/') !== false; }; - $this->dependencies = array_merge( + $this->dependencies = $this->normalizeNames(array_merge( array_fill_keys(array_keys(array_filter($requiredPackages, $filterPackages, ARRAY_FILTER_USE_KEY)), false), - array_fill_keys(array_keys(array_filter($requiredDevPackages, $filterPackages, ARRAY_FILTER_USE_KEY)), true) - ); + array_fill_keys(array_keys(array_filter($requiredPackages, $filterExtensions, ARRAY_FILTER_USE_KEY)), false), + array_fill_keys(array_keys(array_filter($requiredDevPackages, $filterPackages, ARRAY_FILTER_USE_KEY)), true), + array_fill_keys(array_keys(array_filter($requiredDevPackages, $filterExtensions, ARRAY_FILTER_USE_KEY)), true) + )); if (count($this->dependencies) === 0) { - throw new InvalidConfigException("No packages found in $composerJsonPath file."); + throw new InvalidConfigException("No dependencies found in $composerJsonPath file."); + } + } + + /** + * @param array $dependencies + * @return array + */ + private function normalizeNames(array $dependencies): array + { + $normalized = []; + + foreach ($dependencies as $dependency => $isDev) { + if (strpos($dependency, 'ext-') === 0) { + $key = self::normalizeExtensionName($dependency); + } else { + $key = $dependency; + } + + $normalized[$key] = $isDev; } + + return $normalized; + } + + /** + * Zend Opcache -> zend-opcache + */ + public static function normalizeExtensionName(string $extension): string + { + return str_replace(' ', '-', strtolower($extension)); } /** diff --git a/src/Config/Configuration.php b/src/Config/Configuration.php index 7fb7291..8883145 100644 --- a/src/Config/Configuration.php +++ b/src/Config/Configuration.php @@ -14,6 +14,11 @@ class Configuration { + /** + * @var bool + */ + private $extensionsAnalysis = true; + /** * @var bool */ @@ -67,12 +72,12 @@ class Configuration /** * @var array> */ - private $ignoredErrorsOnPackage = []; + private $ignoredErrorsOnDependency = []; /** * @var array>> */ - private $ignoredErrorsOnPackageAndPath = []; + private $ignoredErrorsOnDependencyAndPath = []; /** * @var list @@ -94,6 +99,17 @@ class Configuration */ private $ignoredUnknownFunctionsRegexes = []; + /** + * Disable analysis of ext-* dependencies + * + * @return $this + */ + public function disableExtensionsAnalysis(): self + { + $this->extensionsAnalysis = false; + return $this; + } + /** * @return $this */ @@ -103,6 +119,9 @@ public function disableComposerAutoloadPathScan(): self return $this; } + /** + * @return $this + */ public function disableReportingUnmatchedIgnores(): self { $this->reportUnmatchedIgnores = false; @@ -277,13 +296,34 @@ public function ignoreErrorsOnPaths(array $paths, array $errorTypes): self public function ignoreErrorsOnPackage(string $packageName, array $errorTypes): self { $this->checkPackageName($packageName); - $this->checkAllowedErrorTypeForPackageIgnore($errorTypes); + $this->ignoreErrorsOnDependency($packageName, $errorTypes); + return $this; + } - $previousErrorTypes = $this->ignoredErrorsOnPackage[$packageName] ?? []; - $this->ignoredErrorsOnPackage[$packageName] = array_merge($previousErrorTypes, $errorTypes); + /** + * @param list $errorTypes + * @return $this + * @throws InvalidConfigException + */ + public function ignoreErrorsOnExtension(string $extension, array $errorTypes): self + { + $this->checkExtensionName($extension); + $this->ignoreErrorsOnDependency($extension, $errorTypes); return $this; } + /** + * @param list $errorTypes + * @throws InvalidConfigException + */ + private function ignoreErrorsOnDependency(string $dependency, array $errorTypes): void + { + $this->checkAllowedErrorTypeForPackageIgnore($errorTypes); + + $previousErrorTypes = $this->ignoredErrorsOnDependency[$dependency] ?? []; + $this->ignoredErrorsOnDependency[$dependency] = array_merge($previousErrorTypes, $errorTypes); + } + /** * @param list $packageNames * @param list $errorTypes @@ -299,6 +339,21 @@ public function ignoreErrorsOnPackages(array $packageNames, array $errorTypes): return $this; } + /** + * @param list $extensions + * @param list $errorTypes + * @return $this + * @throws InvalidConfigException + */ + public function ignoreErrorsOnExtensions(array $extensions, array $errorTypes): self + { + foreach ($extensions as $extension) { + $this->ignoreErrorsOnExtension($extension, $errorTypes); + } + + return $this; + } + /** * @param list $errorTypes * @return $this @@ -308,14 +363,37 @@ public function ignoreErrorsOnPackages(array $packageNames, array $errorTypes): public function ignoreErrorsOnPackageAndPath(string $packageName, string $path, array $errorTypes): self { $this->checkPackageName($packageName); + $this->ignoreErrorsOnDependencyAndPath($packageName, $path, $errorTypes); + return $this; + } + + /** + * @param list $errorTypes + * @return $this + * @throws InvalidPathException + * @throws InvalidConfigException + */ + public function ignoreErrorsOnExtensionAndPath(string $extension, string $path, array $errorTypes): self + { + $this->checkExtensionName($extension); + $this->ignoreErrorsOnDependencyAndPath($extension, $path, $errorTypes); + return $this; + } + + /** + * @param list $errorTypes + * @throws InvalidPathException + * @throws InvalidConfigException + */ + private function ignoreErrorsOnDependencyAndPath(string $dependency, string $path, array $errorTypes): void + { $this->checkAllowedErrorTypeForPathIgnore($errorTypes); $this->checkAllowedErrorTypeForPackageIgnore($errorTypes); $realpath = Path::realpath($path); - $previousErrorTypes = $this->ignoredErrorsOnPackageAndPath[$packageName][$realpath] ?? []; - $this->ignoredErrorsOnPackageAndPath[$packageName][$realpath] = array_merge($previousErrorTypes, $errorTypes); - return $this; + $previousErrorTypes = $this->ignoredErrorsOnDependencyAndPath[$dependency][$realpath] ?? []; + $this->ignoredErrorsOnDependencyAndPath[$dependency][$realpath] = array_merge($previousErrorTypes, $errorTypes); } /** @@ -334,6 +412,22 @@ public function ignoreErrorsOnPackageAndPaths(string $packageName, array $paths, return $this; } + /** + * @param list $paths + * @param list $errorTypes + * @return $this + * @throws InvalidPathException + * @throws InvalidConfigException + */ + public function ignoreErrorsOnExtensionAndPaths(string $extension, array $paths, array $errorTypes): self + { + foreach ($paths as $path) { + $this->ignoreErrorsOnExtensionAndPath($extension, $path, $errorTypes); + } + + return $this; + } + /** * @param list $packages * @param list $paths @@ -351,6 +445,23 @@ public function ignoreErrorsOnPackagesAndPaths(array $packages, array $paths, ar return $this; } + /** + * @param list $extensions + * @param list $paths + * @param list $errorTypes + * @return $this + * @throws InvalidPathException + * @throws InvalidConfigException + */ + public function ignoreErrorsOnExtensionsAndPaths(array $extensions, array $paths, array $errorTypes): self + { + foreach ($extensions as $extension) { + $this->ignoreErrorsOnExtensionAndPaths($extension, $paths, $errorTypes); + } + + return $this; + } + /** * @param list $classNames * @return $this @@ -406,8 +517,8 @@ public function getIgnoreList(): IgnoreList return new IgnoreList( $this->ignoredErrors, $this->ignoredErrorsOnPath, - $this->ignoredErrorsOnPackage, - $this->ignoredErrorsOnPackageAndPath, + $this->ignoredErrorsOnDependency, + $this->ignoredErrorsOnDependencyAndPath, $this->ignoredUnknownClasses, $this->ignoredUnknownClassesRegexes, $this->ignoredUnknownFunctions, @@ -439,6 +550,11 @@ public function getPathsToScan(): array return $this->pathsToScan; } + public function shouldAnalyseExtensions(): bool + { + return $this->extensionsAnalysis; + } + public function shouldScanComposerAutoloadPaths(): bool { return $this->scanComposerAutoloadPaths; @@ -476,6 +592,16 @@ private function isFilepathWithinPath(string $filePath, string $path): bool return strpos($filePath, $path) === 0; } + /** + * @throws InvalidConfigException + */ + private function checkExtensionName(string $extension): void + { + if (strpos($extension, 'ext-') !== 0) { + throw new InvalidConfigException("Invalid php extension dependency name '$extension', it is expected to start with ext-"); + } + } + /** * @throws InvalidConfigException */ diff --git a/src/Config/Ignore/IgnoreList.php b/src/Config/Ignore/IgnoreList.php index 59da5a5..92e3e8a 100644 --- a/src/Config/Ignore/IgnoreList.php +++ b/src/Config/Ignore/IgnoreList.php @@ -25,12 +25,12 @@ class IgnoreList /** * @var array> */ - private $ignoredErrorsOnPackage = []; + private $ignoredErrorsOnDependency = []; /** * @var array>> */ - private $ignoredErrorsOnPackageAndPath = []; + private $ignoredErrorsOnDependencyAndPath = []; /** * @var array @@ -55,8 +55,8 @@ class IgnoreList /** * @param list $ignoredErrors * @param array> $ignoredErrorsOnPath - * @param array> $ignoredErrorsOnPackage - * @param array>> $ignoredErrorsOnPackageAndPath + * @param array> $ignoredErrorsOnDependency + * @param array>> $ignoredErrorsOnDependencyAndPath * @param list $ignoredUnknownClasses * @param list $ignoredUnknownClassesRegexes * @param list $ignoredUnknownFunctions @@ -65,8 +65,8 @@ class IgnoreList public function __construct( array $ignoredErrors, array $ignoredErrorsOnPath, - array $ignoredErrorsOnPackage, - array $ignoredErrorsOnPackageAndPath, + array $ignoredErrorsOnDependency, + array $ignoredErrorsOnDependencyAndPath, array $ignoredUnknownClasses, array $ignoredUnknownClassesRegexes, array $ignoredUnknownFunctions, @@ -79,13 +79,13 @@ public function __construct( $this->ignoredErrorsOnPath[$path] = array_fill_keys($errorTypes, false); } - foreach ($ignoredErrorsOnPackage as $packageName => $errorTypes) { - $this->ignoredErrorsOnPackage[$packageName] = array_fill_keys($errorTypes, false); + foreach ($ignoredErrorsOnDependency as $dependency => $errorTypes) { + $this->ignoredErrorsOnDependency[$dependency] = array_fill_keys($errorTypes, false); } - foreach ($ignoredErrorsOnPackageAndPath as $packageName => $paths) { + foreach ($ignoredErrorsOnDependencyAndPath as $dependency => $paths) { foreach ($paths as $path => $errorTypes) { - $this->ignoredErrorsOnPackageAndPath[$packageName][$path] = array_fill_keys($errorTypes, false); + $this->ignoredErrorsOnDependencyAndPath[$dependency][$path] = array_fill_keys($errorTypes, false); } } @@ -116,7 +116,7 @@ public function getUnusedIgnores(): array } } - foreach ($this->ignoredErrorsOnPackage as $packageName => $errorTypes) { + foreach ($this->ignoredErrorsOnDependency as $packageName => $errorTypes) { foreach ($errorTypes as $errorType => $ignored) { if (!$ignored) { $unused[] = new UnusedErrorIgnore($errorType, null, $packageName); @@ -124,7 +124,7 @@ public function getUnusedIgnores(): array } } - foreach ($this->ignoredErrorsOnPackageAndPath as $packageName => $paths) { + foreach ($this->ignoredErrorsOnDependencyAndPath as $packageName => $paths) { foreach ($paths as $path => $errorTypes) { foreach ($errorTypes as $errorType => $ignored) { if (!$ignored) { @@ -240,12 +240,12 @@ private function shouldIgnoreUnknownFunctionByRegex(string $function): bool /** * @param ErrorType::SHADOW_DEPENDENCY|ErrorType::UNUSED_DEPENDENCY|ErrorType::DEV_DEPENDENCY_IN_PROD|ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV $errorType */ - public function shouldIgnoreError(string $errorType, ?string $realPath, ?string $packageName): bool + public function shouldIgnoreError(string $errorType, ?string $realPath, ?string $dependency): bool { $ignoredGlobally = $this->shouldIgnoreErrorGlobally($errorType); $ignoredByPath = $realPath !== null && $this->shouldIgnoreErrorOnPath($errorType, $realPath); - $ignoredByPackage = $packageName !== null && $this->shouldIgnoreErrorOnPackage($errorType, $packageName); - $ignoredByPackageAndPath = $realPath !== null && $packageName !== null && $this->shouldIgnoreErrorOnPackageAndPath($errorType, $packageName, $realPath); + $ignoredByPackage = $dependency !== null && $this->shouldIgnoreErrorOnDependency($errorType, $dependency); + $ignoredByPackageAndPath = $realPath !== null && $dependency !== null && $this->shouldIgnoreErrorOnDependencyAndPath($errorType, $dependency, $realPath); return $ignoredGlobally || $ignoredByPackageAndPath || $ignoredByPath || $ignoredByPackage; } @@ -281,10 +281,10 @@ private function shouldIgnoreErrorOnPath(string $errorType, string $filePath): b /** * @param ErrorType::* $errorType */ - private function shouldIgnoreErrorOnPackage(string $errorType, string $packageName): bool + private function shouldIgnoreErrorOnDependency(string $errorType, string $dependency): bool { - if (isset($this->ignoredErrorsOnPackage[$packageName][$errorType])) { - $this->ignoredErrorsOnPackage[$packageName][$errorType] = true; + if (isset($this->ignoredErrorsOnDependency[$dependency][$errorType])) { + $this->ignoredErrorsOnDependency[$dependency][$errorType] = true; return true; } @@ -294,12 +294,12 @@ private function shouldIgnoreErrorOnPackage(string $errorType, string $packageNa /** * @param ErrorType::* $errorType */ - private function shouldIgnoreErrorOnPackageAndPath(string $errorType, string $packageName, string $filePath): bool + private function shouldIgnoreErrorOnDependencyAndPath(string $errorType, string $packageName, string $filePath): bool { - if (isset($this->ignoredErrorsOnPackageAndPath[$packageName])) { - foreach ($this->ignoredErrorsOnPackageAndPath[$packageName] as $path => $errorTypes) { + if (isset($this->ignoredErrorsOnDependencyAndPath[$packageName])) { + foreach ($this->ignoredErrorsOnDependencyAndPath[$packageName] as $path => $errorTypes) { if ($this->isFilepathWithinPath($filePath, $path) && isset($errorTypes[$errorType])) { - $this->ignoredErrorsOnPackageAndPath[$packageName][$path][$errorType] = true; + $this->ignoredErrorsOnDependencyAndPath[$packageName][$path][$errorType] = true; return true; } } diff --git a/src/Initializer.php b/src/Initializer.php index 8379b86..3c3e36c 100644 --- a/src/Initializer.php +++ b/src/Initializer.php @@ -52,6 +52,8 @@ class Initializer --ignore-shadow-deps Ignore all shadow dependency issues --ignore-dev-in-prod-deps Ignore all dev dependency in production code issues --ignore-prod-only-in-dev-deps Ignore all prod dependency used only in dev paths issues + + --disable-ext-analysis Disable analysis of php extensions (e.g. ext-xml) EOD; /** @@ -116,6 +118,7 @@ public function initConfiguration( $config = new Configuration(); } + $disableExtAnalysis = $options->disableExtAnalysis === true; $ignoreUnknownClasses = $options->ignoreUnknownClasses === true; $ignoreUnknownFunctions = $options->ignoreUnknownFunctions === true; $ignoreUnused = $options->ignoreUnusedDeps === true; @@ -123,6 +126,10 @@ public function initConfiguration( $ignoreDevInProd = $options->ignoreDevInProdDeps === true; $ignoreProdOnlyInDev = $options->ignoreProdOnlyInDevDeps === true; + if ($disableExtAnalysis) { + $config->disableExtensionsAnalysis(); + } + if ($ignoreUnknownClasses) { $config->ignoreErrors([ErrorType::UNKNOWN_CLASS]); } diff --git a/src/UsedSymbolExtractor.php b/src/UsedSymbolExtractor.php index 8f386c8..371bc61 100644 --- a/src/UsedSymbolExtractor.php +++ b/src/UsedSymbolExtractor.php @@ -8,6 +8,7 @@ use function ltrim; use function strlen; use function strpos; +use function strtolower; use function substr; use function token_get_all; use const PHP_VERSION_ID; @@ -19,14 +20,18 @@ use const T_CURLY_OPEN; use const T_DOC_COMMENT; use const T_DOLLAR_OPEN_CURLY_BRACES; +use const T_DOUBLE_COLON; use const T_ENUM; use const T_FUNCTION; +use const T_INSTEADOF; use const T_INTERFACE; use const T_NAME_FULLY_QUALIFIED; use const T_NAME_QUALIFIED; use const T_NAMESPACE; use const T_NEW; use const T_NS_SEPARATOR; +use const T_NULLSAFE_OBJECT_OPERATOR; +use const T_OBJECT_OPERATOR; use const T_STRING; use const T_TRAIT; use const T_USE; @@ -62,10 +67,13 @@ public function __construct(string $code) * It does not produce any local names in current namespace * - this results in very limited functionality in files without namespace * + * @param array $extensionSymbols * @return array>> * @license Inspired by https://github.com/doctrine/annotations/blob/2.0.0/lib/Doctrine/Common/Annotations/TokenParser.php */ - public function parseUsedSymbols(): array + public function parseUsedSymbols( + array $extensionSymbols + ): array { $usedSymbols = []; $useStatements = []; @@ -107,13 +115,16 @@ public function parseUsedSymbols(): array break; case PHP_VERSION_ID >= 80000 ? T_NAMESPACE : -1: + // namespace change $inGlobalScope = false; - $useStatements = []; // reset use statements on namespace change + $useStatements = []; + $useStatementKinds = []; break; case PHP_VERSION_ID >= 80000 ? T_NAME_FULLY_QUALIFIED : -1: $symbolName = $this->normalizeBackslash($token[1]); - $kind = $this->getFqnSymbolKind($this->pointer - 2, $this->pointer, $inAttributeSquareLevel !== null); + $lowerSymbolName = strtolower($symbolName); + $kind = $extensionSymbols[$lowerSymbolName] ?? $this->getFqnSymbolKind($this->pointer - 2, $this->pointer, $inAttributeSquareLevel !== null); $usedSymbols[$kind][$symbolName][] = $token[2]; break; @@ -122,24 +133,37 @@ public function parseUsedSymbols(): array if (isset($useStatements[$neededAlias])) { $symbolName = $useStatements[$neededAlias] . substr($token[1], strlen($neededAlias)); - $kind = $this->getFqnSymbolKind($this->pointer - 2, $this->pointer, $inAttributeSquareLevel !== null); - $usedSymbols[$kind][$symbolName][] = $token[2]; - } elseif ($inGlobalScope) { $symbolName = $token[1]; - $kind = $this->getFqnSymbolKind($this->pointer - 2, $this->pointer, $inAttributeSquareLevel !== null); - $usedSymbols[$kind][$symbolName][] = $token[2]; + } else { + break; } + $lowerSymbolName = strtolower($symbolName); + $kind = $extensionSymbols[$lowerSymbolName] ?? $this->getFqnSymbolKind($this->pointer - 2, $this->pointer, $inAttributeSquareLevel !== null); + $usedSymbols[$kind][$symbolName][] = $token[2]; + break; case PHP_VERSION_ID >= 80000 ? T_STRING : -1: $name = $token[1]; + $lowerName = strtolower($name); + $pointerBeforeName = $this->pointer - 2; + $pointerAfterName = $this->pointer; + + if (!$this->canBeSymbolName($pointerBeforeName, $pointerAfterName)) { + break; + } if (isset($useStatements[$name])) { $symbolName = $useStatements[$name]; $kind = $useStatementKinds[$name]; $usedSymbols[$kind][$symbolName][] = $token[2]; + + } elseif (isset($extensionSymbols[$lowerName])) { + $symbolName = $name; + $kind = $extensionSymbols[$lowerName]; + $usedSymbols[$kind][$symbolName][] = $token[2]; } break; @@ -149,8 +173,10 @@ public function parseUsedSymbols(): array $nextName = $this->parseNameForOldPhp(); if (substr($nextName, 0, 1) !== '\\') { // not a namespace-relative name, but a new namespace declaration - $useStatements = []; // reset use statements on namespace change + // namespace change $inGlobalScope = false; + $useStatements = []; + $useStatementKinds = []; } break; @@ -158,9 +184,10 @@ public function parseUsedSymbols(): array case PHP_VERSION_ID < 80000 ? T_NS_SEPARATOR : -1: $pointerBeforeName = $this->pointer - 2; $symbolName = $this->normalizeBackslash($this->parseNameForOldPhp()); + $lowerSymbolName = strtolower($symbolName); if ($symbolName !== '') { // e.g. \array (NS separator followed by not-a-name) - $kind = $this->getFqnSymbolKind($pointerBeforeName, $this->pointer - 1, false); + $kind = $extensionSymbols[$lowerSymbolName] ?? $this->getFqnSymbolKind($pointerBeforeName, $this->pointer - 1, false); $usedSymbols[$kind][$symbolName][] = $token[2]; } @@ -169,23 +196,34 @@ public function parseUsedSymbols(): array case PHP_VERSION_ID < 80000 ? T_STRING : -1: $pointerBeforeName = $this->pointer - 2; $name = $this->parseNameForOldPhp(); + $lowerName = strtolower($name); + $pointerAfterName = $this->pointer - 1; + + if (!$this->canBeSymbolName($pointerBeforeName, $pointerAfterName)) { + break; + } if (isset($useStatements[$name])) { // unqualified name $symbolName = $useStatements[$name]; $kind = $useStatementKinds[$name]; $usedSymbols[$kind][$symbolName][] = $token[2]; + } elseif (isset($extensionSymbols[$lowerName])) { + $symbolName = $name; + $kind = $extensionSymbols[$lowerName]; + $usedSymbols[$kind][$symbolName][] = $token[2]; + } else { [$neededAlias] = explode('\\', $name, 2); if (isset($useStatements[$neededAlias])) { // qualified name $symbolName = $useStatements[$neededAlias] . substr($name, strlen($neededAlias)); - $kind = $this->getFqnSymbolKind($pointerBeforeName, $this->pointer - 1, false); + $kind = $this->getFqnSymbolKind($pointerBeforeName, $pointerAfterName, false); $usedSymbols[$kind][$symbolName][] = $token[2]; } elseif ($inGlobalScope && strpos($name, '\\') !== false) { $symbolName = $name; - $kind = $this->getFqnSymbolKind($pointerBeforeName, $this->pointer - 1, false); + $kind = $this->getFqnSymbolKind($pointerBeforeName, $pointerAfterName, false); $usedSymbols[$kind][$symbolName][] = $token[2]; } } @@ -216,7 +254,7 @@ public function parseUsedSymbols(): array } } - return $usedSymbols; // @phpstan-ignore-line Not enough precise analysis "Offset 'kind' (1|2|3) does not accept type int<1, max>" + return $usedSymbols; } /** @@ -346,44 +384,88 @@ private function getFqnSymbolKind( return SymbolKind::CLASSLIKE; } + $tokenBeforeName = $this->getTokenBefore($pointerBeforeName); + $tokenAfterName = $this->getTokenAfter($pointerAfterName); + + if ( + $tokenAfterName === '(' + && $tokenBeforeName[0] !== T_NEW // eliminate new \ClassName( + ) { + return SymbolKind::FUNCTION; + } + + return SymbolKind::CLASSLIKE; // constant may fall here, this is eliminated later + } + + private function canBeSymbolName( + int $pointerBeforeName, + int $pointerAfterName + ): bool + { + $tokenBeforeName = $this->getTokenBefore($pointerBeforeName); + $tokenAfterName = $this->getTokenAfter($pointerAfterName); + + if ( + $tokenBeforeName[0] === T_DOUBLE_COLON + || $tokenBeforeName[0] === T_INSTEADOF + || $tokenBeforeName[0] === T_AS + || $tokenBeforeName[0] === T_FUNCTION + || $tokenBeforeName[0] === T_OBJECT_OPERATOR + || $tokenBeforeName[0] === T_NAMESPACE + || $tokenBeforeName[0] === (PHP_VERSION_ID > 80000 ? T_NULLSAFE_OBJECT_OPERATOR : -1) + || $tokenAfterName[0] === T_INSTEADOF + || $tokenAfterName[0] === T_AS + ) { + return false; + } + + return true; + } + + /** + * @return array{int, string}|string + */ + private function getTokenBefore(int $pointer) + { do { - $tokenBeforeName = $this->tokens[$pointerBeforeName]; + $token = $this->tokens[$pointer]; - if (!is_array($tokenBeforeName)) { + if (!is_array($token)) { break; } - if ($tokenBeforeName[0] === T_WHITESPACE || $tokenBeforeName[0] === T_COMMENT || $tokenBeforeName[0] === T_DOC_COMMENT) { - $pointerBeforeName--; + if ($token[0] === T_WHITESPACE || $token[0] === T_COMMENT || $token[0] === T_DOC_COMMENT) { + $pointer--; continue; } break; - } while ($pointerBeforeName >= 0); + } while ($pointer >= 0); + return $token; + } + + /** + * @return array{int, string}|string + */ + private function getTokenAfter(int $pointer) + { do { - $tokenAfterName = $this->tokens[$pointerAfterName]; + $token = $this->tokens[$pointer]; - if (!is_array($tokenAfterName)) { + if (!is_array($token)) { break; } - if ($tokenAfterName[0] === T_WHITESPACE || $tokenAfterName[0] === T_COMMENT || $tokenAfterName[0] === T_DOC_COMMENT) { - $pointerAfterName++; + if ($token[0] === T_WHITESPACE || $token[0] === T_COMMENT || $token[0] === T_DOC_COMMENT) { + $pointer++; continue; } break; - } while ($pointerAfterName < $this->numTokens); - - if ( - $tokenAfterName === '(' - && $tokenBeforeName[0] !== T_NEW // eliminate new \ClassName( - ) { - return SymbolKind::FUNCTION; - } + } while ($pointer < $this->numTokens); - return SymbolKind::CLASSLIKE; // constant may fall here, this is eliminated later + return $token; } } diff --git a/tests/AnalyserTest.php b/tests/AnalyserTest.php index 0f6ed15..aff3d20 100644 --- a/tests/AnalyserTest.php +++ b/tests/AnalyserTest.php @@ -609,6 +609,78 @@ public function testOtherSymbols(): void self::assertEquals($this->createAnalysisResult(1, []), $result); } + public function testExtensions(): void + { + $vendorDir = realpath(__DIR__ . '/../vendor'); + $prodPath = realpath(__DIR__ . '/data/not-autoloaded/extensions/ext-prod-usages.php'); + $devPath = realpath(__DIR__ . '/data/not-autoloaded/extensions/ext-dev-usages.php'); + self::assertNotFalse($vendorDir); + self::assertNotFalse($prodPath); + self::assertNotFalse($devPath); + + $config = new Configuration(); + $config->addPathToScan($prodPath, false); + $config->addPathToScan($devPath, true); + + $detector = new Analyser( + $this->getStopwatchMock(), + $vendorDir, + [$vendorDir => $this->getClassLoaderMock()], + $config, + [ + 'ext-dom' => false, + 'ext-libxml' => true, + 'ext-mbstring' => false, + ] + ); + $result = $detector->run(); + + $this->assertResultsWithoutUsages($this->createAnalysisResult(2, [ + ErrorType::SHADOW_DEPENDENCY => [ + 'ext-pdo' => ['PDO' => [new SymbolUsage($prodPath, 5, SymbolKind::CLASSLIKE)]], + ], + ErrorType::DEV_DEPENDENCY_IN_PROD => [ + 'ext-libxml' => ['LIBXML_NOEMPTYTAG' => [new SymbolUsage($prodPath, 3, SymbolKind::CONSTANT)]], + ], + ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV => [ + 'ext-dom', + ], + ErrorType::UNUSED_DEPENDENCY => [ + 'ext-mbstring', + ], + ]), $result); + } + + public function testDisabledExtensionAnalysis(): void + { + $vendorDir = realpath(__DIR__ . '/../vendor'); + $prodPath = realpath(__DIR__ . '/data/not-autoloaded/extensions/ext-prod-usages.php'); + $devPath = realpath(__DIR__ . '/data/not-autoloaded/extensions/ext-dev-usages.php'); + self::assertNotFalse($vendorDir); + self::assertNotFalse($prodPath); + self::assertNotFalse($devPath); + + $config = new Configuration(); + $config->disableExtensionsAnalysis(); + $config->addPathToScan($prodPath, false); + $config->addPathToScan($devPath, true); + + $detector = new Analyser( + $this->getStopwatchMock(), + $vendorDir, + [$vendorDir => $this->getClassLoaderMock()], + $config, + [ + 'ext-dom' => false, + 'ext-libxml' => true, + 'ext-mbstring' => false, + ] + ); + $result = $detector->run(); + + $this->assertResultsWithoutUsages($this->createAnalysisResult(2, []), $result); + } + public function testPharSupport(): void { $canCreatePhar = ini_set('phar.readonly', '0'); @@ -762,6 +834,7 @@ private function assertResultsWithoutUsages(AnalysisResult $expectedResult, Anal self::assertSame($expectedResult->getScannedFilesCount(), $result->getScannedFilesCount(), 'Scanned files count mismatch'); self::assertEquals($expectedResult->getUnusedIgnores(), $result->getUnusedIgnores(), 'Unused ignores mismatch'); self::assertEquals($expectedResult->getUnknownClassErrors(), $result->getUnknownClassErrors(), 'Unknown class mismatch'); + self::assertEquals($expectedResult->getUnknownFunctionErrors(), $result->getUnknownFunctionErrors(), 'Unknown functions mismatch'); self::assertEquals($expectedResult->getShadowDependencyErrors(), $result->getShadowDependencyErrors(), 'Shadow dependency mismatch'); self::assertEquals($expectedResult->getDevDependencyInProductionErrors(), $result->getDevDependencyInProductionErrors(), 'Dev dependency in production mismatch'); self::assertEquals($expectedResult->getProdDependencyOnlyInDevErrors(), $result->getProdDependencyOnlyInDevErrors(), 'Prod dependency only in dev mismatch'); diff --git a/tests/BinTest.php b/tests/BinTest.php index d95b3ed..daf16a6 100644 --- a/tests/BinTest.php +++ b/tests/BinTest.php @@ -17,7 +17,7 @@ public function test(): void $testsDir = __DIR__; $noComposerJsonError = 'File composer.json not found'; - $noPackagesError = 'No packages found'; + $noPackagesError = 'No dependencies found'; $parseError = 'Failure while parsing'; $junitDumpError = "Cannot use 'junit' format with '--dump-usages' option"; diff --git a/tests/CliTest.php b/tests/CliTest.php index 8bcee01..72f7148 100644 --- a/tests/CliTest.php +++ b/tests/CliTest.php @@ -53,11 +53,12 @@ public function validationDataProvider(): iterable yield 'valid bool options' => [ null, - ['bin/script.php', '--help', '--verbose', '--ignore-shadow-deps', '--ignore-unused-deps', '--ignore-dev-in-prod-deps', '--ignore-unknown-classes', '--ignore-unknown-functions'], + ['bin/script.php', '--help', '--verbose', '--ignore-shadow-deps', '--ignore-unused-deps', '--ignore-dev-in-prod-deps', '--ignore-unknown-classes', '--ignore-unknown-functions', '--disable-ext-analysis'], (static function (): CliOptions { $options = new CliOptions(); $options->help = true; $options->verbose = true; + $options->disableExtAnalysis = true; $options->ignoreShadowDeps = true; $options->ignoreUnusedDeps = true; $options->ignoreDevInProdDeps = true; diff --git a/tests/ConfigurationTest.php b/tests/ConfigurationTest.php index 9edf75d..27c2a26 100644 --- a/tests/ConfigurationTest.php +++ b/tests/ConfigurationTest.php @@ -21,6 +21,7 @@ public function testShouldIgnore(): void $configuration->ignoreUnknownClasses(['Unknown\Clazz']); $configuration->ignoreErrors([ErrorType::UNUSED_DEPENDENCY, ErrorType::UNKNOWN_CLASS]); $configuration->ignoreErrorsOnPath(__DIR__ . '/data/../', [ErrorType::SHADOW_DEPENDENCY]); + $configuration->ignoreErrorsOnExtension('ext-xml', [ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV]); $configuration->ignoreErrorsOnPackage('my/package', [ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV]); $configuration->ignoreErrorsOnPackageAndPath('vendor/package', __DIR__ . '/../tests/data', [ErrorType::DEV_DEPENDENCY_IN_PROD]); @@ -42,6 +43,13 @@ public function testShouldIgnore(): void self::assertTrue($ignoreList->shouldIgnoreError(ErrorType::SHADOW_DEPENDENCY, __DIR__ . DIRECTORY_SEPARATOR . 'app', null)); self::assertTrue($ignoreList->shouldIgnoreError(ErrorType::SHADOW_DEPENDENCY, __DIR__ . DIRECTORY_SEPARATOR . 'app', 'some/package')); + self::assertFalse($ignoreList->shouldIgnoreError(ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV, null, null)); + self::assertFalse($ignoreList->shouldIgnoreError(ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV, null, 'ext-simplexml')); + self::assertTrue($ignoreList->shouldIgnoreError(ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV, null, 'ext-xml')); + self::assertFalse($ignoreList->shouldIgnoreError(ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV, __DIR__, null)); + self::assertFalse($ignoreList->shouldIgnoreError(ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV, __DIR__, 'ext-simplexml')); + self::assertTrue($ignoreList->shouldIgnoreError(ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV, __DIR__, 'ext-xml')); + self::assertFalse($ignoreList->shouldIgnoreError(ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV, null, null)); self::assertFalse($ignoreList->shouldIgnoreError(ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV, null, 'some/package')); self::assertTrue($ignoreList->shouldIgnoreError(ErrorType::PROD_DEPENDENCY_ONLY_IN_DEV, null, 'my/package')); diff --git a/tests/InitializerTest.php b/tests/InitializerTest.php index 5bd4f7f..f8f7113 100644 --- a/tests/InitializerTest.php +++ b/tests/InitializerTest.php @@ -99,6 +99,7 @@ public function testInitCliOptions(): void self::assertNull($options->showAllUsages); self::assertNull($options->composerJson); + self::assertNull($options->disableExtAnalysis); self::assertNull($options->ignoreProdOnlyInDevDeps); self::assertNull($options->ignoreUnknownClasses); self::assertNull($options->ignoreUnknownFunctions); diff --git a/tests/UsedSymbolExtractorTest.php b/tests/UsedSymbolExtractorTest.php index 8f35c8b..bc6e397 100644 --- a/tests/UsedSymbolExtractorTest.php +++ b/tests/UsedSymbolExtractorTest.php @@ -4,6 +4,7 @@ use PHPUnit\Framework\TestCase; use function file_get_contents; +use function strtolower; use const PHP_VERSION_ID; class UsedSymbolExtractorTest extends TestCase @@ -11,20 +12,24 @@ class UsedSymbolExtractorTest extends TestCase /** * @param array>> $expectedUsages + * @param array $extensionSymbols * @dataProvider provideVariants */ - public function test(string $path, array $expectedUsages): void + public function test(string $path, array $expectedUsages, array $extensionSymbols = []): void { $code = file_get_contents($path); self::assertNotFalse($code); $extractor = new UsedSymbolExtractor($code); - self::assertSame($expectedUsages, $extractor->parseUsedSymbols()); + self::assertSame( + $expectedUsages, + $extractor->parseUsedSymbols($extensionSymbols) + ); } /** - * @return iterable>>}> + * @return iterable>>, 2?: array}> */ public function provideVariants(): iterable { @@ -49,6 +54,14 @@ public function provideVariants(): iterable ], ]; + yield 'T_STRING issues' => [ + __DIR__ . '/data/not-autoloaded/used-symbols/t-string-issues.php', + [], + [ + strtolower('PDO') => SymbolKind::CLASSLIKE, + ], + ]; + yield 'various usages' => [ __DIR__ . '/data/not-autoloaded/used-symbols/various-usages.php', [ @@ -118,6 +131,54 @@ public function provideVariants(): iterable [], ]; + yield 'extensions' => [ + __DIR__ . '/data/not-autoloaded/used-symbols/extensions.php', + [ + SymbolKind::FUNCTION => [ + 'json_encode' => [8], + 'DDTrace\active_span' => [12], + 'DDTrace\root_span' => [13], + 'DDTrace\Integrations\Exec\proc_get_pid' => [16], + 'json_decode' => [21], + ], + SymbolKind::CONSTANT => [ + 'LIBXML_ERR_FATAL' => [9], + 'LIBXML_ERR_ERROR' => [10], + 'DDTrace\DBM_PROPAGATION_FULL' => [14], + ], + SymbolKind::CLASSLIKE => [ + 'PDO' => [11], + 'My\App\XMLReader' => [15], + 'CURLOPT_SSL_VERIFYHOST' => [19], + ], + ], + self::extensionSymbolsForExtensionsTestCases(), + ]; + + yield 'extensions global' => [ + __DIR__ . '/data/not-autoloaded/used-symbols/extensions-global.php', + [ + SymbolKind::FUNCTION => [ + 'json_encode' => [8], + 'DDTrace\active_span' => [12], + 'DDTrace\root_span' => [13], + 'DDTrace\Integrations\Exec\proc_get_pid' => [16], + 'json_decode' => [21], + ], + SymbolKind::CONSTANT => [ + 'LIBXML_ERR_FATAL' => [9], + 'LIBXML_ERR_ERROR' => [10], + 'DDTrace\DBM_PROPAGATION_FULL' => [14], + ], + SymbolKind::CLASSLIKE => [ + 'PDO' => [11], + 'My\App\XMLReader' => [15], + 'CURLOPT_SSL_VERIFYHOST' => [19], + ], + ], + self::extensionSymbolsForExtensionsTestCases(), + ]; + if (PHP_VERSION_ID >= 80000) { yield 'attribute' => [ __DIR__ . '/data/not-autoloaded/used-symbols/attribute.php', @@ -139,4 +200,22 @@ public function provideVariants(): iterable } } + /** + * @return array + */ + private static function extensionSymbolsForExtensionsTestCases(): array + { + return [ + strtolower('XMLReader') => SymbolKind::CLASSLIKE, + strtolower('PDO') => SymbolKind::CLASSLIKE, + strtolower('json_encode') => SymbolKind::FUNCTION, + strtolower('DDTrace\active_span') => SymbolKind::FUNCTION, + strtolower('DDTrace\root_span') => SymbolKind::FUNCTION, + strtolower('LIBXML_ERR_FATAL') => SymbolKind::CONSTANT, + strtolower('LIBXML_ERR_ERROR') => SymbolKind::CONSTANT, + strtolower('DDTrace\DBM_PROPAGATION_FULL') => SymbolKind::CONSTANT, + strtolower('DDTrace\Integrations\Exec\proc_get_pid') => SymbolKind::FUNCTION, + ]; + } + } diff --git a/tests/data/not-autoloaded/extensions/ext-dev-usages.php b/tests/data/not-autoloaded/extensions/ext-dev-usages.php new file mode 100644 index 0000000..8eb727f --- /dev/null +++ b/tests/data/not-autoloaded/extensions/ext-dev-usages.php @@ -0,0 +1,4 @@ +array_filter(); + $this?->array_filter(); + self::array_filter(); + } + +}