* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Symfony\Bridge\PhpUnit\DeprecationErrorHandler; use PHPUnit\Framework\TestCase; use PHPUnit\Framework\TestSuite; use PHPUnit\Metadata\Api\Groups; use PHPUnit\Util\Test; use Symfony\Bridge\PhpUnit\Legacy\SymfonyTestsListenerFor; use Symfony\Component\ErrorHandler\DebugClassLoader; class_exists(Groups::class); /** * @internal */ class Deprecation { public const PATH_TYPE_VENDOR = 'path_type_vendor'; public const PATH_TYPE_SELF = 'path_type_internal'; public const PATH_TYPE_UNDETERMINED = 'path_type_undetermined'; public const TYPE_SELF = 'type_self'; public const TYPE_DIRECT = 'type_direct'; public const TYPE_INDIRECT = 'type_indirect'; public const TYPE_UNDETERMINED = 'type_undetermined'; private $trace = []; private $message; private $languageDeprecation; private $originClass; private $originMethod; private $triggeringFile; /** @var string[] Absolute paths to vendor directories */ private static $vendors; /** * @var string[] Absolute paths to source or tests of the project, cache * directories excluded because it is based on autoloading * rules and cache systems typically do not use those */ private static $internalPaths = []; private $originalFilesStack; /** * @param string $message * @param string $file * @param bool $languageDeprecation */ public function __construct($message, array $trace, $file, $languageDeprecation = false) { if (isset($trace[2]['function']) && 'trigger_deprecation' === $trace[2]['function']) { $file = $trace[2]['file']; array_splice($trace, 1, 1); } $this->trace = $trace; $this->message = $message; $this->languageDeprecation = $languageDeprecation; $i = \count($trace); while (1 < $i && $this->lineShouldBeSkipped($trace[--$i])) { // No-op } $line = $trace[$i]; $this->triggeringFile = $file; for ($j = 1; $j < $i; ++$j) { if (!isset($trace[$j]['function'], $trace[1 + $j]['class'], $trace[1 + $j]['args'][0])) { continue; } if ('trigger_error' === $trace[$j]['function'] && !isset($trace[$j]['class'])) { if (DebugClassLoader::class === $trace[1 + $j]['class']) { $class = $trace[1 + $j]['args'][0]; $this->triggeringFile = isset($trace[1 + $j]['args'][1]) ? realpath($trace[1 + $j]['args'][1]) : (new \ReflectionClass($class))->getFileName(); $this->getOriginalFilesStack(); array_splice($this->originalFilesStack, 0, $j, [$this->triggeringFile]); if (preg_match('/(?|"([^"]++)" that is deprecated|should implement method "(?:static )?([^:]++))/', $message, $m) || (false === strpos($message, '()" will return') && false === strpos($message, 'native return type declaration') && preg_match('/^(?:The|Method) "([^":]++)/', $message, $m))) { $this->triggeringFile = (new \ReflectionClass($m[1]))->getFileName(); array_unshift($this->originalFilesStack, $this->triggeringFile); } } break; } } if (!isset($line['object']) && !isset($line['class'])) { return; } set_error_handler(function () {}); try { $parsedMsg = unserialize($this->message); } finally { restore_error_handler(); } if ($parsedMsg && isset($parsedMsg['deprecation'])) { $this->message = $parsedMsg['deprecation']; $this->originClass = $parsedMsg['class']; $this->originMethod = $parsedMsg['method']; if (isset($parsedMsg['files_stack'])) { $this->originalFilesStack = $parsedMsg['files_stack']; } // If the deprecation has been triggered via // \Symfony\Bridge\PhpUnit\Legacy\SymfonyTestsListenerTrait::endTest() // then we need to use the serialized information to determine // if the error has been triggered from vendor code. if (isset($parsedMsg['triggering_file'])) { $this->triggeringFile = $parsedMsg['triggering_file']; } return; } if (!isset($line['class'], $trace[$i - 2]['function']) || 0 !== strpos($line['class'], SymfonyTestsListenerFor::class)) { $this->originClass = isset($line['object']) ? \get_class($line['object']) : $line['class']; $this->originMethod = $line['function']; return; } $test = $line['args'][0] ?? null; if (($test instanceof TestCase || $test instanceof TestSuite) && ('trigger_error' !== $trace[$i - 2]['function'] || isset($trace[$i - 2]['class']))) { $this->originClass = \get_class($test); $this->originMethod = $test->getName(); return; } } /** * @return bool */ private function lineShouldBeSkipped(array $line) { if (!isset($line['class'])) { return true; } $class = $line['class']; return 'ReflectionMethod' === $class || 0 === strpos($class, 'PHPUnit\\'); } /** * @return bool */ public function originatesFromAnObject() { return isset($this->originClass); } /** * @return string */ public function originatingClass() { if (null === $this->originClass) { throw new \LogicException('Check with originatesFromAnObject() before calling this method.'); } $class = $this->originClass; return false !== strpos($class, "@anonymous\0") ? (get_parent_class($class) ?: key(class_implements($class)) ?: 'class').'@anonymous' : $class; } /** * @return string */ public function originatingMethod() { if (null === $this->originMethod) { throw new \LogicException('Check with originatesFromAnObject() before calling this method.'); } return $this->originMethod; } /** * @return string */ public function getMessage() { return $this->message; } /** * @return bool */ public function isLegacy() { if (!$this->originClass || (new \ReflectionClass($this->originClass))->isInternal()) { return false; } $method = $this->originatingMethod(); $groups = class_exists(Groups::class, false) ? [new Groups(), 'groups'] : [Test::class, 'getGroups']; return 0 === strpos($method, 'testLegacy') || 0 === strpos($method, 'provideLegacy') || 0 === strpos($method, 'getLegacy') || strpos($this->originClass, '\Legacy') || \in_array('legacy', $groups($this->originClass, $method), true); } /** * @return bool */ public function isMuted() { if ('Function ReflectionType::__toString() is deprecated' !== $this->message) { return false; } if (isset($this->trace[1]['class'])) { return 0 === strpos($this->trace[1]['class'], 'PHPUnit\\'); } return false !== strpos($this->triggeringFile, \DIRECTORY_SEPARATOR.'vendor'.\DIRECTORY_SEPARATOR.'phpunit'.\DIRECTORY_SEPARATOR); } /** * Tells whether both the calling package and the called package are vendor * packages. * * @return string */ public function getType() { $pathType = $this->getPathType($this->triggeringFile); if ($this->languageDeprecation && self::PATH_TYPE_VENDOR === $pathType) { // the triggering file must be used for language deprecations return self::TYPE_INDIRECT; } if (self::PATH_TYPE_SELF === $pathType) { return self::TYPE_SELF; } if (self::PATH_TYPE_UNDETERMINED === $pathType) { return self::TYPE_UNDETERMINED; } $erroringFile = $erroringPackage = null; foreach ($this->getOriginalFilesStack() as $file) { if ('-' === $file || 'Standard input code' === $file || !realpath($file)) { continue; } if (self::PATH_TYPE_SELF === $pathType = $this->getPathType($file)) { return self::TYPE_DIRECT; } if (self::PATH_TYPE_UNDETERMINED === $pathType) { return self::TYPE_UNDETERMINED; } if (null !== $erroringFile && null !== $erroringPackage) { $package = $this->getPackage($file); if ('composer' !== $package && $package !== $erroringPackage) { return self::TYPE_INDIRECT; } continue; } $erroringFile = $file; $erroringPackage = $this->getPackage($file); } return self::TYPE_DIRECT; } private function getOriginalFilesStack() { if (null === $this->originalFilesStack) { $this->originalFilesStack = []; foreach ($this->trace as $frame) { if (!isset($frame['file'], $frame['function']) || (!isset($frame['class']) && \in_array($frame['function'], ['require', 'require_once', 'include', 'include_once'], true))) { continue; } $this->originalFilesStack[] = $frame['file']; } } return $this->originalFilesStack; } /** * getPathType() should always be called prior to calling this method. * * @param string $path * * @return string */ private function getPackage($path) { $path = realpath($path) ?: $path; foreach (self::getVendors() as $vendorRoot) { if (0 === strpos($path, $vendorRoot)) { $relativePath = substr($path, \strlen($vendorRoot) + 1); $vendor = strstr($relativePath, \DIRECTORY_SEPARATOR, true); if (false === $vendor) { return 'symfony'; } return rtrim($vendor.'/'.strstr(substr($relativePath, \strlen($vendor) + 1), \DIRECTORY_SEPARATOR, true), '/'); } } throw new \RuntimeException(sprintf('No vendors found for path "%s".', $path)); } /** * @return string[] */ private static function getVendors() { if (null === self::$vendors) { self::$vendors = $paths = []; self::$vendors[] = \dirname(__DIR__).\DIRECTORY_SEPARATOR.'Legacy'; if (class_exists(DebugClassLoader::class, false)) { self::$vendors[] = \dirname((new \ReflectionClass(DebugClassLoader::class))->getFileName()); } foreach (get_declared_classes() as $class) { if ('C' === $class[0] && 0 === strpos($class, 'ComposerAutoloaderInit')) { $r = new \ReflectionClass($class); $v = \dirname($r->getFileName(), 2); if (file_exists($v.'/composer/installed.json')) { self::$vendors[] = $v; $loader = require $v.'/autoload.php'; $paths = self::addSourcePathsFromPrefixes( array_merge($loader->getPrefixes(), $loader->getPrefixesPsr4()), $paths ); } } } foreach ($paths as $path) { foreach (self::$vendors as $vendor) { if (0 !== strpos($path, $vendor)) { self::$internalPaths[] = $path; } } } } return self::$vendors; } private static function addSourcePathsFromPrefixes(array $prefixesByNamespace, array $paths): array { foreach ($prefixesByNamespace as $prefixes) { foreach ($prefixes as $prefix) { if (false !== realpath($prefix)) { $paths[] = realpath($prefix); } } } return $paths; } /** * @param string $path * * @return string */ private function getPathType($path) { $realPath = realpath($path); if (false === $realPath && '-' !== $path && 'Standard input code' !== $path) { return self::PATH_TYPE_UNDETERMINED; } foreach (self::getVendors() as $vendor) { if (0 === strpos($realPath, $vendor) && false !== strpbrk(substr($realPath, \strlen($vendor), 1), '/'.\DIRECTORY_SEPARATOR)) { return self::PATH_TYPE_VENDOR; } } foreach (self::$internalPaths as $internalPath) { if (0 === strpos($realPath, $internalPath)) { return self::PATH_TYPE_SELF; } } return self::PATH_TYPE_UNDETERMINED; } /** * @return string */ public function toString() { $exception = new \Exception($this->message); $reflection = new \ReflectionProperty($exception, 'trace'); $reflection->setAccessible(true); $reflection->setValue($exception, $this->trace); return ($this->originatesFromAnObject() ? 'deprecation triggered by '.$this->originatingClass().'::'.$this->originatingMethod().":\n" : '') .$this->message."\n" ."Stack trace:\n" .str_replace(' '.getcwd().\DIRECTORY_SEPARATOR, ' ', $exception->getTraceAsString())."\n"; } }