*/ private const BUILT_IN_TYPES_PRECEDENCE = [ 'bool' => 1, 'int' => 2, 'float' => 3, 'string' => 4, 'array' => 5, 'callable' => 6, 'iterable' => 7, 'object' => 8, 'static' => 9, 'mixed' => 10, 'void' => 11, 'false' => 12, 'true' => 13, 'null' => 14, 'never' => 15, ]; /** @psalm-var array */ private const NOT_NULLABLE_TYPES = [ 'null' => null, 'false' => null, 'true' => null, 'void' => null, 'mixed' => null, 'never' => null, ]; /** A regex pattern to match valid class/interface/trait names */ private const VALID_IDENTIFIER_MATCHER = '/^[a-zA-Z_\x80-\xff][a-zA-Z0-9_\x80-\xff]*' . '(\\\\[a-zA-Z_\x80-\xff][a-zA-Z0-9_\x80-\xff]*)*$/'; /** @psalm-var value-of|0 */ public int $sortIndex; /** @psalm-var non-empty-string */ public string $type; /** * @psalm-param non-empty-string $type * @psalm-param value-of|0 $sortIndex */ private function __construct(string $type, int $sortIndex) { $this->type = $type; $this->sortIndex = $sortIndex; } /** * @psalm-pure * @throws InvalidArgumentException */ public static function fromString(string $type): self { $trimmedType = '\\' === ($type[0] ?? '') ? substr($type, 1) : $type; $lowerCaseType = strtolower($trimmedType); if (array_key_exists($lowerCaseType, self::BUILT_IN_TYPES_PRECEDENCE)) { if ($lowerCaseType !== strtolower($type)) { throw new InvalidArgumentException(sprintf( 'Provided type "%s" is a built-in type, and should not be prefixed with "\\"', $type )); } return new self($lowerCaseType, self::BUILT_IN_TYPES_PRECEDENCE[$lowerCaseType]); } if (1 !== preg_match(self::VALID_IDENTIFIER_MATCHER, $trimmedType)) { throw new InvalidArgumentException(sprintf( 'Provided type "%s" is not recognized as a valid expression: ' . 'it must match "%s" or be one of the built-in types (%s)', $type, self::VALID_IDENTIFIER_MATCHER, implode(', ', self::BUILT_IN_TYPES_PRECEDENCE) )); } assert('' !== $trimmedType); return new self($trimmedType, 0); } /** * @psalm-pure * @throws InvalidArgumentException */ public static function fromReflectionNamedTypeAndClass( ReflectionNamedType $type, ?ReflectionClass $currentClass ): self { $name = $type->getName(); $lowerCaseName = strtolower($name); if ('self' === $lowerCaseName && $currentClass) { return new self($currentClass->getName(), 0); } if ('parent' === $lowerCaseName && $currentClass && $parentClass = $currentClass->getParentClass()) { return new self($parentClass->getName(), 0); } return self::fromString($name); } /** @psalm-return non-empty-string */ public function fullyQualifiedName(): string { return array_key_exists($this->type, self::BUILT_IN_TYPES_PRECEDENCE) ? $this->type : '\\' . $this->type; } /** @return non-empty-string */ public function toString(): string { return $this->type; } /** @throws InvalidArgumentException */ public function assertCanUnionWith(self|IntersectionType $other): void { if ($other instanceof IntersectionType) { $other->assertCanUnionWith($this); return; } if ( 'mixed' === $this->type || 'void' === $this->type || 'never' === $this->type ) { throw new InvalidArgumentException(sprintf( 'Type "%s" cannot be composed in a union with any other types', $this->type )); } if ($other->type === $this->type) { throw new InvalidArgumentException(sprintf( 'Type "%s" cannot be composed in a union with the same type "%s"', $this->type, $other->type )); } if ( ('true' === $other->type && 'false' === $this->type) || ('false' === $other->type && 'true' === $this->type) ) { throw new InvalidArgumentException(sprintf( 'Type "%s" cannot be composed in a union with type "%s"', $this->type, $other->type )); } } /** @throws InvalidArgumentException */ public function assertCanIntersectWith(AtomicType $other): void { if (array_key_exists($this->type, self::BUILT_IN_TYPES_PRECEDENCE)) { throw new InvalidArgumentException(sprintf( 'Type "%s" cannot be composed in an intersection with any other types', $this->type )); } if ($other->type === $this->type) { throw new InvalidArgumentException(sprintf( 'Type "%s" cannot be composed in an intersection with the same type "%s"', $this->type, $other->type )); } } /** @throws InvalidArgumentException */ public function assertCanBeStandaloneNullable(): void { if (array_key_exists($this->type, self::NOT_NULLABLE_TYPES)) { throw new InvalidArgumentException(sprintf( 'Type "%s" cannot be nullable', $this->type )); } } } __halt_compiler();----SIGNATURE:----xdviYPOGhINtPR0pUB2tHVxFl4NeD6hpe0wbBijT/nAkyf5Q7aR/tzJ94/AIuZ5eW34GuarYzSb2OJWNEW0ufnkTnEsaEi9L5jVUmXjGfEXAcTbMjFo3FANHXhxpLdotjHU+SHlAtyvSpqEnamrgO7CGBGY8ZFM47YoI9q8RVtK+GHunYO0x18Atr1h2P4dMu/PYflctQOooB1xzosJ6LLE7ON6EO75Cda4CaLZEwfrSAUKRUiISNvXfypGJdGETlBiiuPbsg+QrD0o+tFEe3Xh7uF+qBUW3oPsVpgE5df3H+j1lytdVWQ/hWiSIha/d4AAPz2kL1cRnUAd8LaZY8tFcA87Jh24vnkpHSfspJ1+OAPj698CQFMRjaYILLl2gyJdgREfuMluoOsYIDUrizz20/kkDO7ne9+ttYjr02hs87au3WFvcwnGMSSLFyrUM8gTndCHMUiEgduelCZANrlQQg+WJ3Y/DY4aKErwyOveHPRvGyUnMyyWmCbn8nSMrWcg6Vz3OFWvn4TRIfn3Khwk/HBZwH3+NhprSa6lCCRre/GOzqR7lf3lStF+8+dCuKTQstYQXD5d2VNYH4pxvnA2Z2Sr3BzvMtjIlJ8wSBbjHIvpvOIRqrZTnuAdW+L6sOH9ZULj1oE7DFAxvmL+oLsPsXolmudbZUQc2a++rqB4=----ATTACHMENT:----NDUxOTUzMjEyOTU1OTU4OCA1NjA4Njk2NjI1NTUyNzY3IDQ4Mjk1OTg5MzA3MjY1NTI=