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 /** * This file is part of the Zephir. * * (c) Phalcon Team <[email protected]..

Decoded Output download

<?php

/**
 * This file is part of the Zephir.
 *
 * (c) Phalcon Team <[email protected]>
 *
 * For the full copyright and license information, please view
 * the LICENSE file that was distributed with this source code.
 */

declare(strict_types=1);

namespace Zephir\Operators\Comparison;

use ReflectionException;
use Zephir\CompilationContext;
use Zephir\CompiledExpression;
use Zephir\Exception;
use Zephir\Exception\CompilerException;
use Zephir\Expression;
use Zephir\Operators\AbstractOperator;

use function strtolower;

/**
 * This is the base operator for comparison operators
 */
class ComparisonBaseOperator extends AbstractOperator
{
    protected bool $commutative = false;

    /**
     * Compile the expression.
     *
     * @param array              $expression
     * @param CompilationContext $compilationContext
     *
     * @return CompiledExpression
     *
     * @throws ReflectionException
     * @throws Exception
     */
    public function compile(array $expression, CompilationContext $compilationContext)
    {
        $conditions = $this->optimizeTypeOf($expression, $compilationContext);
        if (null !== $conditions) {
            return $conditions;
        }

        $this->checkLeft($expression, CompilerException::class, $expression);
        $this->checkRight($expression, CompilerException::class, $expression);

        $leftExpr = new Expression($expression['left']);
        $leftExpr->setReadOnly(true);
        $left = $leftExpr->compile($compilationContext);

        $rightExpr = new Expression($expression['right']);
        $rightExpr->setReadOnly(true);
        $right = $rightExpr->compile($compilationContext);

        switch ($left->getType()) {
            case 'null':
                switch ($right->getType()) {
                    case 'null':
                        return new CompiledExpression('bool', '(0 ' . $this->operator . ' 0)', $expression);

                    case 'int':
                    case 'uint':
                    case 'long':
                    case 'ulong':
                        return new CompiledExpression(
                            'bool',
                            '(0 ' . $this->operator . ' ' . $right->getCode() . ')',
                            $expression
                        );

                    case 'char':
                    case 'uchar':
                        return new CompiledExpression(
                            'bool',
                            '(\'\0\' ' . $this->operator . ' \'' . $right->getCode() . '\')',
                            $expression
                        );

                    case 'double':
                        return new CompiledExpression(
                            'bool',
                            '(0 ' . $this->operator . ' (int) ' . $right->getCode() . ')',
                            $expression
                        );

                    case 'variable':
                        $variableRight = $compilationContext->symbolTable->getVariableForRead(
                            $right->getCode(),
                            $compilationContext,
                            $expression['left']
                        );
                        switch ($variableRight->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                $compilationContext->headersManager->add('kernel/operators');

                                return new CompiledExpression(
                                    'bool',
                                    '0 ' . $this->operator . ' ' . $variableRight->getName(),
                                    $expression
                                );

                            case 'variable':
                            case 'mixed':
                            case 'string':
                                $compilationContext->headersManager->add('kernel/operators');
                                $condition = $compilationContext->backend->getTypeofCondition(
                                    $variableRight,
                                    $this->operator,
                                    'null'
                                );

                                return new CompiledExpression('bool', $condition, $expression);

                            default:
                                throw new CompilerException(
                                    'Unknown type: ' . $variableRight->getType(),
                                    $expression['right']
                                );
                        }


                    default:
                        throw new CompilerException('Unknown type: ' . $right->getType(), $expression);
                }


            case 'int':
            case 'uint':
            case 'long':
            case 'double':
            case 'ulong':
            case 'char':
            case 'uchar':
                switch ($right->getType()) {
                    case 'null':
                        return new CompiledExpression('bool', $left->getCode() . ' ' . $this->operator, $expression);

                    case 'int':
                    case 'uint':
                    case 'long':
                    case 'ulong':
                        return new CompiledExpression(
                            'bool',
                            $left->getCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                            $expression
                        );

                    case 'char':
                    case 'uchar':
                        return new CompiledExpression(
                            'bool',
                            $left->getCode() . ' ' . $this->operator . ' \'' . $right->getCode() . '\'',
                            $expression
                        );

                    case 'double':
                        return new CompiledExpression(
                            'bool',
                            $left->getCode() . ' ' . $this->operator . ' (int) ' . $right->getCode(),
                            $expression
                        );

                    case 'variable':
                        $variableRight = $compilationContext->symbolTable->getVariableForRead(
                            $right->getCode(),
                            $compilationContext,
                            $expression['left']
                        );
                        switch ($variableRight->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                $compilationContext->headersManager->add('kernel/operators');

                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                    $expression
                                );

                            case 'variable':
                            case 'mixed':
                                $compilationContext->headersManager->add('kernel/operators');
                                $variableCode = $compilationContext->backend->getVariableCode($variableRight);

                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalLongNegOperator . '(' . $variableCode . ', ' . $left->getCode() . ')',
                                    $expression
                                );

                            default:
                                throw new CompilerException(
                                    'Unknown type: ' . $variableRight->getType(),
                                    $expression['right']
                                );
                        }


                    default:
                        throw new CompilerException(
                            'Cannot compare ' . $left->getType() . ' with ' . $right->getType(),
                            $expression
                        );
                }


            case 'bool':
                switch ($right->getType()) {
                    case 'null':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' 0',
                            $expression
                        );

                    case 'int':
                    case 'uint':
                    case 'long':
                    case 'ulong':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                            $expression
                        );

                    case 'char':
                    case 'uchar':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' \'' . $right->getCode() . '\'',
                            $expression
                        );

                    case 'double':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' (int) ' . $right->getCode(),
                            $expression
                        );

                    case 'bool':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' ' . $right->getBooleanCode(),
                            $expression
                        );

                    case 'variable':
                        $variableRight = $compilationContext->symbolTable->getVariableForRead(
                            $right->getCode(),
                            $compilationContext,
                            $expression['left']
                        );
                        switch ($variableRight->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                $compilationContext->headersManager->add('kernel/operators');

                                return new CompiledExpression(
                                    'bool',
                                    $left->getBooleanCode() . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                    $expression
                                );

                            case 'variable':
                            case 'mixed':
                                $compilationContext->headersManager->add('kernel/operators');
                                $boolOperator  = '1' == $left->getBooleanCode(
                                ) ? $this->zvalBoolTrueOperator : $this->zvalBoolFalseOperator;
                                $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                return new CompiledExpression(
                                    'bool',
                                    $boolOperator . '(' . $variableRight . ')',
                                    $expression
                                );
                            default:
                                throw new CompilerException(
                                    'Unknown type: ' . $variableRight->getType(),
                                    $expression['right']
                                );
                        }


                    default:
                        throw new CompilerException(
                            'Cannot compare ' . $left->getType() . ' with ' . $right->getType(),
                            $expression
                        );
                }


            case 'string':
                $variableLeft     = $compilationContext->symbolTable->getTempLocalVariableForWrite(
                    'variable',
                    $compilationContext
                );
                $variableLeftCode = $compilationContext->backend->getVariableCode($variableLeft);
                $compilationContext->backend->assignString(
                    $variableLeft,
                    $left->getCode(),
                    $compilationContext,
                    true
                );
                switch ($right->getType()) {
                    case 'null':
                        return new CompiledExpression(
                            'bool',
                            $this->zvalNullOperator . '(' . $variableLeftCode . ')',
                            $expression['left']
                        );


                    case 'string':
                        $compilationContext->headersManager->add('kernel/operators');

                        return new CompiledExpression(
                            'bool',
                            $this->zvalStringOperator . '(' . $variableLeftCode . ', "' . $right->getCode() . '")',
                            $expression['left']
                        );


                    case 'variable':
                        $variableRight = $compilationContext->symbolTable->getVariableForRead(
                            $right->getCode(),
                            $compilationContext,
                            $expression['left']
                        );
                        switch ($variableRight->getType()) {
                            case 'string':
                            case 'variable':
                            case 'mixed':
                                $compilationContext->headersManager->add('kernel/operators');
                                $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalOperator . '(' . $variableLeftCode . ', ' . $variableRight . ')',
                                    $expression
                                );


                            default:
                                throw new CompilerException(
                                    'Unknown type: ' . $variableRight->getType(),
                                    $expression['right']
                                );
                        }


                    default:
                        throw new CompilerException('Unknown type: ' . $right->getType(), $expression['left']);
                }


            case 'variable':
                $variable     = $compilationContext->symbolTable->getVariableForRead(
                    $left->getCode(),
                    $compilationContext,
                    $expression['left']
                );
                $variableCode = $compilationContext->backend->getVariableCode($variable);
                switch ($variable->getType()) {
                    case 'int':
                    case 'uint':
                    case 'long':
                    case 'ulong':
                    case 'char':
                    case 'uchar':
                        switch ($right->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                                    $expression
                                );

                            case 'char':
                            case 'uchar':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' \'' . $right->getCode() . '\'',
                                    $expression
                                );

                            case 'bool':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getBooleanCode(),
                                    $expression
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );
                                switch ($variableRight->getType()) {
                                    case 'int':
                                    case 'uint':
                                    case 'long':
                                    case 'ulong':
                                    case 'char':
                                    case 'uchar':
                                    case 'double':
                                        return new CompiledExpression(
                                            'bool',
                                            $variable->getName(
                                            ) . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                            $expression
                                        );

                                    case 'variable':
                                    case 'mixed':
                                        $compilationContext->headersManager->add('kernel/operators');
                                        $variableRightCode = $compilationContext->backend->getVariableCode(
                                            $variableRight
                                        );
                                        $variableCode      = $compilationContext->backend->getVariableCode($variable);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalLongNegOperator . '(' . $variableRightCode . ', ' . $variableCode . ')',
                                            $expression
                                        );


                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException(
                                    'Cannot compare variable: ' . $variable->getType() . ' with: ' . $right->getType(),
                                    $expression
                                );
                        }


                    case 'double':
                        switch ($right->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                                    $expression
                                );

                            case 'bool':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getBooleanCode(),
                                    $expression
                                );

                            case 'char':
                            case 'uchar':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' \'' . $right->getCode() . '\'',
                                    $expression
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );
                                switch ($variableRight->getType()) {
                                    case 'int':
                                    case 'uint':
                                    case 'long':
                                    case 'ulong':
                                    case 'double':
                                        return new CompiledExpression(
                                            'bool',
                                            $variable->getName(
                                            ) . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                            $expression
                                        );

                                    case 'variable':
                                    case 'mixed':
                                        $compilationContext->headersManager->add('kernel/operators');
                                        $variableRightCode = $compilationContext->backend->getVariableCode(
                                            $variableRight
                                        );
                                        $variableCode      = $compilationContext->backend->getVariableCode($variable);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalDoubleNegOperator . '(' . $variableRightCode . ', ' . $variableCode . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException(
                                    'Cannot compare variable: ' . $variable->getType() . ' with: ' . $right->getType(),
                                    $expression
                                );
                        }


                    case 'bool':
                        switch ($right->getType()) {
                            case 'int':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                                    $expression['left']
                                );

                            case 'bool':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getBooleanCode(),
                                    $expression['left']
                                );

                            case 'null':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' 0',
                                    $expression['left']
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );

                                switch ($variableRight->getType()) {
                                    case 'int':
                                    case 'uint':
                                    case 'long':
                                    case 'ulong':
                                    case 'bool':
                                    case 'double':
                                        return new CompiledExpression(
                                            'bool',
                                            $variable->getName(
                                            ) . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                            $expression
                                        );

                                    case 'variable':
                                    case 'mixed':
                                        $compilationContext->headersManager->add('kernel/operators');
                                        $boolOperator      = '1' == $left->getBooleanCode(
                                        ) ? $this->zvalBoolTrueOperator : $this->zvalBoolFalseOperator;
                                        $variableRightCode = $compilationContext->backend->getVariableCode(
                                            $variableRight
                                        );

                                        return new CompiledExpression(
                                            'bool',
                                            $boolOperator . '(' . $variableRightCode . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException(
                                    'Cannot compare variable: ' . $variable->getType() . ' with: ' . $right->getType(),
                                    $expression
                                );
                        }


                    case 'array':
                        switch ($right->getType()) {
                            case 'null':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalNullOperator . '(' . $variableCode . ')',
                                    $expression['left']
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );
                                switch ($variableRight->getType()) {
                                    case 'string':
                                    case 'variable':
                                    case 'mixed':
                                    case 'array':
                                        $compilationContext->headersManager->add('kernel/operators');
                                        $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalOperator . '(' . $variableCode . ', ' . $variableRight . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException('Unknown type: ' . $right->getType(), $expression['left']);
                        }


                    case 'string':
                        $compilationContext->headersManager->add('kernel/operators');

                        switch ($right->getType()) {
                            case 'null':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalNullOperator . '(' . $variableCode . ')',
                                    $expression['left']
                                );

                            case 'string':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalStringOperator . '(' . $variableCode . ', "' . $right->getCode() . '")',
                                    $expression['left']
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );

                                switch ($variableRight->getType()) {
                                    case 'string':
                                    case 'variable':
                                    case 'mixed':
                                        $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalOperator . '(' . $variableCode . ', ' . $variableRight . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException('Unknown type: ' . $right->getType(), $expression['left']);
                        }


                    case 'variable':
                    case 'mixed':
                        $compilationContext->headersManager->add('kernel/operators');

                        switch ($right->getType()) {
                            case 'null':
                                $condition = $compilationContext->backend->getTypeofCondition(
                                    $variable,
                                    $this->operator,
                                    'null'
                                );

                                return new CompiledExpression('bool', $condition, $expression['left']);

                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalLongOperator . '(' . $variableCode . ', ' . $right->getCode() . ')',
                                    $expression['left']
                                );

                            case 'char':
                            case 'uchar':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalLongOperator . '(' . $variableCode . ', \'' . $right->getCode() . '\')',
                                    $expression['left']
                                );

                            case 'bool':
                                $zvalBoolOperator = 'true' === $right->getCode(
                                ) ? $this->zvalBoolTrueOperator : $this->zvalBoolFalseOperator;

                                return new CompiledExpression(
                                    'bool',
                                    $zvalBoolOperator . '(' . $variableCode . ')',
                                    $expression['left']
                                );

                            case 'string':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalStringOperator . '(' . $variableCode . ', "' . $right->getCode() . '")',
                                    $expression['left']
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );
                                switch ($variableRight->getType()) {
                                    case 'int':
                                    case 'uint':
                                    case 'long':
                                    case 'ulong':
                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalLongOperator . '(' . $variableCode . ', ' . $variableRight->getName(
                                            ) . ')',
                                            $expression
                                        );

                                    case 'double':
                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalDoubleOperator . '(' . $variableCode . ', ' . $variableRight->getName(
                                            ) . ')',
                                            $expression
                                        );

                                    case 'bool':
                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalBoolOperator . '(' . $variableCode . ', ' . $variableRight->getName(
                                            ) . ')',
                                            $expression
                                        );

                                    case 'string':
                                    case 'variable':
                                    case 'mixed':
                                    case 'array':
                                        $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalOperator . '(' . $variableCode . ', ' . $variableRight . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException('Unknown type: ' . $right->getType(), $expression['left']);
                        }


                    default:
                        throw new CompilerException('Unknown type: ' . $variable->getType(), $expression);
                }


            default:
                throw new CompilerException('Unknown type: ' . $left->getType(), $expression);
        }
    }

    /**
     * @param array              $expr
     * @param CompilationContext $compilationContext
     *
     * @return CompiledExpression|null
     *
     * @throws Exception
     * @throws ReflectionException
     */
    public function optimizeTypeOf(array $expr, CompilationContext $compilationContext): ?CompiledExpression
    {
        if (!isset($expr['left'])) {
            return null;
        }

        if (!isset($expr['right']) && !isset($expr['right']['value'])) {
            return null;
        }

        if ('typeof' !== $expr['left']['type']) {
            return null;
        }

        if ('string' !== $expr['right']['type']) {
            $compilationContext->logger->warning(
                "Possible invalid comparison for 'typeof' operator with non-string",
                ['invalid-typeof-comparison', $expr['right']]
            );

            return null;
        }

        switch ($expr['type']) {
            case 'identical':
            case 'equals':
                $operator = '==';
                break;

            case 'not-identical':
            case 'not-equals':
                $operator = '!=';
                break;

            default:
                return null;
        }

        $code             = (new Expression($expr['left']['left']))->compile($compilationContext)->getCode();
        $variableVariable = $compilationContext->symbolTable->getVariableForRead($code, $compilationContext, $expr);

        if ('string' !== $expr['right']['type']) {
            throw new CompilerException('Right expression of typeof operator must be "string" type', $expr['right']);
        }

        $value = strtolower($expr['right']['value']);

        switch ($variableVariable->getType()) {
            case 'double':
                $condition = match ($value) {
                    'double', 'float' => '1 ' . $operator . ' 1',
                    default           => '1 ' . $operator . ' 0',
                };
                break;

            case 'int':
            case 'integer':
            case 'long':
                $condition = match ($value) {
                    'int', 'integer', 'long' => '1 ' . $operator . ' 1',
                    default                  => '1 ' . $operator . ' 0',
                };
                break;

            case 'bool':
                $condition = match ($value) {
                    'bool', 'boolean' => '1 ' . $operator . ' 1',
                    default           => '1 ' . $operator . ' 0',
                };
                break;

            case 'array':
                $condition = match ($value) {
                    'array' => '1 ' . $operator . ' 1',
                    default => '1 ' . $operator . ' 0',
                };
                break;

            case 'string':
                $condition = match ($value) {
                    'string' => '1 ' . $operator . ' 1',
                    default  => '1 ' . $operator . ' 0',
                };
                break;

            case 'variable':
                $condition = $compilationContext->backend->getTypeofCondition($variableVariable, $operator, $value);
                break;

            default:
                return null;
        }

        return new CompiledExpression('bool', $condition, $expr);
    }
}
 ?>

Did this file decode correctly?

Original Code

<?php

/**
 * This file is part of the Zephir.
 *
 * (c) Phalcon Team <[email protected]>
 *
 * For the full copyright and license information, please view
 * the LICENSE file that was distributed with this source code.
 */

declare(strict_types=1);

namespace Zephir\Operators\Comparison;

use ReflectionException;
use Zephir\CompilationContext;
use Zephir\CompiledExpression;
use Zephir\Exception;
use Zephir\Exception\CompilerException;
use Zephir\Expression;
use Zephir\Operators\AbstractOperator;

use function strtolower;

/**
 * This is the base operator for comparison operators
 */
class ComparisonBaseOperator extends AbstractOperator
{
    protected bool $commutative = false;

    /**
     * Compile the expression.
     *
     * @param array              $expression
     * @param CompilationContext $compilationContext
     *
     * @return CompiledExpression
     *
     * @throws ReflectionException
     * @throws Exception
     */
    public function compile(array $expression, CompilationContext $compilationContext)
    {
        $conditions = $this->optimizeTypeOf($expression, $compilationContext);
        if (null !== $conditions) {
            return $conditions;
        }

        $this->checkLeft($expression, CompilerException::class, $expression);
        $this->checkRight($expression, CompilerException::class, $expression);

        $leftExpr = new Expression($expression['left']);
        $leftExpr->setReadOnly(true);
        $left = $leftExpr->compile($compilationContext);

        $rightExpr = new Expression($expression['right']);
        $rightExpr->setReadOnly(true);
        $right = $rightExpr->compile($compilationContext);

        switch ($left->getType()) {
            case 'null':
                switch ($right->getType()) {
                    case 'null':
                        return new CompiledExpression('bool', '(0 ' . $this->operator . ' 0)', $expression);

                    case 'int':
                    case 'uint':
                    case 'long':
                    case 'ulong':
                        return new CompiledExpression(
                            'bool',
                            '(0 ' . $this->operator . ' ' . $right->getCode() . ')',
                            $expression
                        );

                    case 'char':
                    case 'uchar':
                        return new CompiledExpression(
                            'bool',
                            '(\'\\0\' ' . $this->operator . ' \'' . $right->getCode() . '\')',
                            $expression
                        );

                    case 'double':
                        return new CompiledExpression(
                            'bool',
                            '(0 ' . $this->operator . ' (int) ' . $right->getCode() . ')',
                            $expression
                        );

                    case 'variable':
                        $variableRight = $compilationContext->symbolTable->getVariableForRead(
                            $right->getCode(),
                            $compilationContext,
                            $expression['left']
                        );
                        switch ($variableRight->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                $compilationContext->headersManager->add('kernel/operators');

                                return new CompiledExpression(
                                    'bool',
                                    '0 ' . $this->operator . ' ' . $variableRight->getName(),
                                    $expression
                                );

                            case 'variable':
                            case 'mixed':
                            case 'string':
                                $compilationContext->headersManager->add('kernel/operators');
                                $condition = $compilationContext->backend->getTypeofCondition(
                                    $variableRight,
                                    $this->operator,
                                    'null'
                                );

                                return new CompiledExpression('bool', $condition, $expression);

                            default:
                                throw new CompilerException(
                                    'Unknown type: ' . $variableRight->getType(),
                                    $expression['right']
                                );
                        }


                    default:
                        throw new CompilerException('Unknown type: ' . $right->getType(), $expression);
                }


            case 'int':
            case 'uint':
            case 'long':
            case 'double':
            case 'ulong':
            case 'char':
            case 'uchar':
                switch ($right->getType()) {
                    case 'null':
                        return new CompiledExpression('bool', $left->getCode() . ' ' . $this->operator, $expression);

                    case 'int':
                    case 'uint':
                    case 'long':
                    case 'ulong':
                        return new CompiledExpression(
                            'bool',
                            $left->getCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                            $expression
                        );

                    case 'char':
                    case 'uchar':
                        return new CompiledExpression(
                            'bool',
                            $left->getCode() . ' ' . $this->operator . ' \'' . $right->getCode() . '\'',
                            $expression
                        );

                    case 'double':
                        return new CompiledExpression(
                            'bool',
                            $left->getCode() . ' ' . $this->operator . ' (int) ' . $right->getCode(),
                            $expression
                        );

                    case 'variable':
                        $variableRight = $compilationContext->symbolTable->getVariableForRead(
                            $right->getCode(),
                            $compilationContext,
                            $expression['left']
                        );
                        switch ($variableRight->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                $compilationContext->headersManager->add('kernel/operators');

                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                    $expression
                                );

                            case 'variable':
                            case 'mixed':
                                $compilationContext->headersManager->add('kernel/operators');
                                $variableCode = $compilationContext->backend->getVariableCode($variableRight);

                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalLongNegOperator . '(' . $variableCode . ', ' . $left->getCode() . ')',
                                    $expression
                                );

                            default:
                                throw new CompilerException(
                                    'Unknown type: ' . $variableRight->getType(),
                                    $expression['right']
                                );
                        }


                    default:
                        throw new CompilerException(
                            'Cannot compare ' . $left->getType() . ' with ' . $right->getType(),
                            $expression
                        );
                }


            case 'bool':
                switch ($right->getType()) {
                    case 'null':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' 0',
                            $expression
                        );

                    case 'int':
                    case 'uint':
                    case 'long':
                    case 'ulong':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                            $expression
                        );

                    case 'char':
                    case 'uchar':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' \'' . $right->getCode() . '\'',
                            $expression
                        );

                    case 'double':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' (int) ' . $right->getCode(),
                            $expression
                        );

                    case 'bool':
                        return new CompiledExpression(
                            'bool',
                            $left->getBooleanCode() . ' ' . $this->operator . ' ' . $right->getBooleanCode(),
                            $expression
                        );

                    case 'variable':
                        $variableRight = $compilationContext->symbolTable->getVariableForRead(
                            $right->getCode(),
                            $compilationContext,
                            $expression['left']
                        );
                        switch ($variableRight->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                $compilationContext->headersManager->add('kernel/operators');

                                return new CompiledExpression(
                                    'bool',
                                    $left->getBooleanCode() . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                    $expression
                                );

                            case 'variable':
                            case 'mixed':
                                $compilationContext->headersManager->add('kernel/operators');
                                $boolOperator  = '1' == $left->getBooleanCode(
                                ) ? $this->zvalBoolTrueOperator : $this->zvalBoolFalseOperator;
                                $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                return new CompiledExpression(
                                    'bool',
                                    $boolOperator . '(' . $variableRight . ')',
                                    $expression
                                );
                            default:
                                throw new CompilerException(
                                    'Unknown type: ' . $variableRight->getType(),
                                    $expression['right']
                                );
                        }


                    default:
                        throw new CompilerException(
                            'Cannot compare ' . $left->getType() . ' with ' . $right->getType(),
                            $expression
                        );
                }


            case 'string':
                $variableLeft     = $compilationContext->symbolTable->getTempLocalVariableForWrite(
                    'variable',
                    $compilationContext
                );
                $variableLeftCode = $compilationContext->backend->getVariableCode($variableLeft);
                $compilationContext->backend->assignString(
                    $variableLeft,
                    $left->getCode(),
                    $compilationContext,
                    true
                );
                switch ($right->getType()) {
                    case 'null':
                        return new CompiledExpression(
                            'bool',
                            $this->zvalNullOperator . '(' . $variableLeftCode . ')',
                            $expression['left']
                        );


                    case 'string':
                        $compilationContext->headersManager->add('kernel/operators');

                        return new CompiledExpression(
                            'bool',
                            $this->zvalStringOperator . '(' . $variableLeftCode . ', "' . $right->getCode() . '")',
                            $expression['left']
                        );


                    case 'variable':
                        $variableRight = $compilationContext->symbolTable->getVariableForRead(
                            $right->getCode(),
                            $compilationContext,
                            $expression['left']
                        );
                        switch ($variableRight->getType()) {
                            case 'string':
                            case 'variable':
                            case 'mixed':
                                $compilationContext->headersManager->add('kernel/operators');
                                $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalOperator . '(' . $variableLeftCode . ', ' . $variableRight . ')',
                                    $expression
                                );


                            default:
                                throw new CompilerException(
                                    'Unknown type: ' . $variableRight->getType(),
                                    $expression['right']
                                );
                        }


                    default:
                        throw new CompilerException('Unknown type: ' . $right->getType(), $expression['left']);
                }


            case 'variable':
                $variable     = $compilationContext->symbolTable->getVariableForRead(
                    $left->getCode(),
                    $compilationContext,
                    $expression['left']
                );
                $variableCode = $compilationContext->backend->getVariableCode($variable);
                switch ($variable->getType()) {
                    case 'int':
                    case 'uint':
                    case 'long':
                    case 'ulong':
                    case 'char':
                    case 'uchar':
                        switch ($right->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                                    $expression
                                );

                            case 'char':
                            case 'uchar':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' \'' . $right->getCode() . '\'',
                                    $expression
                                );

                            case 'bool':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getBooleanCode(),
                                    $expression
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );
                                switch ($variableRight->getType()) {
                                    case 'int':
                                    case 'uint':
                                    case 'long':
                                    case 'ulong':
                                    case 'char':
                                    case 'uchar':
                                    case 'double':
                                        return new CompiledExpression(
                                            'bool',
                                            $variable->getName(
                                            ) . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                            $expression
                                        );

                                    case 'variable':
                                    case 'mixed':
                                        $compilationContext->headersManager->add('kernel/operators');
                                        $variableRightCode = $compilationContext->backend->getVariableCode(
                                            $variableRight
                                        );
                                        $variableCode      = $compilationContext->backend->getVariableCode($variable);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalLongNegOperator . '(' . $variableRightCode . ', ' . $variableCode . ')',
                                            $expression
                                        );


                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException(
                                    'Cannot compare variable: ' . $variable->getType() . ' with: ' . $right->getType(),
                                    $expression
                                );
                        }


                    case 'double':
                        switch ($right->getType()) {
                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                                    $expression
                                );

                            case 'bool':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getBooleanCode(),
                                    $expression
                                );

                            case 'char':
                            case 'uchar':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' \'' . $right->getCode() . '\'',
                                    $expression
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );
                                switch ($variableRight->getType()) {
                                    case 'int':
                                    case 'uint':
                                    case 'long':
                                    case 'ulong':
                                    case 'double':
                                        return new CompiledExpression(
                                            'bool',
                                            $variable->getName(
                                            ) . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                            $expression
                                        );

                                    case 'variable':
                                    case 'mixed':
                                        $compilationContext->headersManager->add('kernel/operators');
                                        $variableRightCode = $compilationContext->backend->getVariableCode(
                                            $variableRight
                                        );
                                        $variableCode      = $compilationContext->backend->getVariableCode($variable);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalDoubleNegOperator . '(' . $variableRightCode . ', ' . $variableCode . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException(
                                    'Cannot compare variable: ' . $variable->getType() . ' with: ' . $right->getType(),
                                    $expression
                                );
                        }


                    case 'bool':
                        switch ($right->getType()) {
                            case 'int':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getCode(),
                                    $expression['left']
                                );

                            case 'bool':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' ' . $right->getBooleanCode(),
                                    $expression['left']
                                );

                            case 'null':
                                return new CompiledExpression(
                                    'bool',
                                    $left->getCode() . ' ' . $this->operator . ' 0',
                                    $expression['left']
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );

                                switch ($variableRight->getType()) {
                                    case 'int':
                                    case 'uint':
                                    case 'long':
                                    case 'ulong':
                                    case 'bool':
                                    case 'double':
                                        return new CompiledExpression(
                                            'bool',
                                            $variable->getName(
                                            ) . ' ' . $this->operator . ' ' . $variableRight->getName(),
                                            $expression
                                        );

                                    case 'variable':
                                    case 'mixed':
                                        $compilationContext->headersManager->add('kernel/operators');
                                        $boolOperator      = '1' == $left->getBooleanCode(
                                        ) ? $this->zvalBoolTrueOperator : $this->zvalBoolFalseOperator;
                                        $variableRightCode = $compilationContext->backend->getVariableCode(
                                            $variableRight
                                        );

                                        return new CompiledExpression(
                                            'bool',
                                            $boolOperator . '(' . $variableRightCode . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException(
                                    'Cannot compare variable: ' . $variable->getType() . ' with: ' . $right->getType(),
                                    $expression
                                );
                        }


                    case 'array':
                        switch ($right->getType()) {
                            case 'null':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalNullOperator . '(' . $variableCode . ')',
                                    $expression['left']
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );
                                switch ($variableRight->getType()) {
                                    case 'string':
                                    case 'variable':
                                    case 'mixed':
                                    case 'array':
                                        $compilationContext->headersManager->add('kernel/operators');
                                        $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalOperator . '(' . $variableCode . ', ' . $variableRight . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException('Unknown type: ' . $right->getType(), $expression['left']);
                        }


                    case 'string':
                        $compilationContext->headersManager->add('kernel/operators');

                        switch ($right->getType()) {
                            case 'null':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalNullOperator . '(' . $variableCode . ')',
                                    $expression['left']
                                );

                            case 'string':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalStringOperator . '(' . $variableCode . ', "' . $right->getCode() . '")',
                                    $expression['left']
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );

                                switch ($variableRight->getType()) {
                                    case 'string':
                                    case 'variable':
                                    case 'mixed':
                                        $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalOperator . '(' . $variableCode . ', ' . $variableRight . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException('Unknown type: ' . $right->getType(), $expression['left']);
                        }


                    case 'variable':
                    case 'mixed':
                        $compilationContext->headersManager->add('kernel/operators');

                        switch ($right->getType()) {
                            case 'null':
                                $condition = $compilationContext->backend->getTypeofCondition(
                                    $variable,
                                    $this->operator,
                                    'null'
                                );

                                return new CompiledExpression('bool', $condition, $expression['left']);

                            case 'int':
                            case 'uint':
                            case 'long':
                            case 'ulong':
                            case 'double':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalLongOperator . '(' . $variableCode . ', ' . $right->getCode() . ')',
                                    $expression['left']
                                );

                            case 'char':
                            case 'uchar':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalLongOperator . '(' . $variableCode . ', \'' . $right->getCode() . '\')',
                                    $expression['left']
                                );

                            case 'bool':
                                $zvalBoolOperator = 'true' === $right->getCode(
                                ) ? $this->zvalBoolTrueOperator : $this->zvalBoolFalseOperator;

                                return new CompiledExpression(
                                    'bool',
                                    $zvalBoolOperator . '(' . $variableCode . ')',
                                    $expression['left']
                                );

                            case 'string':
                                return new CompiledExpression(
                                    'bool',
                                    $this->zvalStringOperator . '(' . $variableCode . ', "' . $right->getCode() . '")',
                                    $expression['left']
                                );

                            case 'variable':
                                $variableRight = $compilationContext->symbolTable->getVariableForRead(
                                    $right->getCode(),
                                    $compilationContext,
                                    $expression['left']
                                );
                                switch ($variableRight->getType()) {
                                    case 'int':
                                    case 'uint':
                                    case 'long':
                                    case 'ulong':
                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalLongOperator . '(' . $variableCode . ', ' . $variableRight->getName(
                                            ) . ')',
                                            $expression
                                        );

                                    case 'double':
                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalDoubleOperator . '(' . $variableCode . ', ' . $variableRight->getName(
                                            ) . ')',
                                            $expression
                                        );

                                    case 'bool':
                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalBoolOperator . '(' . $variableCode . ', ' . $variableRight->getName(
                                            ) . ')',
                                            $expression
                                        );

                                    case 'string':
                                    case 'variable':
                                    case 'mixed':
                                    case 'array':
                                        $variableRight = $compilationContext->backend->getVariableCode($variableRight);

                                        return new CompiledExpression(
                                            'bool',
                                            $this->zvalOperator . '(' . $variableCode . ', ' . $variableRight . ')',
                                            $expression
                                        );

                                    default:
                                        throw new CompilerException(
                                            'Unknown type: ' . $variableRight->getType(),
                                            $expression['right']
                                        );
                                }


                            default:
                                throw new CompilerException('Unknown type: ' . $right->getType(), $expression['left']);
                        }


                    default:
                        throw new CompilerException('Unknown type: ' . $variable->getType(), $expression);
                }


            default:
                throw new CompilerException('Unknown type: ' . $left->getType(), $expression);
        }
    }

    /**
     * @param array              $expr
     * @param CompilationContext $compilationContext
     *
     * @return CompiledExpression|null
     *
     * @throws Exception
     * @throws ReflectionException
     */
    public function optimizeTypeOf(array $expr, CompilationContext $compilationContext): ?CompiledExpression
    {
        if (!isset($expr['left'])) {
            return null;
        }

        if (!isset($expr['right']) && !isset($expr['right']['value'])) {
            return null;
        }

        if ('typeof' !== $expr['left']['type']) {
            return null;
        }

        if ('string' !== $expr['right']['type']) {
            $compilationContext->logger->warning(
                "Possible invalid comparison for 'typeof' operator with non-string",
                ['invalid-typeof-comparison', $expr['right']]
            );

            return null;
        }

        switch ($expr['type']) {
            case 'identical':
            case 'equals':
                $operator = '==';
                break;

            case 'not-identical':
            case 'not-equals':
                $operator = '!=';
                break;

            default:
                return null;
        }

        $code             = (new Expression($expr['left']['left']))->compile($compilationContext)->getCode();
        $variableVariable = $compilationContext->symbolTable->getVariableForRead($code, $compilationContext, $expr);

        if ('string' !== $expr['right']['type']) {
            throw new CompilerException('Right expression of typeof operator must be "string" type', $expr['right']);
        }

        $value = strtolower($expr['right']['value']);

        switch ($variableVariable->getType()) {
            case 'double':
                $condition = match ($value) {
                    'double', 'float' => '1 ' . $operator . ' 1',
                    default           => '1 ' . $operator . ' 0',
                };
                break;

            case 'int':
            case 'integer':
            case 'long':
                $condition = match ($value) {
                    'int', 'integer', 'long' => '1 ' . $operator . ' 1',
                    default                  => '1 ' . $operator . ' 0',
                };
                break;

            case 'bool':
                $condition = match ($value) {
                    'bool', 'boolean' => '1 ' . $operator . ' 1',
                    default           => '1 ' . $operator . ' 0',
                };
                break;

            case 'array':
                $condition = match ($value) {
                    'array' => '1 ' . $operator . ' 1',
                    default => '1 ' . $operator . ' 0',
                };
                break;

            case 'string':
                $condition = match ($value) {
                    'string' => '1 ' . $operator . ' 1',
                    default  => '1 ' . $operator . ' 0',
                };
                break;

            case 'variable':
                $condition = $compilationContext->backend->getTypeofCondition($variableVariable, $operator, $value);
                break;

            default:
                return null;
        }

        return new CompiledExpression('bool', $condition, $expr);
    }
}

Function Calls

None

Variables

None

Stats

MD5 96c771a09dc6bc6046dbcb14c8bde230
Eval Count 0
Decode Time 101 ms