-
Notifications
You must be signed in to change notification settings - Fork 474
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Use AutoloadSourceLocator to see existing classes in memory for stati…
…c reflection
- Loading branch information
1 parent
775aee8
commit c0cb3a6
Showing
7 changed files
with
219 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
200 changes: 200 additions & 0 deletions
200
src/Reflection/BetterReflection/SourceLocator/AutoloadSourceLocator.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,200 @@ | ||
<?php declare(strict_types = 1); | ||
|
||
namespace PHPStan\Reflection\BetterReflection\SourceLocator; | ||
|
||
use PHPStan\File\FileReader; | ||
use ReflectionClass; | ||
use ReflectionException; | ||
use Roave\BetterReflection\Identifier\Identifier; | ||
use Roave\BetterReflection\Identifier\IdentifierType; | ||
use Roave\BetterReflection\Reflection\Reflection; | ||
use Roave\BetterReflection\Reflector\Exception\IdentifierNotFound; | ||
use Roave\BetterReflection\Reflector\Reflector; | ||
use Roave\BetterReflection\SourceLocator\Ast\Exception\ParseToAstFailure; | ||
use Roave\BetterReflection\SourceLocator\Ast\Locator as AstLocator; | ||
use Roave\BetterReflection\SourceLocator\Located\LocatedSource; | ||
use Roave\BetterReflection\SourceLocator\Type\SourceLocator; | ||
use function file_exists; | ||
|
||
/** | ||
* Use PHP's built in autoloader to locate a class, without actually loading. | ||
* | ||
* There are some prerequisites... | ||
* - we expect the autoloader to load classes from a file (i.e. using require/include) | ||
* | ||
* Modified code from Roave/BetterReflection, Copyright (c) 2017 Roave, LLC. | ||
*/ | ||
class AutoloadSourceLocator implements SourceLocator | ||
{ | ||
|
||
/** @var AstLocator */ | ||
private $astLocator; | ||
|
||
/** | ||
* Primarily used by the non-loading-autoloader magic trickery to determine | ||
* the filename used during autoloading. | ||
* | ||
* @var string|null | ||
*/ | ||
private static $autoloadLocatedFile; | ||
|
||
/** @var AstLocator|null */ | ||
private static $currentAstLocator; | ||
|
||
/** | ||
* Note: the constructor has been made a 0-argument constructor because `\stream_wrapper_register` | ||
* is a piece of trash, and doesn't accept instances, just class names. | ||
*/ | ||
public function __construct(?AstLocator $astLocator = null) | ||
{ | ||
$validLocator = $astLocator ?? self::$currentAstLocator; | ||
if ($validLocator === null) { | ||
throw new \PHPStan\ShouldNotHappenException(); | ||
} | ||
|
||
$this->astLocator = $validLocator; | ||
} | ||
|
||
/** | ||
* {@inheritDoc} | ||
* | ||
* @throws ParseToAstFailure | ||
*/ | ||
public function locateIdentifier(Reflector $reflector, Identifier $identifier): ?Reflection | ||
{ | ||
if (!$identifier->isClass()) { | ||
return null; | ||
} | ||
|
||
$locateResult = $this->locateClassByName($identifier->getName()); | ||
if ($locateResult === null) { | ||
return null; | ||
} | ||
[$potentiallyLocatedFile, $className] = $locateResult; | ||
|
||
$locatedSource = new LocatedSource( | ||
FileReader::read($potentiallyLocatedFile), | ||
$potentiallyLocatedFile | ||
); | ||
|
||
try { | ||
return $this->astLocator->findReflection($reflector, $locatedSource, new Identifier($className, $identifier->getType())); | ||
} catch (IdentifierNotFound $exception) { | ||
return null; | ||
} | ||
} | ||
|
||
public function locateIdentifiersByType(Reflector $reflector, IdentifierType $identifierType): array | ||
{ | ||
return []; // todo | ||
} | ||
|
||
/** | ||
* Attempt to locate a class by name. | ||
* | ||
* If class already exists, simply use internal reflection API to get the | ||
* filename and store it. | ||
* | ||
* If class does not exist, we make an assumption that whatever autoloaders | ||
* that are registered will be loading a file. We then override the file:// | ||
* protocol stream wrapper to "capture" the filename we expect the class to | ||
* be in, and then restore it. Note that class_exists will cause an error | ||
* that it cannot find the file, so we squelch the errors by overriding the | ||
* error handler temporarily. | ||
* | ||
* @throws ReflectionException | ||
* @return array{string, string}|null | ||
*/ | ||
private function locateClassByName(string $className): ?array | ||
{ | ||
if (class_exists($className, false) || interface_exists($className, false) || trait_exists($className, false)) { | ||
$reflection = new ReflectionClass($className); | ||
$filename = $reflection->getFileName(); | ||
|
||
if (!is_string($filename)) { | ||
return null; | ||
} | ||
|
||
if (!file_exists($filename)) { | ||
return null; | ||
} | ||
|
||
return [$filename, $reflection->getName()]; | ||
} | ||
|
||
self::$autoloadLocatedFile = null; | ||
self::$currentAstLocator = $this->astLocator; // passing the locator on to the implicitly instantiated `self` | ||
$previousErrorHandler = set_error_handler(static function (): bool { | ||
return true; | ||
}); | ||
stream_wrapper_unregister('file'); | ||
stream_wrapper_register('file', self::class); | ||
class_exists($className); | ||
stream_wrapper_restore('file'); | ||
set_error_handler($previousErrorHandler); | ||
|
||
if (self::$autoloadLocatedFile === null) { | ||
return null; | ||
} | ||
|
||
return [self::$autoloadLocatedFile, $className]; | ||
} | ||
|
||
/** | ||
* Our wrapper simply records which file we tried to load and returns | ||
* boolean false indicating failure. | ||
* | ||
* @see https://php.net/manual/en/class.streamwrapper.php | ||
* @see https://php.net/manual/en/streamwrapper.stream-open.php | ||
* | ||
* @param string $path | ||
* @param string $mode | ||
* @param int $options | ||
* @param string $opened_path | ||
* | ||
* @phpcsSuppress SlevomatCodingStandard.TypeHints.ParameterTypeHint.MissingNativeTypeHint | ||
*/ | ||
public function stream_open($path, $mode, $options, &$opened_path): bool | ||
{ | ||
self::$autoloadLocatedFile = $path; | ||
|
||
return false; | ||
} | ||
|
||
/** | ||
* url_stat is triggered by calls like "file_exists". The call to "file_exists" must not be overloaded. | ||
* This function restores the original "file" stream, issues a call to "stat" to get the real results, | ||
* and then re-registers the AutoloadSourceLocator stream wrapper. | ||
* | ||
* @see https://php.net/manual/en/class.streamwrapper.php | ||
* @see https://php.net/manual/en/streamwrapper.url-stat.php | ||
* | ||
* @param string $path | ||
* @param int $flags | ||
* | ||
* @return mixed[]|bool | ||
* | ||
* @phpcsSuppress SlevomatCodingStandard.TypeHints.ParameterTypeHint.MissingNativeTypeHint | ||
*/ | ||
public function url_stat($path, $flags) | ||
{ | ||
stream_wrapper_restore('file'); | ||
|
||
if (($flags & STREAM_URL_STAT_QUIET) !== 0) { | ||
set_error_handler(static function (): bool { | ||
// Use native error handler | ||
return false; | ||
}); | ||
$result = @stat($path); | ||
restore_error_handler(); | ||
} else { | ||
$result = stat($path); | ||
} | ||
|
||
stream_wrapper_unregister('file'); | ||
stream_wrapper_register('file', self::class); | ||
|
||
return $result; | ||
} | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters