Find this useful? Enter your email to receive occasional updates for securing PHP code.

Signing you up...

Thank you for signing up!

PHP Decode

<?php namespace Laminas\Code\Generator\TypeGenerator; use Laminas\Code\Generator\Excepti..

Decoded Output download

<?php

namespace Laminas\Code\Generator\TypeGenerator;

use Laminas\Code\Generator\Exception\InvalidArgumentException;
use ReflectionClass;
use ReflectionNamedType;

use function array_key_exists;
use function assert;
use function implode;
use function preg_match;
use function sprintf;
use function strtolower;
use function substr;

/**
 * Represents a single/indivisible (atomic) type, as supported by PHP.
 * This means that this object can be composed into more complex union, intersection
 * and nullable types.
 *
 * @internal the {@see AtomicType} is an implementation detail of the type generator,
 *
 * @psalm-immutable
 */
final class AtomicType
{
    /**
     * Built-in type sorting, ascending.
     *
     * @psalm-var array<non-empty-string, positive-int>
     */
    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<non-empty-string, null> */
    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_-][a-zA-Z0-9_-]*'
        . '(\\[a-zA-Z_-][a-zA-Z0-9_-]*)*$/';

    /** @psalm-var value-of<AtomicType::BUILT_IN_TYPES_PRECEDENCE>|0 */
    public int $sortIndex;

    /** @psalm-var non-empty-string */
    public string $type;

    /**
     * @psalm-param non-empty-string $type
     * @psalm-param value-of<AtomicType::BUILT_IN_TYPES_PRECEDENCE>|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
            ));
        }
    }
}
 ?>

Did this file decode correctly?

Original Code

<?php

namespace Laminas\Code\Generator\TypeGenerator;

use Laminas\Code\Generator\Exception\InvalidArgumentException;
use ReflectionClass;
use ReflectionNamedType;

use function array_key_exists;
use function assert;
use function implode;
use function preg_match;
use function sprintf;
use function strtolower;
use function substr;

/**
 * Represents a single/indivisible (atomic) type, as supported by PHP.
 * This means that this object can be composed into more complex union, intersection
 * and nullable types.
 *
 * @internal the {@see AtomicType} is an implementation detail of the type generator,
 *
 * @psalm-immutable
 */
final class AtomicType
{
    /**
     * Built-in type sorting, ascending.
     *
     * @psalm-var array<non-empty-string, positive-int>
     */
    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<non-empty-string, null> */
    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<AtomicType::BUILT_IN_TYPES_PRECEDENCE>|0 */
    public int $sortIndex;

    /** @psalm-var non-empty-string */
    public string $type;

    /**
     * @psalm-param non-empty-string $type
     * @psalm-param value-of<AtomicType::BUILT_IN_TYPES_PRECEDENCE>|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
            ));
        }
    }
}

Function Calls

None

Variables

None

Stats

MD5 f52b5c6217244cc141b22b82bb1495cb
Eval Count 0
Decode Time 91 ms