123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445 |
- <?php declare(strict_types=1);
- /*
- * This file is part of sebastian/code-unit.
- *
- * (c) Sebastian Bergmann <sebastian@phpunit.de>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace SebastianBergmann\CodeUnit;
- use function range;
- use function sprintf;
- use ReflectionClass;
- use ReflectionFunction;
- use ReflectionMethod;
- /**
- * @psalm-immutable
- */
- abstract class CodeUnit
- {
- /**
- * @var string
- */
- private $name;
- /**
- * @var string
- */
- private $sourceFileName;
- /**
- * @var array
- * @psalm-var list<int>
- */
- private $sourceLines;
- /**
- * @psalm-param class-string $className
- *
- * @throws InvalidCodeUnitException
- * @throws ReflectionException
- */
- public static function forClass(string $className): ClassUnit
- {
- self::ensureUserDefinedClass($className);
- $reflector = self::reflectorForClass($className);
- return new ClassUnit(
- $className,
- $reflector->getFileName(),
- range(
- $reflector->getStartLine(),
- $reflector->getEndLine()
- )
- );
- }
- /**
- * @psalm-param class-string $className
- *
- * @throws InvalidCodeUnitException
- * @throws ReflectionException
- */
- public static function forClassMethod(string $className, string $methodName): ClassMethodUnit
- {
- self::ensureUserDefinedClass($className);
- $reflector = self::reflectorForClassMethod($className, $methodName);
- return new ClassMethodUnit(
- $className . '::' . $methodName,
- $reflector->getFileName(),
- range(
- $reflector->getStartLine(),
- $reflector->getEndLine()
- )
- );
- }
- /**
- * @psalm-param class-string $interfaceName
- *
- * @throws InvalidCodeUnitException
- * @throws ReflectionException
- */
- public static function forInterface(string $interfaceName): InterfaceUnit
- {
- self::ensureUserDefinedInterface($interfaceName);
- $reflector = self::reflectorForClass($interfaceName);
- return new InterfaceUnit(
- $interfaceName,
- $reflector->getFileName(),
- range(
- $reflector->getStartLine(),
- $reflector->getEndLine()
- )
- );
- }
- /**
- * @psalm-param class-string $interfaceName
- *
- * @throws InvalidCodeUnitException
- * @throws ReflectionException
- */
- public static function forInterfaceMethod(string $interfaceName, string $methodName): InterfaceMethodUnit
- {
- self::ensureUserDefinedInterface($interfaceName);
- $reflector = self::reflectorForClassMethod($interfaceName, $methodName);
- return new InterfaceMethodUnit(
- $interfaceName . '::' . $methodName,
- $reflector->getFileName(),
- range(
- $reflector->getStartLine(),
- $reflector->getEndLine()
- )
- );
- }
- /**
- * @psalm-param class-string $traitName
- *
- * @throws InvalidCodeUnitException
- * @throws ReflectionException
- */
- public static function forTrait(string $traitName): TraitUnit
- {
- self::ensureUserDefinedTrait($traitName);
- $reflector = self::reflectorForClass($traitName);
- return new TraitUnit(
- $traitName,
- $reflector->getFileName(),
- range(
- $reflector->getStartLine(),
- $reflector->getEndLine()
- )
- );
- }
- /**
- * @psalm-param class-string $traitName
- *
- * @throws InvalidCodeUnitException
- * @throws ReflectionException
- */
- public static function forTraitMethod(string $traitName, string $methodName): TraitMethodUnit
- {
- self::ensureUserDefinedTrait($traitName);
- $reflector = self::reflectorForClassMethod($traitName, $methodName);
- return new TraitMethodUnit(
- $traitName . '::' . $methodName,
- $reflector->getFileName(),
- range(
- $reflector->getStartLine(),
- $reflector->getEndLine()
- )
- );
- }
- /**
- * @psalm-param callable-string $functionName
- *
- * @throws InvalidCodeUnitException
- * @throws ReflectionException
- */
- public static function forFunction(string $functionName): FunctionUnit
- {
- $reflector = self::reflectorForFunction($functionName);
- if (!$reflector->isUserDefined()) {
- throw new InvalidCodeUnitException(
- sprintf(
- '"%s" is not a user-defined function',
- $functionName
- )
- );
- }
- return new FunctionUnit(
- $functionName,
- $reflector->getFileName(),
- range(
- $reflector->getStartLine(),
- $reflector->getEndLine()
- )
- );
- }
- /**
- * @psalm-param list<int> $sourceLines
- */
- private function __construct(string $name, string $sourceFileName, array $sourceLines)
- {
- $this->name = $name;
- $this->sourceFileName = $sourceFileName;
- $this->sourceLines = $sourceLines;
- }
- public function name(): string
- {
- return $this->name;
- }
- public function sourceFileName(): string
- {
- return $this->sourceFileName;
- }
- /**
- * @psalm-return list<int>
- */
- public function sourceLines(): array
- {
- return $this->sourceLines;
- }
- public function isClass(): bool
- {
- return false;
- }
- public function isClassMethod(): bool
- {
- return false;
- }
- public function isInterface(): bool
- {
- return false;
- }
- public function isInterfaceMethod(): bool
- {
- return false;
- }
- public function isTrait(): bool
- {
- return false;
- }
- public function isTraitMethod(): bool
- {
- return false;
- }
- public function isFunction(): bool
- {
- return false;
- }
- /**
- * @psalm-param class-string $className
- *
- * @throws InvalidCodeUnitException
- */
- private static function ensureUserDefinedClass(string $className): void
- {
- try {
- $reflector = new ReflectionClass($className);
- if ($reflector->isInterface()) {
- throw new InvalidCodeUnitException(
- sprintf(
- '"%s" is an interface and not a class',
- $className
- )
- );
- }
- if ($reflector->isTrait()) {
- throw new InvalidCodeUnitException(
- sprintf(
- '"%s" is a trait and not a class',
- $className
- )
- );
- }
- if (!$reflector->isUserDefined()) {
- throw new InvalidCodeUnitException(
- sprintf(
- '"%s" is not a user-defined class',
- $className
- )
- );
- }
- // @codeCoverageIgnoreStart
- } catch (\ReflectionException $e) {
- throw new ReflectionException(
- $e->getMessage(),
- (int) $e->getCode(),
- $e
- );
- }
- // @codeCoverageIgnoreEnd
- }
- /**
- * @psalm-param class-string $interfaceName
- *
- * @throws InvalidCodeUnitException
- */
- private static function ensureUserDefinedInterface(string $interfaceName): void
- {
- try {
- $reflector = new ReflectionClass($interfaceName);
- if (!$reflector->isInterface()) {
- throw new InvalidCodeUnitException(
- sprintf(
- '"%s" is not an interface',
- $interfaceName
- )
- );
- }
- if (!$reflector->isUserDefined()) {
- throw new InvalidCodeUnitException(
- sprintf(
- '"%s" is not a user-defined interface',
- $interfaceName
- )
- );
- }
- // @codeCoverageIgnoreStart
- } catch (\ReflectionException $e) {
- throw new ReflectionException(
- $e->getMessage(),
- (int) $e->getCode(),
- $e
- );
- }
- // @codeCoverageIgnoreEnd
- }
- /**
- * @psalm-param class-string $traitName
- *
- * @throws InvalidCodeUnitException
- */
- private static function ensureUserDefinedTrait(string $traitName): void
- {
- try {
- $reflector = new ReflectionClass($traitName);
- if (!$reflector->isTrait()) {
- throw new InvalidCodeUnitException(
- sprintf(
- '"%s" is not a trait',
- $traitName
- )
- );
- }
- // @codeCoverageIgnoreStart
- if (!$reflector->isUserDefined()) {
- throw new InvalidCodeUnitException(
- sprintf(
- '"%s" is not a user-defined trait',
- $traitName
- )
- );
- }
- } catch (\ReflectionException $e) {
- throw new ReflectionException(
- $e->getMessage(),
- (int) $e->getCode(),
- $e
- );
- }
- // @codeCoverageIgnoreEnd
- }
- /**
- * @psalm-param class-string $className
- *
- * @throws ReflectionException
- */
- private static function reflectorForClass(string $className): ReflectionClass
- {
- try {
- return new ReflectionClass($className);
- // @codeCoverageIgnoreStart
- } catch (\ReflectionException $e) {
- throw new ReflectionException(
- $e->getMessage(),
- (int) $e->getCode(),
- $e
- );
- }
- // @codeCoverageIgnoreEnd
- }
- /**
- * @psalm-param class-string $className
- *
- * @throws ReflectionException
- */
- private static function reflectorForClassMethod(string $className, string $methodName): ReflectionMethod
- {
- try {
- return new ReflectionMethod($className, $methodName);
- // @codeCoverageIgnoreStart
- } catch (\ReflectionException $e) {
- throw new ReflectionException(
- $e->getMessage(),
- (int) $e->getCode(),
- $e
- );
- }
- // @codeCoverageIgnoreEnd
- }
- /**
- * @psalm-param callable-string $functionName
- *
- * @throws ReflectionException
- */
- private static function reflectorForFunction(string $functionName): ReflectionFunction
- {
- try {
- return new ReflectionFunction($functionName);
- // @codeCoverageIgnoreStart
- } catch (\ReflectionException $e) {
- throw new ReflectionException(
- $e->getMessage(),
- (int) $e->getCode(),
- $e
- );
- }
- // @codeCoverageIgnoreEnd
- }
- }
|