*/ private $fullyQualifiedNameByHash = []; public function __construct(UseImportNameMatcher $useImportNameMatcher, UseImportsResolver $useImportsResolver, ReflectionProvider $reflectionProvider) { $this->useImportNameMatcher = $useImportNameMatcher; $this->useImportsResolver = $useImportsResolver; $this->reflectionProvider = $reflectionProvider; } /** * @api doctrine */ public function resolveTagToKnownFullyQualifiedName(string $tag, Property $property) : ?string { return $this->_resolveTagFullyQualifiedName($tag, $property, \true); } public function resolveTagFullyQualifiedName(string $tag, Node $node) : ?string { return $this->_resolveTagFullyQualifiedName($tag, $node, \false); } private function _resolveTagFullyQualifiedName(string $tag, Node $node, bool $returnNullOnUnknownClass) : ?string { $uniqueHash = $tag . \spl_object_hash($node); if (isset($this->fullyQualifiedNameByHash[$uniqueHash])) { return $this->fullyQualifiedNameByHash[$uniqueHash]; } $tag = \ltrim($tag, '@'); $uses = $this->useImportsResolver->resolveForNode($node); $fullyQualifiedClass = $this->resolveFullyQualifiedClass($uses, $node, $tag, $returnNullOnUnknownClass); if ($fullyQualifiedClass === null) { if ($returnNullOnUnknownClass) { return null; } $fullyQualifiedClass = $tag; } $this->fullyQualifiedNameByHash[$uniqueHash] = $fullyQualifiedClass; return $fullyQualifiedClass; } /** * @param Use_[]|GroupUse[] $uses */ private function resolveFullyQualifiedClass(array $uses, Node $node, string $tag, bool $returnNullOnUnknownClass) : ?string { $scope = $node->getAttribute(AttributeKey::SCOPE); if ($scope instanceof Scope) { $namespace = $scope->getNamespace(); if ($namespace !== null) { $namespacedTag = $namespace . '\\' . $tag; if ($this->reflectionProvider->hasClass($namespacedTag)) { return $namespacedTag; } if (\strpos($tag, '\\') === \false) { return $this->resolveAsAliased($uses, $tag, $returnNullOnUnknownClass); } if ($this->isPreslashedExistingClass($tag)) { // Global or absolute Class return $tag; } } } $class = $this->useImportNameMatcher->matchNameWithUses($tag, $uses); return $this->resolveClass($class, $returnNullOnUnknownClass); } /** * @param Use_[]|GroupUse[] $uses */ private function resolveAsAliased(array $uses, string $tag, bool $returnNullOnUnknownClass) : ?string { foreach ($uses as $use) { $prefix = $this->useImportsResolver->resolvePrefix($use); foreach ($use->uses as $useUse) { if (!$useUse->alias instanceof Identifier) { continue; } if ($useUse->alias->toString() === $tag) { $class = $prefix . $useUse->name->toString(); return $this->resolveClass($class, $returnNullOnUnknownClass); } } } $class = $this->useImportNameMatcher->matchNameWithUses($tag, $uses); return $this->resolveClass($class, $returnNullOnUnknownClass); } private function resolveClass(?string $class, bool $returnNullOnUnknownClass) : ?string { if ($class === null) { return null; } $resolvedClass = $this->reflectionProvider->hasClass($class) ? $class : null; return $returnNullOnUnknownClass ? $resolvedClass : $class; } private function isPreslashedExistingClass(string $tag) : bool { if (\strncmp($tag, '\\', \strlen('\\')) !== 0) { return \false; } return $this->reflectionProvider->hasClass($tag); } }