diff --git a/CHANGELOG b/CHANGELOG index 5b3fa9eb5a0..cf6ae38ae81 100644 --- a/CHANGELOG +++ b/CHANGELOG @@ -1,6 +1,6 @@ # 3.17.0 (2024-XX-XX) - * n/a + * Support underscores in number literals # 3.16.0 (2024-11-29) diff --git a/doc/templates.rst b/doc/templates.rst index 2f2775ab707..4235e6e9b51 100644 --- a/doc/templates.rst +++ b/doc/templates.rst @@ -612,7 +612,8 @@ exist: * ``42`` / ``42.23``: Integers and floating point numbers are created by writing the number down. If a dot is present the number is a float, - otherwise an integer. + otherwise an integer. Underscores can be used as digits separator to + improve readability (``-3_141.592_65`` is equivalent to ``-3141.59265``). * ``["first_name", "last_name"]``: Sequences are defined by a sequence of expressions separated by a comma (``,``) and wrapped with squared brackets (``[]``). @@ -1144,4 +1145,4 @@ Twig can be extended. If you want to create your own extensions, read the .. _`Modern Twig`: https://marketplace.visualstudio.com/items?itemName=Stanislav.vscode-twig .. _`Twig Language Server`: https://github.com/kaermorchen/twig-language-server/tree/master/packages/language-server .. _`Twiggy`: https://marketplace.visualstudio.com/items?itemName=moetelo.twiggy -.. _`PHP spaceship operator documentation`: https://www.php.net/manual/en/language.operators.comparison.php \ No newline at end of file +.. _`PHP spaceship operator documentation`: https://www.php.net/manual/en/language.operators.comparison.php diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index 1121ae1b235..eabe060f2d6 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -23,3 +23,9 @@ parameters: identifier: parameter.phpDocType count: 5 path: src/Node/Node.php + + - # Adding 0 to the string representation of a number is valid and what we want here + message: '#^Binary operation "\+" between 0 and string results in an error\.$#' + identifier: binaryOp.invalid + count: 1 + path: src/Lexer.php diff --git a/src/Lexer.php b/src/Lexer.php index 4983313cf03..0338fd874ff 100644 --- a/src/Lexer.php +++ b/src/Lexer.php @@ -44,8 +44,16 @@ class Lexer public const STATE_INTERPOLATION = 4; public const REGEX_NAME = '/[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*/A'; - public const REGEX_NUMBER = '/[0-9]+(?:\.[0-9]+)?([Ee][\+\-][0-9]+)?/A'; public const REGEX_STRING = '/"([^#"\\\\]*(?:\\\\.[^#"\\\\]*)*)"|\'([^\'\\\\]*(?:\\\\.[^\'\\\\]*)*)\'/As'; + + public const REGEX_NUMBER = '/(?(DEFINE) + (?[0-9]+(_[0-9]+)*) # Integers (with underscores) 123_456 + (?\.(?&LNUM)) # Fractional part .456 + (?[eE][+-]?(?&LNUM)) # Exponent part E+10 + (?(?&LNUM)(?:(?&FRAC))?) # Decimal number 123_456.456 + )(?:(?&DNUM)(?:(?&EXPONENT))?) # 123_456.456E+10 + /Ax'; + public const REGEX_DQ_STRING_DELIM = '/"/A'; public const REGEX_DQ_STRING_PART = '/[^#"\\\\]*(?:(?:\\\\.|#(?!\{))[^#"\\\\]*)*/As'; public const REGEX_INLINE_COMMENT = '/#[^\n]*/A'; @@ -346,11 +354,7 @@ private function lexExpression(): void } // numbers elseif (preg_match(self::REGEX_NUMBER, $this->code, $match, 0, $this->cursor)) { - $number = (float) $match[0]; // floats - if (ctype_digit($match[0]) && $number <= \PHP_INT_MAX) { - $number = (int) $match[0]; // integers lower than the maximum - } - $this->pushToken(Token::NUMBER_TYPE, $number); + $this->pushToken(Token::NUMBER_TYPE, 0 + str_replace('_', '', $match[0])); $this->moveCursor($match[0]); } // punctuation diff --git a/tests/Fixtures/expressions/underscored_numbers.test b/tests/Fixtures/expressions/underscored_numbers.test new file mode 100644 index 00000000000..4a163ccb518 --- /dev/null +++ b/tests/Fixtures/expressions/underscored_numbers.test @@ -0,0 +1,24 @@ +--TEST-- +Twig compile numbers literals with underscores correctly +--TEMPLATE-- +{{ 0_0 is same as 0 ? 'ok' : 'ko' }} +{{ 1_23 is same as 123 ? 'ok' : 'ko' }} +{{ 12_3 is same as 123 ? 'ok' : 'ko' }} +{{ 1_2_3 is same as 123 ? 'ok' : 'ko' }} +{{ -1_2 is same as -12 ? 'ok' : 'ko' }} +{{ 1_2.3_4 is same as 12.34 ? 'ok' : 'ko' }} +{{ -1_2.3_4 is same as -12.34 ? 'ok' : 'ko' }} +{{ 1.2_3e-4 is same as 1.23e-4 ? 'ok' : 'ko' }} +{{ -1.2_3e+4 is same as -1.23e+4 ? 'ok' : 'ko' }} +--DATA-- +return [] +--EXPECT-- +ok +ok +ok +ok +ok +ok +ok +ok +ok diff --git a/tests/Fixtures/expressions/underscored_numbers_error.test b/tests/Fixtures/expressions/underscored_numbers_error.test new file mode 100644 index 00000000000..839d606f128 --- /dev/null +++ b/tests/Fixtures/expressions/underscored_numbers_error.test @@ -0,0 +1,8 @@ +--TEST-- +Twig does not allow to use 2 underscored between digits in numbers +--TEMPLATE-- +{{ 1__2 }} +--DATA-- +return [] +--EXCEPTION-- +Twig\Error\SyntaxError: Unexpected token "name" of value "__2" ("end of print statement" expected) in "index.twig" at line 2.