diff --git a/.gitattributes b/.gitattributes index 1daa7443..984945c7 100644 --- a/.gitattributes +++ b/.gitattributes @@ -13,5 +13,6 @@ tmp export-ignore build-abnfgen.sh export-ignore CODE_OF_CONDUCT.md export-ignore Makefile export-ignore +phpstan-baseline.neon export-ignore phpstan.neon export-ignore phpunit.xml export-ignore diff --git a/.github/workflows/apiref.yml b/.github/workflows/apiref.yml index 4b3abbb0..27d3fb92 100644 --- a/.github/workflows/apiref.yml +++ b/.github/workflows/apiref.yml @@ -5,7 +5,7 @@ name: "Deploy API Reference" on: push: branches: - - "1.22.x" + - "2.1.x" concurrency: group: "pages" @@ -16,9 +16,18 @@ jobs: name: "Build API References Pages" runs-on: "ubuntu-latest" + strategy: + matrix: + branch: + - "1.23.x" + - "2.0.x" + - "2.1.x" + steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 + with: + ref: ${{ matrix.branch }} - name: "Install PHP" uses: "shivammathur/setup-php@v2" @@ -36,18 +45,36 @@ jobs: run: "composer install --no-interaction --no-progress" - name: "Run ApiGen" - run: "apigen/vendor/bin/apigen -c apigen/apigen.neon --output docs -- src" + run: "apigen/vendor/bin/apigen -c apigen/apigen.neon --output docs/${{ matrix.branch }} -- src" - name: "Copy favicon" run: "cp apigen/favicon.png docs/favicon.png" + - uses: actions/upload-artifact@v4 + with: + name: docs-${{ matrix.branch }} + path: docs/* + + merge: + name: "Merge docs" + needs: build + + runs-on: "ubuntu-latest" + + steps: + - uses: actions/download-artifact@v4 + with: + pattern: docs-* + path: docs + merge-multiple: true + - name: Upload artifact - uses: actions/upload-pages-artifact@v1 + uses: actions/upload-pages-artifact@v3 with: path: 'docs' deploy: - needs: build + needs: merge # from https://github.com/actions/deploy-pages @@ -63,4 +90,4 @@ jobs: steps: - name: Deploy to GitHub Pages id: deployment - uses: actions/deploy-pages@v2 + uses: actions/deploy-pages@v4 diff --git a/.github/workflows/backward-compatibility.yml b/.github/workflows/backward-compatibility.yml index 1cd62723..ecfb7fbd 100644 --- a/.github/workflows/backward-compatibility.yml +++ b/.github/workflows/backward-compatibility.yml @@ -6,7 +6,7 @@ on: pull_request: push: branches: - - "1.21.x" + - "2.1.x" jobs: backward-compatibility: @@ -17,7 +17,7 @@ jobs: steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 with: fetch-depth: 0 @@ -25,14 +25,11 @@ jobs: uses: "shivammathur/setup-php@v2" with: coverage: "none" - php-version: "7.4" + php-version: "8.3" - name: "Install dependencies" run: "composer install --no-dev --no-interaction --no-progress --no-suggest" - - name: "allow composer plugins" - run: "composer config --no-plugins --global allow-plugins.ocramius/package-versions true" - - name: "Install BackwardCompatibilityCheck" run: "composer global require --dev roave/backward-compatibility-check" diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 3342b55f..9118fafd 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -6,7 +6,7 @@ on: pull_request: push: branches: - - "1.21.x" + - "2.1.x" jobs: lint: @@ -16,16 +16,16 @@ jobs: strategy: matrix: php-version: - - "7.2" - - "7.3" - "7.4" - "8.0" - "8.1" - "8.2" + - "8.3" + - "8.4" steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: "Install PHP" uses: "shivammathur/setup-php@v2" @@ -39,10 +39,6 @@ jobs: - name: "Install dependencies" run: "composer install --no-interaction --no-progress" - - name: "Downgrade PHPUnit" - if: matrix.php-version == '7.2' || matrix.php-version == '7.3' - run: "composer require --dev phpunit/phpunit:^7.5.20 --update-with-dependencies" - - name: "Lint" run: "make lint" @@ -53,13 +49,14 @@ jobs: steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: "Checkout build-cs" - uses: actions/checkout@v3 + uses: actions/checkout@v4 with: repository: "phpstan/build-cs" path: "build-cs" + ref: "2.x" - name: "Install PHP" uses: "shivammathur/setup-php@v2" @@ -85,25 +82,26 @@ jobs: tests: name: "Tests" - runs-on: "ubuntu-latest" + runs-on: ${{ matrix.operating-system }} strategy: fail-fast: false matrix: + operating-system: [ubuntu-latest, windows-latest] php-version: - - "7.2" - - "7.3" - "7.4" - "8.0" - "8.1" - "8.2" + - "8.3" + - "8.4" dependencies: - "lowest" - "highest" steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: "Install PHP" uses: "shivammathur/setup-php@v2" @@ -119,10 +117,6 @@ jobs: if: ${{ matrix.dependencies == 'highest' }} run: "composer update --no-interaction --no-progress" - - name: "Downgrade PHPUnit" - if: matrix.php-version == '7.2' || matrix.php-version == '7.3' - run: "composer require --dev phpunit/phpunit:^7.5.20 --update-with-dependencies" - - name: "Tests" run: "make tests" @@ -134,16 +128,16 @@ jobs: fail-fast: false matrix: php-version: - - "7.2" - - "7.3" - "7.4" - "8.0" - "8.1" - "8.2" + - "8.3" + - "8.4" steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: "Install PHP" uses: "shivammathur/setup-php@v2" @@ -156,9 +150,5 @@ jobs: - name: "Install dependencies" run: "composer update --no-interaction --no-progress" - - name: "Downgrade PHPUnit" - if: matrix.php-version == '7.2' || matrix.php-version == '7.3' - run: "composer require --dev phpunit/phpunit:^7.5.20 --update-with-dependencies" - - name: "PHPStan" run: "make phpstan" diff --git a/.github/workflows/create-tag.yml b/.github/workflows/create-tag.yml index 8452d986..a8535014 100644 --- a/.github/workflows/create-tag.yml +++ b/.github/workflows/create-tag.yml @@ -21,7 +21,7 @@ jobs: runs-on: "ubuntu-latest" steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 with: fetch-depth: 0 token: ${{ secrets.PHPSTAN_BOT_TOKEN }} diff --git a/.github/workflows/lock-closed-issues.yml b/.github/workflows/lock-closed-issues.yml index 4c7990df..69545301 100644 --- a/.github/workflows/lock-closed-issues.yml +++ b/.github/workflows/lock-closed-issues.yml @@ -2,13 +2,13 @@ name: 'Lock Issues' on: schedule: - - cron: '0 0 * * *' + - cron: '8 0 * * *' jobs: lock: runs-on: ubuntu-latest steps: - - uses: dessant/lock-threads@v4 + - uses: dessant/lock-threads@v5 with: github-token: ${{ github.token }} issue-inactive-days: '31' diff --git a/.github/workflows/merge-maintained-branch.yml b/.github/workflows/merge-maintained-branch.yml index 6ea17262..18d17974 100644 --- a/.github/workflows/merge-maintained-branch.yml +++ b/.github/workflows/merge-maintained-branch.yml @@ -5,7 +5,7 @@ name: Merge maintained branch on: push: branches: - - "1.21.x" + - "1.22.x" jobs: merge: @@ -13,11 +13,11 @@ jobs: runs-on: ubuntu-latest steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: "Merge branch" uses: everlytic/branch-merge@1.1.5 with: github_token: "${{ secrets.PHPSTAN_BOT_TOKEN }}" source_ref: ${{ github.ref }} - target_branch: '1.22.x' + target_branch: '1.23.x' commit_message_template: 'Merge branch {source_ref} into {target_branch}' diff --git a/.github/workflows/release-toot.yml b/.github/workflows/release-toot.yml index 6a1c8156..1ba4fd77 100644 --- a/.github/workflows/release-toot.yml +++ b/.github/workflows/release-toot.yml @@ -10,7 +10,7 @@ jobs: toot: runs-on: ubuntu-latest steps: - - uses: cbrgm/mastodon-github-action@v1 + - uses: cbrgm/mastodon-github-action@v2 if: ${{ !github.event.repository.private }} with: # GitHub event payload diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index 92b72547..be6cad08 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -14,11 +14,11 @@ jobs: steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Generate changelog id: changelog - uses: metcalfc/changelog-generator@v4.1.0 + uses: metcalfc/changelog-generator@v4.5.0 with: myToken: ${{ secrets.PHPSTAN_BOT_TOKEN }} diff --git a/.github/workflows/send-pr.yml b/.github/workflows/send-pr.yml index bc305f97..2503256e 100644 --- a/.github/workflows/send-pr.yml +++ b/.github/workflows/send-pr.yml @@ -18,12 +18,12 @@ jobs: php-version: "8.1" - name: "Checkout phpstan-src" - uses: actions/checkout@v3 + uses: actions/checkout@v4 with: repository: phpstan/phpstan-src path: phpstan-src token: ${{ secrets.PHPSTAN_BOT_TOKEN }} - ref: 1.10.x + ref: 2.1.x - name: "Install dependencies" working-directory: ./phpstan-src @@ -35,7 +35,7 @@ jobs: - name: "Create Pull Request" id: create-pr - uses: peter-evans/create-pull-request@v5 + uses: peter-evans/create-pull-request@v6 with: token: ${{ secrets.PHPSTAN_BOT_TOKEN }} path: ./phpstan-src diff --git a/.github/workflows/test-slevomat-coding-standard.yml b/.github/workflows/test-slevomat-coding-standard.yml index ab90c303..8ce55469 100644 --- a/.github/workflows/test-slevomat-coding-standard.yml +++ b/.github/workflows/test-slevomat-coding-standard.yml @@ -6,7 +6,7 @@ on: pull_request: push: branches: - - "1.21.x" + - "2.1.x" jobs: tests: @@ -17,22 +17,23 @@ jobs: fail-fast: false matrix: php-version: - - "7.3" - "7.4" - "8.0" - "8.1" - "8.2" + - "8.3" + - "8.4" steps: - name: "Checkout" - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: "Checkout Slevomat Coding Standard" - uses: actions/checkout@v3 + uses: actions/checkout@v4 with: repository: slevomat/coding-standard path: slevomat-cs - ref: 710c256bf3f0f696ec8d4f9d2218321c3eb0f7d2 + ref: 88602f9ae5450a933133108d052eeb2edee0a4b7 - name: "Install PHP" uses: "shivammathur/setup-php@v2" diff --git a/LICENSE b/LICENSE index 98a854e4..e5f34e60 100644 --- a/LICENSE +++ b/LICENSE @@ -1,6 +1,7 @@ MIT License Copyright (c) 2016 Ondřej Mirtes +Copyright (c) 2025 PHPStan s.r.o. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal @@ -18,4 +19,4 @@ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -SOFTWARE. \ No newline at end of file +SOFTWARE. diff --git a/Makefile b/Makefile index 843b6679..294ad0c5 100644 --- a/Makefile +++ b/Makefile @@ -2,7 +2,7 @@ check: lint cs tests phpstan .PHONY: tests -tests: build-abnfgen +tests: php vendor/bin/phpunit .PHONY: lint @@ -16,7 +16,7 @@ lint: .PHONY: cs-install cs-install: git clone https://github.com/phpstan/build-cs.git || true - git -C build-cs fetch origin && git -C build-cs reset --hard origin/main + git -C build-cs fetch origin && git -C build-cs reset --hard origin/2.x composer install --working-dir build-cs .PHONY: cs @@ -34,7 +34,3 @@ phpstan: .PHONY: phpstan-generate-baseline phpstan-generate-baseline: php vendor/bin/phpstan --generate-baseline - -.PHONY: build-abnfgen -build-abnfgen: - ./build-abnfgen.sh diff --git a/README.md b/README.md index 67312fd9..15ac9a97 100644 --- a/README.md +++ b/README.md @@ -13,7 +13,9 @@ For the complete list of supported PHPDoc features check out PHPStan documentati * [PHPDoc Basics](https://phpstan.org/writing-php-code/phpdocs-basics) (list of PHPDoc tags) * [PHPDoc Types](https://phpstan.org/writing-php-code/phpdoc-types) (list of PHPDoc types) -* [phpdoc-parser API Reference](https://phpstan.github.io/phpdoc-parser/namespace-PHPStan.PhpDocParser.html) with all the AST node types etc. +* [phpdoc-parser API Reference](https://phpstan.github.io/phpdoc-parser/2.1.x/namespace-PHPStan.PhpDocParser.html) with all the AST node types etc. + +This parser also supports parsing [Doctrine Annotations](https://github.com/doctrine/annotations). The AST nodes live in the [PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine namespace](https://phpstan.github.io/phpdoc-parser/2.1.x/namespace-PHPStan.PhpDocParser.Ast.PhpDoc.Doctrine.html). ## Installation @@ -32,6 +34,7 @@ use PHPStan\PhpDocParser\Ast\PhpDoc\ParamTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocNode; use PHPStan\PhpDocParser\Ast\Type\IdentifierTypeNode; use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\ParserConfig; use PHPStan\PhpDocParser\Parser\ConstExprParser; use PHPStan\PhpDocParser\Parser\PhpDocParser; use PHPStan\PhpDocParser\Parser\TokenIterator; @@ -39,10 +42,11 @@ use PHPStan\PhpDocParser\Parser\TypeParser; // basic setup -$lexer = new Lexer(); -$constExprParser = new ConstExprParser(); -$typeParser = new TypeParser($constExprParser); -$phpDocParser = new PhpDocParser($typeParser, $constExprParser); +$config = new ParserConfig(usedAttributes: []); +$lexer = new Lexer($config); +$constExprParser = new ConstExprParser($config); +$typeParser = new TypeParser($config, $constExprParser); +$phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); // parsing and reading a PHPDoc string @@ -70,6 +74,7 @@ use PHPStan\PhpDocParser\Ast\NodeVisitor\CloningVisitor; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocNode; use PHPStan\PhpDocParser\Ast\Type\IdentifierTypeNode; use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\ParserConfig; use PHPStan\PhpDocParser\Parser\ConstExprParser; use PHPStan\PhpDocParser\Parser\PhpDocParser; use PHPStan\PhpDocParser\Parser\TokenIterator; @@ -78,12 +83,11 @@ use PHPStan\PhpDocParser\Printer\Printer; // basic setup with enabled required lexer attributes -$usedAttributes = ['lines' => true, 'indexes' => true]; - -$lexer = new Lexer(); -$constExprParser = new ConstExprParser(true, true, $usedAttributes); -$typeParser = new TypeParser($constExprParser, true, $usedAttributes); -$phpDocParser = new PhpDocParser($typeParser, $constExprParser, true, true, $usedAttributes); +$config = new ParserConfig(usedAttributes: ['lines' => true, 'indexes' => true, 'comments' => true]); +$lexer = new Lexer($config); +$constExprParser = new ConstExprParser($config); +$typeParser = new TypeParser($config, $constExprParser); +$phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); $tokens = new TokenIterator($lexer->tokenize('/** @param Lorem $a */')); $phpDocNode = $phpDocParser->parse($tokens); // PhpDocNode diff --git a/UPGRADING.md b/UPGRADING.md new file mode 100644 index 00000000..b673bccf --- /dev/null +++ b/UPGRADING.md @@ -0,0 +1,129 @@ +Upgrading from phpstan/phpdoc-parser 1.x to 2.0 +================================= + +### PHP version requirements + +phpstan/phpdoc-parser now requires PHP 7.4 or newer to run. + +### Changed constructors of parser classes + +Instead of different arrays and boolean values passed into class constructors during setup, parser classes now share a common ParserConfig object. + +Before: + +```php +use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\Parser\ConstExprParser; +use PHPStan\PhpDocParser\Parser\TypeParser; +use PHPStan\PhpDocParser\Parser\PhpDocParser; + +$usedAttributes = ['lines' => true, 'indexes' => true]; + +$lexer = new Lexer(); +$constExprParser = new ConstExprParser(true, true, $usedAttributes); +$typeParser = new TypeParser($constExprParser, true, $usedAttributes); +$phpDocParser = new PhpDocParser($typeParser, $constExprParser, true, true, $usedAttributes); +``` + +After: + +```php +use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\ParserConfig; +use PHPStan\PhpDocParser\Parser\ConstExprParser; +use PHPStan\PhpDocParser\Parser\TypeParser; +use PHPStan\PhpDocParser\Parser\PhpDocParser; + +$config = new ParserConfig(usedAttributes: ['lines' => true, 'indexes' => true]); +$lexer = new Lexer($config); +$constExprParser = new ConstExprParser($config); +$typeParser = new TypeParser($config, $constExprParser); +$phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); +``` + +The point of ParserConfig is that over the course of phpstan/phpdoc-parser 2.x development series it's most likely going to gain new optional parameters akin to PHPStan's [bleeding edge](https://phpstan.org/blog/what-is-bleeding-edge). These parameters will allow opting in to new behaviour which will become the default in 3.0. + +With ParserConfig object, it's now going to be impossible to configure parser classes inconsistently. Which [happened to users](https://github.com/phpstan/phpdoc-parser/issues/251#issuecomment-2333927959) when they were separate boolean values. + +### Support for parsing Doctrine annotations + +This parser now supports parsing [Doctrine Annotations](https://github.com/doctrine/annotations). The AST nodes representing Doctrine Annotations live in the [PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine namespace](https://phpstan.github.io/phpdoc-parser/2.0.x/namespace-PHPStan.PhpDocParser.Ast.PhpDoc.Doctrine.html). + +### Whitespace before description is required + +phpdoc-parser 1.x sometimes silently consumed invalid part of a PHPDoc type as description: + +```php +/** @return \Closure(...int, string): string */ +``` + +This became `IdentifierTypeNode` of `\Closure` and with `(...int, string): string` as description. (Valid callable syntax is: `\Closure(int ...$u, string): string`.) + +Another example: + +```php +/** @return array{foo: int}} */ +``` + +The extra `}` also became description. + +Both of these examples are now InvalidTagValueNode. + +If these parts are supposed to be PHPDoc descriptions, you need to put whitespace between the type and the description text: + +```php +/** @return \Closure (...int, string): string */ +/** @return array{foo: int} } */ +``` + +### Type aliases with invalid types are preserved + +In phpdoc-parser 1.x, invalid type alias syntax was represented as [`InvalidTagValueNode`](https://phpstan.github.io/phpdoc-parser/2.0.x/PHPStan.PhpDocParser.Ast.PhpDoc.InvalidTagValueNode.html), losing information about a type alias being present. + +```php +/** + * @phpstan-type TypeAlias + */ +``` + +This `@phpstan-type` is missing the actual type to alias. In phpdoc-parser 2.0 this is now represented as [`TypeAliasTagValueNode`](https://phpstan.github.io/phpdoc-parser/2.0.x/PHPStan.PhpDocParser.Ast.PhpDoc.TypeAliasTagValueNode.html) (instead of `InvalidTagValueNode`) with [`InvalidTypeNode`](https://phpstan.github.io/phpdoc-parser/2.0.x/PHPStan.PhpDocParser.Ast.Type.InvalidTypeNode.html) in place of the type. + +### Removal of QuoteAwareConstExprStringNode + +The class [QuoteAwareConstExprStringNode](https://phpstan.github.io/phpdoc-parser/1.23.x/PHPStan.PhpDocParser.Ast.ConstExpr.QuoteAwareConstExprStringNode.html) has been removed. + +Instead, [ConstExprStringNode](https://phpstan.github.io/phpdoc-parser/2.0.x/PHPStan.PhpDocParser.Ast.ConstExpr.ConstExprStringNode.html) gained information about the kind of quotes being used. + +### Removed 2nd parameter of `ConstExprParser::parse()` (`$trimStrings`) + +`ConstExprStringNode::$value` now contains unescaped values without surrounding `''` or `""` quotes. + +Use `ConstExprStringNode::__toString()` or [`Printer`](https://phpstan.github.io/phpdoc-parser/2.0.x/PHPStan.PhpDocParser.Printer.Printer.html) to get the escaped value along with surrounding quotes. + +### Text between tags always belongs to description + +Multi-line descriptions between tags were previously represented as separate [PhpDocTextNode](https://phpstan.github.io/phpdoc-parser/2.0.x/PHPStan.PhpDocParser.Ast.PhpDoc.PhpDocTextNode.html): + +```php +/** + * @param Foo $foo 1st multi world description + * some text in the middle + * @param Bar $bar 2nd multi world description + */ +``` + +The line with `some text in the middle` in phpdoc-parser 2.0 is now part of the description of the first `@param` tag. + +### `ArrayShapeNode` construction changes + +`ArrayShapeNode` constructor made private, added public static methods `createSealed()` and `createUnsealed()`. + +### Minor BC breaks + +* Constructor parameter `$isEquality` in `AssertTag*ValueNode` made required +* Constructor parameter `$templateTypes` in `MethodTagValueNode` made required +* Constructor parameter `$isReference` in `ParamTagValueNode` made required +* Constructor parameter `$isReference` in `TypelessParamTagValueNode` made required +* Constructor parameter `$templateTypes` in `CallableTypeNode` made required +* Constructor parameters `$expectedTokenValue` and `$currentTokenLine` in `ParserException` made required +* `ArrayShapeItemNode` and `ObjectShapeItemNode` are not standalone TypeNode, just Node diff --git a/apigen/theme/blocks/head.latte b/apigen/theme/blocks/head.latte index e7f54525..6077db0a 100644 --- a/apigen/theme/blocks/head.latte +++ b/apigen/theme/blocks/head.latte @@ -1,4 +1,4 @@ {define head} - + {/define} diff --git a/composer.json b/composer.json index 30b879b7..8047c49e 100644 --- a/composer.json +++ b/composer.json @@ -3,16 +3,17 @@ "description": "PHPDoc parser with support for nullable, intersection and generic types", "license": "MIT", "require": { - "php": "^7.2 || ^8.0" + "php": "^7.4 || ^8.0" }, "require-dev": { - "nikic/php-parser": "^4.15", + "doctrine/annotations": "^2.0", + "nikic/php-parser": "^5.3.0", "php-parallel-lint/php-parallel-lint": "^1.2", "phpstan/extension-installer": "^1.0", - "phpstan/phpstan": "^1.5", - "phpstan/phpstan-phpunit": "^1.1", - "phpstan/phpstan-strict-rules": "^1.0", - "phpunit/phpunit": "^9.5", + "phpstan/phpstan": "^2.0", + "phpstan/phpstan-phpunit": "^2.0", + "phpstan/phpstan-strict-rules": "^2.0", + "phpunit/phpunit": "^9.6", "symfony/process": "^5.2" }, "config": { diff --git a/doc/grammars/type.abnf b/doc/grammars/type.abnf index f4bd3c6a..ad955a9b 100644 --- a/doc/grammars/type.abnf +++ b/doc/grammars/type.abnf @@ -35,7 +35,13 @@ GenericTypeArgument / TokenWildcard Callable - = TokenParenthesesOpen [CallableParameters] TokenParenthesesClose TokenColon CallableReturnType + = [CallableTemplate] TokenParenthesesOpen [CallableParameters] TokenParenthesesClose TokenColon CallableReturnType + +CallableTemplate + = TokenAngleBracketOpen CallableTemplateArgument *(TokenComma CallableTemplateArgument) TokenAngleBracketClose + +CallableTemplateArgument + = TokenIdentifier [1*ByteHorizontalWs TokenOf Type] [1*ByteHorizontalWs TokenSuper Type] ["=" Type] CallableParameters = CallableParameter *(TokenComma CallableParameter) @@ -85,18 +91,18 @@ ConstantExpr / ConstantFetch *ByteHorizontalWs ConstantFloat - = ["-"] 1*ByteDecDigit *("_" 1*ByteDecDigit) "." [1*ByteDecDigit *("_" 1*ByteDecDigit)] [ConstantFloatExp] - / ["-"] 1*ByteDecDigit *("_" 1*ByteDecDigit) ConstantFloatExp - / ["-"] "." 1*ByteDecDigit *("_" 1*ByteDecDigit) [ConstantFloatExp] + = [ByteNumberSign] 1*ByteDecDigit *("_" 1*ByteDecDigit) "." [1*ByteDecDigit *("_" 1*ByteDecDigit)] [ConstantFloatExp] + / [ByteNumberSign] 1*ByteDecDigit *("_" 1*ByteDecDigit) ConstantFloatExp + / [ByteNumberSign] "." 1*ByteDecDigit *("_" 1*ByteDecDigit) [ConstantFloatExp] ConstantFloatExp - = "e" ["-"] 1*ByteDecDigit *("_" 1*ByteDecDigit) + = "e" [ByteNumberSign] 1*ByteDecDigit *("_" 1*ByteDecDigit) ConstantInt - = ["-"] "0b" 1*ByteBinDigit *("_" 1*ByteBinDigit) - / ["-"] "0o" 1*ByteOctDigit *("_" 1*ByteOctDigit) - / ["-"] "0x" 1*ByteHexDigit *("_" 1*ByteHexDigit) - / ["-"] 1*ByteDecDigit *("_" 1*ByteDecDigit) + = [ByteNumberSign] "0b" 1*ByteBinDigit *("_" 1*ByteBinDigit) + / [ByteNumberSign] "0o" 1*ByteOctDigit *("_" 1*ByteOctDigit) + / [ByteNumberSign] "0x" 1*ByteHexDigit *("_" 1*ByteHexDigit) + / [ByteNumberSign] 1*ByteDecDigit *("_" 1*ByteDecDigit) ConstantTrue = "true" @@ -192,6 +198,12 @@ TokenIs TokenNot = %s"not" 1*ByteHorizontalWs +TokenOf + = %s"of" 1*ByteHorizontalWs + +TokenSuper + = %s"super" 1*ByteHorizontalWs + TokenContravariant = %s"contravariant" 1*ByteHorizontalWs @@ -211,7 +223,11 @@ TokenIdentifier ByteHorizontalWs = %x09 ; horizontal tab - / %x20 ; space + / " " + +ByteNumberSign + = "+" + / "-" ByteBinDigit = %x30-31 ; 0-1 @@ -229,16 +245,13 @@ ByteHexDigit ByteIdentifierFirst = %x41-5A ; A-Z - / %x5F ; _ + / "_" / %x61-7A ; a-z / %x80-FF ByteIdentifierSecond - = %x30-39 ; 0-9 - / %x41-5A ; A-Z - / %x5F ; _ - / %x61-7A ; a-z - / %x80-FF + = ByteIdentifierFirst + / %x30-39 ; 0-9 ByteSingleQuote = %x27 ; ' diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index 04100fcd..3bc8ae89 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -1,31 +1,31 @@ parameters: ignoreErrors: - - message: "#^Method PHPStan\\\\PhpDocParser\\\\Ast\\\\ConstExpr\\\\QuoteAwareConstExprStringNode\\:\\:escapeDoubleQuotedString\\(\\) should return string but returns string\\|null\\.$#" + message: '#^Method PHPStan\\PhpDocParser\\Ast\\ConstExpr\\ConstExprStringNode\:\:escapeDoubleQuotedString\(\) should return string but returns string\|null\.$#' + identifier: return.type count: 1 - path: src/Ast/ConstExpr/QuoteAwareConstExprStringNode.php + path: src/Ast/ConstExpr/ConstExprStringNode.php - - message: "#^Cannot use array destructuring on array\\\\|int\\|string\\>\\|null\\.$#" + message: '#^Cannot use array destructuring on list\\|int\|string\>\|null\.$#' + identifier: offsetAccess.nonArray count: 1 path: src/Ast/NodeTraverser.php - - message: "#^Strict comparison using \\=\\=\\= between 2 and 2 will always evaluate to true\\.$#" - count: 2 - path: src/Ast/NodeTraverser.php - - - - message: "#^Variable property access on PHPStan\\\\PhpDocParser\\\\Ast\\\\Node\\.$#" + message: '#^Variable property access on PHPStan\\PhpDocParser\\Ast\\Node\.$#' + identifier: property.dynamicName count: 1 path: src/Ast/NodeTraverser.php - - message: "#^Method PHPStan\\\\PhpDocParser\\\\Parser\\\\StringUnescaper\\:\\:parseEscapeSequences\\(\\) should return string but returns string\\|null\\.$#" + message: '#^Method PHPStan\\PhpDocParser\\Parser\\StringUnescaper\:\:parseEscapeSequences\(\) should return string but returns string\|null\.$#' + identifier: return.type count: 1 path: src/Parser/StringUnescaper.php - - message: "#^Variable property access on PHPStan\\\\PhpDocParser\\\\Ast\\\\Node\\.$#" + message: '#^Variable property access on PHPStan\\PhpDocParser\\Ast\\Node\.$#' + identifier: property.dynamicName count: 2 path: src/Printer/Printer.php diff --git a/phpstan.neon b/phpstan.neon index 0c336514..92ffe420 100644 --- a/phpstan.neon +++ b/phpstan.neon @@ -7,6 +7,10 @@ parameters: - tests excludePaths: - tests/PHPStan/*/data/* + - tests/PHPStan/Parser/Doctrine/ApiResource.php level: 8 ignoreErrors: - '#^Dynamic call to static method PHPUnit\\Framework\\Assert#' + - + identifier: return.unusedType + path: tests/* diff --git a/src/Ast/Attribute.php b/src/Ast/Attribute.php index cd3a0a29..1f770ded 100644 --- a/src/Ast/Attribute.php +++ b/src/Ast/Attribute.php @@ -13,4 +13,6 @@ final class Attribute public const ORIGINAL_NODE = 'originalNode'; + public const COMMENTS = 'comments'; + } diff --git a/src/Ast/Comment.php b/src/Ast/Comment.php new file mode 100644 index 00000000..79e24ebb --- /dev/null +++ b/src/Ast/Comment.php @@ -0,0 +1,28 @@ +text = $text; + $this->startLine = $startLine; + $this->startIndex = $startIndex; + } + + public function getReformattedText(): string + { + return trim($this->text); + } + +} diff --git a/src/Ast/ConstExpr/ConstExprArrayItemNode.php b/src/Ast/ConstExpr/ConstExprArrayItemNode.php index ef144521..cd067cac 100644 --- a/src/Ast/ConstExpr/ConstExprArrayItemNode.php +++ b/src/Ast/ConstExpr/ConstExprArrayItemNode.php @@ -10,11 +10,9 @@ class ConstExprArrayItemNode implements ConstExprNode use NodeAttributes; - /** @var ConstExprNode|null */ - public $key; + public ?ConstExprNode $key = null; - /** @var ConstExprNode */ - public $value; + public ConstExprNode $value; public function __construct(?ConstExprNode $key, ConstExprNode $value) { diff --git a/src/Ast/ConstExpr/ConstExprArrayNode.php b/src/Ast/ConstExpr/ConstExprArrayNode.php index 1f9def37..dc7ad4a9 100644 --- a/src/Ast/ConstExpr/ConstExprArrayNode.php +++ b/src/Ast/ConstExpr/ConstExprArrayNode.php @@ -11,7 +11,7 @@ class ConstExprArrayNode implements ConstExprNode use NodeAttributes; /** @var ConstExprArrayItemNode[] */ - public $items; + public array $items; /** * @param ConstExprArrayItemNode[] $items diff --git a/src/Ast/ConstExpr/ConstExprFloatNode.php b/src/Ast/ConstExpr/ConstExprFloatNode.php index a4192fba..30ab41de 100644 --- a/src/Ast/ConstExpr/ConstExprFloatNode.php +++ b/src/Ast/ConstExpr/ConstExprFloatNode.php @@ -9,8 +9,7 @@ class ConstExprFloatNode implements ConstExprNode use NodeAttributes; - /** @var string */ - public $value; + public string $value; public function __construct(string $value) { diff --git a/src/Ast/ConstExpr/ConstExprIntegerNode.php b/src/Ast/ConstExpr/ConstExprIntegerNode.php index 5339bb5a..9f0285f3 100644 --- a/src/Ast/ConstExpr/ConstExprIntegerNode.php +++ b/src/Ast/ConstExpr/ConstExprIntegerNode.php @@ -9,8 +9,7 @@ class ConstExprIntegerNode implements ConstExprNode use NodeAttributes; - /** @var string */ - public $value; + public string $value; public function __construct(string $value) { diff --git a/src/Ast/ConstExpr/ConstExprStringNode.php b/src/Ast/ConstExpr/ConstExprStringNode.php index fa44c262..26e5ef4a 100644 --- a/src/Ast/ConstExpr/ConstExprStringNode.php +++ b/src/Ast/ConstExpr/ConstExprStringNode.php @@ -3,24 +3,78 @@ namespace PHPStan\PhpDocParser\Ast\ConstExpr; use PHPStan\PhpDocParser\Ast\NodeAttributes; +use function addcslashes; +use function assert; +use function dechex; +use function ord; +use function preg_replace_callback; +use function sprintf; +use function str_pad; +use function strlen; +use const STR_PAD_LEFT; class ConstExprStringNode implements ConstExprNode { + public const SINGLE_QUOTED = 1; + public const DOUBLE_QUOTED = 2; + use NodeAttributes; - /** @var string */ - public $value; + public string $value; + + /** @var self::SINGLE_QUOTED|self::DOUBLE_QUOTED */ + public $quoteType; - public function __construct(string $value) + /** + * @param self::SINGLE_QUOTED|self::DOUBLE_QUOTED $quoteType + */ + public function __construct(string $value, int $quoteType) { $this->value = $value; + $this->quoteType = $quoteType; } public function __toString(): string { - return $this->value; + if ($this->quoteType === self::SINGLE_QUOTED) { + // from https://github.com/nikic/PHP-Parser/blob/0ffddce52d816f72d0efc4d9b02e276d3309ef01/lib/PhpParser/PrettyPrinter/Standard.php#L1007 + return sprintf("'%s'", addcslashes($this->value, '\'\\')); + } + + // from https://github.com/nikic/PHP-Parser/blob/0ffddce52d816f72d0efc4d9b02e276d3309ef01/lib/PhpParser/PrettyPrinter/Standard.php#L1010-L1040 + return sprintf('"%s"', $this->escapeDoubleQuotedString()); + } + + private function escapeDoubleQuotedString(): string + { + $quote = '"'; + $escaped = addcslashes($this->value, "\n\r\t\f\v$" . $quote . '\\'); + + // Escape control characters and non-UTF-8 characters. + // Regex based on https://stackoverflow.com/a/11709412/385378. + $regex = '/( + [\x00-\x08\x0E-\x1F] # Control characters + | [\xC0-\xC1] # Invalid UTF-8 Bytes + | [\xF5-\xFF] # Invalid UTF-8 Bytes + | \xE0(?=[\x80-\x9F]) # Overlong encoding of prior code point + | \xF0(?=[\x80-\x8F]) # Overlong encoding of prior code point + | [\xC2-\xDF](?![\x80-\xBF]) # Invalid UTF-8 Sequence Start + | [\xE0-\xEF](?![\x80-\xBF]{2}) # Invalid UTF-8 Sequence Start + | [\xF0-\xF4](?![\x80-\xBF]{3}) # Invalid UTF-8 Sequence Start + | (?<=[\x00-\x7F\xF5-\xFF])[\x80-\xBF] # Invalid UTF-8 Sequence Middle + | (?value = $value; + } + + public function __toString(): string + { + return self::escape($this->value); + } + + public static function unescape(string $value): string + { + // from https://github.com/doctrine/annotations/blob/a9ec7af212302a75d1f92fa65d3abfbd16245a2a/lib/Doctrine/Common/Annotations/DocLexer.php#L103-L107 + return str_replace('""', '"', substr($value, 1, strlen($value) - 2)); + } + + private static function escape(string $value): string + { + // from https://github.com/phpstan/phpdoc-parser/issues/205#issuecomment-1662323656 + return sprintf('"%s"', str_replace('"', '""', $value)); + } + +} diff --git a/src/Ast/ConstExpr/QuoteAwareConstExprStringNode.php b/src/Ast/ConstExpr/QuoteAwareConstExprStringNode.php deleted file mode 100644 index f2792b1b..00000000 --- a/src/Ast/ConstExpr/QuoteAwareConstExprStringNode.php +++ /dev/null @@ -1,78 +0,0 @@ -quoteType = $quoteType; - } - - - public function __toString(): string - { - if ($this->quoteType === self::SINGLE_QUOTED) { - // from https://github.com/nikic/PHP-Parser/blob/0ffddce52d816f72d0efc4d9b02e276d3309ef01/lib/PhpParser/PrettyPrinter/Standard.php#L1007 - return sprintf("'%s'", addcslashes($this->value, '\'\\')); - } - - // from https://github.com/nikic/PHP-Parser/blob/0ffddce52d816f72d0efc4d9b02e276d3309ef01/lib/PhpParser/PrettyPrinter/Standard.php#L1010-L1040 - return sprintf('"%s"', $this->escapeDoubleQuotedString()); - } - - private function escapeDoubleQuotedString(): string - { - $quote = '"'; - $escaped = addcslashes($this->value, "\n\r\t\f\v$" . $quote . '\\'); - - // Escape control characters and non-UTF-8 characters. - // Regex based on https://stackoverflow.com/a/11709412/385378. - $regex = '/( - [\x00-\x08\x0E-\x1F] # Control characters - | [\xC0-\xC1] # Invalid UTF-8 Bytes - | [\xF5-\xFF] # Invalid UTF-8 Bytes - | \xE0(?=[\x80-\x9F]) # Overlong encoding of prior code point - | \xF0(?=[\x80-\x8F]) # Overlong encoding of prior code point - | [\xC2-\xDF](?![\x80-\xBF]) # Invalid UTF-8 Sequence Start - | [\xE0-\xEF](?![\x80-\xBF]{2}) # Invalid UTF-8 Sequence Start - | [\xF0-\xF4](?![\x80-\xBF]{3}) # Invalid UTF-8 Sequence Start - | (?<=[\x00-\x7F\xF5-\xFF])[\x80-\xBF] # Invalid UTF-8 Sequence Middle - | (? */ - private $attributes = []; + private array $attributes = []; /** * @param mixed $value */ public function setAttribute(string $key, $value): void { + if ($value === null) { + unset($this->attributes[$key]); + return; + } $this->attributes[$key] = $value; } diff --git a/src/Ast/NodeTraverser.php b/src/Ast/NodeTraverser.php index 63b25c37..76c88f0c 100644 --- a/src/Ast/NodeTraverser.php +++ b/src/Ast/NodeTraverser.php @@ -62,10 +62,10 @@ final class NodeTraverser public const DONT_TRAVERSE_CURRENT_AND_CHILDREN = 4; /** @var list Visitors */ - private $visitors = []; + private array $visitors = []; /** @var bool Whether traversal should be stopped */ - private $stopTraversal; + private bool $stopTraversal; /** * @param list $visitors @@ -151,7 +151,7 @@ private function traverseNode(Node $node): Node break 2; } else { throw new LogicException( - 'enterNode() returned invalid value of type ' . gettype($return) + 'enterNode() returned invalid value of type ' . gettype($return), ); } } @@ -176,11 +176,11 @@ private function traverseNode(Node $node): Node } elseif (is_array($return)) { throw new LogicException( 'leaveNode() may only return an array ' . - 'if the parent structure is an array' + 'if the parent structure is an array', ); } else { throw new LogicException( - 'leaveNode() returned invalid value of type ' . gettype($return) + 'leaveNode() returned invalid value of type ' . gettype($return), ); } } @@ -237,7 +237,7 @@ private function traverseArray(array $nodes): array break 2; } else { throw new LogicException( - 'enterNode() returned invalid value of type ' . gettype($return) + 'enterNode() returned invalid value of type ' . gettype($return), ); } } @@ -267,7 +267,7 @@ private function traverseArray(array $nodes): array break 2; } else { throw new LogicException( - 'leaveNode() returned invalid value of type ' . gettype($return) + 'leaveNode() returned invalid value of type ' . gettype($return), ); } } diff --git a/src/Ast/NodeVisitor/CloningVisitor.php b/src/Ast/NodeVisitor/CloningVisitor.php index 7200f3af..486e2aae 100644 --- a/src/Ast/NodeVisitor/CloningVisitor.php +++ b/src/Ast/NodeVisitor/CloningVisitor.php @@ -9,7 +9,7 @@ final class CloningVisitor extends AbstractNodeVisitor { - public function enterNode(Node $originalNode) + public function enterNode(Node $originalNode): Node { $node = clone $originalNode; $node->setAttribute(Attribute::ORIGINAL_NODE, $originalNode); diff --git a/src/Ast/PhpDoc/AssertTagMethodValueNode.php b/src/Ast/PhpDoc/AssertTagMethodValueNode.php index cf4f5563..0dfee314 100644 --- a/src/Ast/PhpDoc/AssertTagMethodValueNode.php +++ b/src/Ast/PhpDoc/AssertTagMethodValueNode.php @@ -11,25 +11,20 @@ class AssertTagMethodValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; - /** @var string */ - public $parameter; + public string $parameter; - /** @var string */ - public $method; + public string $method; - /** @var bool */ - public $isNegated; + public bool $isNegated; - /** @var bool */ - public $isEquality; + public bool $isEquality; /** @var string (may be empty) */ - public $description; + public string $description; - public function __construct(TypeNode $type, string $parameter, string $method, bool $isNegated, string $description, bool $isEquality = false) + public function __construct(TypeNode $type, string $parameter, string $method, bool $isNegated, string $description, bool $isEquality) { $this->type = $type; $this->parameter = $parameter; diff --git a/src/Ast/PhpDoc/AssertTagPropertyValueNode.php b/src/Ast/PhpDoc/AssertTagPropertyValueNode.php index 4fb31807..8bfd1d0e 100644 --- a/src/Ast/PhpDoc/AssertTagPropertyValueNode.php +++ b/src/Ast/PhpDoc/AssertTagPropertyValueNode.php @@ -11,25 +11,20 @@ class AssertTagPropertyValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; - /** @var string */ - public $parameter; + public string $parameter; - /** @var string */ - public $property; + public string $property; - /** @var bool */ - public $isNegated; + public bool $isNegated; - /** @var bool */ - public $isEquality; + public bool $isEquality; /** @var string (may be empty) */ - public $description; + public string $description; - public function __construct(TypeNode $type, string $parameter, string $property, bool $isNegated, string $description, bool $isEquality = false) + public function __construct(TypeNode $type, string $parameter, string $property, bool $isNegated, string $description, bool $isEquality) { $this->type = $type; $this->parameter = $parameter; diff --git a/src/Ast/PhpDoc/AssertTagValueNode.php b/src/Ast/PhpDoc/AssertTagValueNode.php index d6423f50..5dc9e8c3 100644 --- a/src/Ast/PhpDoc/AssertTagValueNode.php +++ b/src/Ast/PhpDoc/AssertTagValueNode.php @@ -11,22 +11,18 @@ class AssertTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; - /** @var string */ - public $parameter; + public string $parameter; - /** @var bool */ - public $isNegated; + public bool $isNegated; - /** @var bool */ - public $isEquality; + public bool $isEquality; /** @var string (may be empty) */ - public $description; + public string $description; - public function __construct(TypeNode $type, string $parameter, bool $isNegated, string $description, bool $isEquality = false) + public function __construct(TypeNode $type, string $parameter, bool $isNegated, string $description, bool $isEquality) { $this->type = $type; $this->parameter = $parameter; diff --git a/src/Ast/PhpDoc/DeprecatedTagValueNode.php b/src/Ast/PhpDoc/DeprecatedTagValueNode.php index abf2f1a6..facdd2b4 100644 --- a/src/Ast/PhpDoc/DeprecatedTagValueNode.php +++ b/src/Ast/PhpDoc/DeprecatedTagValueNode.php @@ -11,7 +11,7 @@ class DeprecatedTagValueNode implements PhpDocTagValueNode use NodeAttributes; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(string $description) { diff --git a/src/Ast/PhpDoc/Doctrine/DoctrineAnnotation.php b/src/Ast/PhpDoc/Doctrine/DoctrineAnnotation.php new file mode 100644 index 00000000..778b21fa --- /dev/null +++ b/src/Ast/PhpDoc/Doctrine/DoctrineAnnotation.php @@ -0,0 +1,34 @@ + */ + public array $arguments; + + /** + * @param list $arguments + */ + public function __construct(string $name, array $arguments) + { + $this->name = $name; + $this->arguments = $arguments; + } + + public function __toString(): string + { + $arguments = implode(', ', $this->arguments); + return $this->name . '(' . $arguments . ')'; + } + +} diff --git a/src/Ast/PhpDoc/Doctrine/DoctrineArgument.php b/src/Ast/PhpDoc/Doctrine/DoctrineArgument.php new file mode 100644 index 00000000..30fe9879 --- /dev/null +++ b/src/Ast/PhpDoc/Doctrine/DoctrineArgument.php @@ -0,0 +1,42 @@ +key = $key; + $this->value = $value; + } + + + public function __toString(): string + { + if ($this->key === null) { + return (string) $this->value; + } + + return $this->key . '=' . $this->value; + } + +} diff --git a/src/Ast/PhpDoc/Doctrine/DoctrineArray.php b/src/Ast/PhpDoc/Doctrine/DoctrineArray.php new file mode 100644 index 00000000..06686a52 --- /dev/null +++ b/src/Ast/PhpDoc/Doctrine/DoctrineArray.php @@ -0,0 +1,32 @@ + */ + public array $items; + + /** + * @param list $items + */ + public function __construct(array $items) + { + $this->items = $items; + } + + public function __toString(): string + { + $items = implode(', ', $this->items); + + return '{' . $items . '}'; + } + +} diff --git a/src/Ast/PhpDoc/Doctrine/DoctrineArrayItem.php b/src/Ast/PhpDoc/Doctrine/DoctrineArrayItem.php new file mode 100644 index 00000000..d2dbf2b6 --- /dev/null +++ b/src/Ast/PhpDoc/Doctrine/DoctrineArrayItem.php @@ -0,0 +1,47 @@ +key = $key; + $this->value = $value; + } + + + public function __toString(): string + { + if ($this->key === null) { + return (string) $this->value; + } + + return $this->key . '=' . $this->value; + } + +} diff --git a/src/Ast/PhpDoc/Doctrine/DoctrineTagValueNode.php b/src/Ast/PhpDoc/Doctrine/DoctrineTagValueNode.php new file mode 100644 index 00000000..3940e24f --- /dev/null +++ b/src/Ast/PhpDoc/Doctrine/DoctrineTagValueNode.php @@ -0,0 +1,35 @@ +annotation = $annotation; + $this->description = $description; + } + + + public function __toString(): string + { + return trim("{$this->annotation} {$this->description}"); + } + +} diff --git a/src/Ast/PhpDoc/ExtendsTagValueNode.php b/src/Ast/PhpDoc/ExtendsTagValueNode.php index 3bf53e13..d9bbeec5 100644 --- a/src/Ast/PhpDoc/ExtendsTagValueNode.php +++ b/src/Ast/PhpDoc/ExtendsTagValueNode.php @@ -11,11 +11,10 @@ class ExtendsTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var GenericTypeNode */ - public $type; + public GenericTypeNode $type; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(GenericTypeNode $type, string $description) { diff --git a/src/Ast/PhpDoc/GenericTagValueNode.php b/src/Ast/PhpDoc/GenericTagValueNode.php index 026aa153..de77051a 100644 --- a/src/Ast/PhpDoc/GenericTagValueNode.php +++ b/src/Ast/PhpDoc/GenericTagValueNode.php @@ -10,7 +10,7 @@ class GenericTagValueNode implements PhpDocTagValueNode use NodeAttributes; /** @var string (may be empty) */ - public $value; + public string $value; public function __construct(string $value) { diff --git a/src/Ast/PhpDoc/ImplementsTagValueNode.php b/src/Ast/PhpDoc/ImplementsTagValueNode.php index 99043d91..34a30b0d 100644 --- a/src/Ast/PhpDoc/ImplementsTagValueNode.php +++ b/src/Ast/PhpDoc/ImplementsTagValueNode.php @@ -11,11 +11,10 @@ class ImplementsTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var GenericTypeNode */ - public $type; + public GenericTypeNode $type; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(GenericTypeNode $type, string $description) { diff --git a/src/Ast/PhpDoc/InvalidTagValueNode.php b/src/Ast/PhpDoc/InvalidTagValueNode.php index ca7b4f20..7bb20b22 100644 --- a/src/Ast/PhpDoc/InvalidTagValueNode.php +++ b/src/Ast/PhpDoc/InvalidTagValueNode.php @@ -17,10 +17,10 @@ class InvalidTagValueNode implements PhpDocTagValueNode use NodeAttributes; /** @var string (may be empty) */ - public $value; + public string $value; /** @var mixed[] */ - private $exceptionArgs; + private array $exceptionArgs; public function __construct(string $value, ParserException $exception) { diff --git a/src/Ast/PhpDoc/MethodTagValueNode.php b/src/Ast/PhpDoc/MethodTagValueNode.php index 211510be..223d6e47 100644 --- a/src/Ast/PhpDoc/MethodTagValueNode.php +++ b/src/Ast/PhpDoc/MethodTagValueNode.php @@ -12,29 +12,26 @@ class MethodTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var bool */ - public $isStatic; + public bool $isStatic; - /** @var TypeNode|null */ - public $returnType; + public ?TypeNode $returnType = null; - /** @var string */ - public $methodName; + public string $methodName; /** @var TemplateTagValueNode[] */ - public $templateTypes; + public array $templateTypes; /** @var MethodTagValueParameterNode[] */ - public $parameters; + public array $parameters; /** @var string (may be empty) */ - public $description; + public string $description; /** * @param MethodTagValueParameterNode[] $parameters * @param TemplateTagValueNode[] $templateTypes */ - public function __construct(bool $isStatic, ?TypeNode $returnType, string $methodName, array $parameters, string $description, array $templateTypes = []) + public function __construct(bool $isStatic, ?TypeNode $returnType, string $methodName, array $parameters, string $description, array $templateTypes) { $this->isStatic = $isStatic; $this->returnType = $returnType; diff --git a/src/Ast/PhpDoc/MethodTagValueParameterNode.php b/src/Ast/PhpDoc/MethodTagValueParameterNode.php index 7c17e44c..ebf33e32 100644 --- a/src/Ast/PhpDoc/MethodTagValueParameterNode.php +++ b/src/Ast/PhpDoc/MethodTagValueParameterNode.php @@ -12,20 +12,15 @@ class MethodTagValueParameterNode implements Node use NodeAttributes; - /** @var TypeNode|null */ - public $type; + public ?TypeNode $type = null; - /** @var bool */ - public $isReference; + public bool $isReference; - /** @var bool */ - public $isVariadic; + public bool $isVariadic; - /** @var string */ - public $parameterName; + public string $parameterName; - /** @var ConstExprNode|null */ - public $defaultValue; + public ?ConstExprNode $defaultValue = null; public function __construct(?TypeNode $type, bool $isReference, bool $isVariadic, string $parameterName, ?ConstExprNode $defaultValue) { diff --git a/src/Ast/PhpDoc/MixinTagValueNode.php b/src/Ast/PhpDoc/MixinTagValueNode.php index d9b7d78a..7a4e43ea 100644 --- a/src/Ast/PhpDoc/MixinTagValueNode.php +++ b/src/Ast/PhpDoc/MixinTagValueNode.php @@ -11,11 +11,10 @@ class MixinTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(TypeNode $type, string $description) { diff --git a/src/Ast/PhpDoc/ParamClosureThisTagValueNode.php b/src/Ast/PhpDoc/ParamClosureThisTagValueNode.php new file mode 100644 index 00000000..54feff9e --- /dev/null +++ b/src/Ast/PhpDoc/ParamClosureThisTagValueNode.php @@ -0,0 +1,33 @@ +type = $type; + $this->parameterName = $parameterName; + $this->description = $description; + } + + public function __toString(): string + { + return trim("{$this->type} {$this->parameterName} {$this->description}"); + } + +} diff --git a/src/Ast/PhpDoc/ParamImmediatelyInvokedCallableTagValueNode.php b/src/Ast/PhpDoc/ParamImmediatelyInvokedCallableTagValueNode.php new file mode 100644 index 00000000..9a6761f7 --- /dev/null +++ b/src/Ast/PhpDoc/ParamImmediatelyInvokedCallableTagValueNode.php @@ -0,0 +1,29 @@ +parameterName = $parameterName; + $this->description = $description; + } + + public function __toString(): string + { + return trim("{$this->parameterName} {$this->description}"); + } + +} diff --git a/src/Ast/PhpDoc/ParamLaterInvokedCallableTagValueNode.php b/src/Ast/PhpDoc/ParamLaterInvokedCallableTagValueNode.php new file mode 100644 index 00000000..84db67a9 --- /dev/null +++ b/src/Ast/PhpDoc/ParamLaterInvokedCallableTagValueNode.php @@ -0,0 +1,29 @@ +parameterName = $parameterName; + $this->description = $description; + } + + public function __toString(): string + { + return trim("{$this->parameterName} {$this->description}"); + } + +} diff --git a/src/Ast/PhpDoc/ParamOutTagValueNode.php b/src/Ast/PhpDoc/ParamOutTagValueNode.php index 9f374bf1..3e89f9d7 100644 --- a/src/Ast/PhpDoc/ParamOutTagValueNode.php +++ b/src/Ast/PhpDoc/ParamOutTagValueNode.php @@ -11,14 +11,12 @@ class ParamOutTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; - /** @var string */ - public $parameterName; + public string $parameterName; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(TypeNode $type, string $parameterName, string $description) { diff --git a/src/Ast/PhpDoc/ParamTagValueNode.php b/src/Ast/PhpDoc/ParamTagValueNode.php index f93af0ea..8d1ef27a 100644 --- a/src/Ast/PhpDoc/ParamTagValueNode.php +++ b/src/Ast/PhpDoc/ParamTagValueNode.php @@ -11,22 +11,18 @@ class ParamTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; - /** @var bool */ - public $isReference; + public bool $isReference; - /** @var bool */ - public $isVariadic; + public bool $isVariadic; - /** @var string */ - public $parameterName; + public string $parameterName; /** @var string (may be empty) */ - public $description; + public string $description; - public function __construct(TypeNode $type, bool $isVariadic, string $parameterName, string $description, bool $isReference = false) + public function __construct(TypeNode $type, bool $isVariadic, string $parameterName, string $description, bool $isReference) { $this->type = $type; $this->isReference = $isReference; diff --git a/src/Ast/PhpDoc/PhpDocNode.php b/src/Ast/PhpDoc/PhpDocNode.php index 25f1939c..6abad3d0 100644 --- a/src/Ast/PhpDoc/PhpDocNode.php +++ b/src/Ast/PhpDoc/PhpDocNode.php @@ -15,7 +15,7 @@ class PhpDocNode implements Node use NodeAttributes; /** @var PhpDocChildNode[] */ - public $children; + public array $children; /** * @param PhpDocChildNode[] $children @@ -31,9 +31,7 @@ public function __construct(array $children) */ public function getTags(): array { - return array_filter($this->children, static function (PhpDocChildNode $child): bool { - return $child instanceof PhpDocTagNode; - }); + return array_filter($this->children, static fn (PhpDocChildNode $child): bool => $child instanceof PhpDocTagNode); } @@ -42,9 +40,7 @@ public function getTags(): array */ public function getTagsByName(string $tagName): array { - return array_filter($this->getTags(), static function (PhpDocTagNode $tag) use ($tagName): bool { - return $tag->name === $tagName; - }); + return array_filter($this->getTags(), static fn (PhpDocTagNode $tag): bool => $tag->name === $tagName); } @@ -55,9 +51,7 @@ public function getVarTagValues(string $tagName = '@var'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof VarTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof VarTagValueNode, ); } @@ -69,9 +63,7 @@ public function getParamTagValues(string $tagName = '@param'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof ParamTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof ParamTagValueNode, ); } @@ -83,13 +75,57 @@ public function getTypelessParamTagValues(string $tagName = '@param'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof TypelessParamTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof TypelessParamTagValueNode, ); } + /** + * @return ParamImmediatelyInvokedCallableTagValueNode[] + */ + public function getParamImmediatelyInvokedCallableTagValues(string $tagName = '@param-immediately-invoked-callable'): array + { + return array_filter( + array_column($this->getTagsByName($tagName), 'value'), + static fn (PhpDocTagValueNode $value): bool => $value instanceof ParamImmediatelyInvokedCallableTagValueNode, + ); + } + + + /** + * @return ParamLaterInvokedCallableTagValueNode[] + */ + public function getParamLaterInvokedCallableTagValues(string $tagName = '@param-later-invoked-callable'): array + { + return array_filter( + array_column($this->getTagsByName($tagName), 'value'), + static fn (PhpDocTagValueNode $value): bool => $value instanceof ParamLaterInvokedCallableTagValueNode, + ); + } + + + /** + * @return ParamClosureThisTagValueNode[] + */ + public function getParamClosureThisTagValues(string $tagName = '@param-closure-this'): array + { + return array_filter( + array_column($this->getTagsByName($tagName), 'value'), + static fn (PhpDocTagValueNode $value): bool => $value instanceof ParamClosureThisTagValueNode, + ); + } + + /** + * @return PureUnlessCallableIsImpureTagValueNode[] + */ + public function getPureUnlessCallableIsImpureTagValues(string $tagName = '@pure-unless-callable-is-impure'): array + { + return array_filter( + array_column($this->getTagsByName($tagName), 'value'), + static fn (PhpDocTagValueNode $value): bool => $value instanceof PureUnlessCallableIsImpureTagValueNode, + ); + } + /** * @return TemplateTagValueNode[] */ @@ -97,9 +133,7 @@ public function getTemplateTagValues(string $tagName = '@template'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof TemplateTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof TemplateTagValueNode, ); } @@ -111,9 +145,7 @@ public function getExtendsTagValues(string $tagName = '@extends'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof ExtendsTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof ExtendsTagValueNode, ); } @@ -125,9 +157,7 @@ public function getImplementsTagValues(string $tagName = '@implements'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof ImplementsTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof ImplementsTagValueNode, ); } @@ -139,9 +169,7 @@ public function getUsesTagValues(string $tagName = '@use'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof UsesTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof UsesTagValueNode, ); } @@ -153,9 +181,7 @@ public function getReturnTagValues(string $tagName = '@return'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof ReturnTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof ReturnTagValueNode, ); } @@ -167,9 +193,7 @@ public function getThrowsTagValues(string $tagName = '@throws'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof ThrowsTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof ThrowsTagValueNode, ); } @@ -181,12 +205,31 @@ public function getMixinTagValues(string $tagName = '@mixin'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof MixinTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof MixinTagValueNode, + ); + } + + /** + * @return RequireExtendsTagValueNode[] + */ + public function getRequireExtendsTagValues(string $tagName = '@phpstan-require-extends'): array + { + return array_filter( + array_column($this->getTagsByName($tagName), 'value'), + static fn (PhpDocTagValueNode $value): bool => $value instanceof RequireExtendsTagValueNode, ); } + /** + * @return RequireImplementsTagValueNode[] + */ + public function getRequireImplementsTagValues(string $tagName = '@phpstan-require-implements'): array + { + return array_filter( + array_column($this->getTagsByName($tagName), 'value'), + static fn (PhpDocTagValueNode $value): bool => $value instanceof RequireImplementsTagValueNode, + ); + } /** * @return DeprecatedTagValueNode[] @@ -195,9 +238,7 @@ public function getDeprecatedTagValues(): array { return array_filter( array_column($this->getTagsByName('@deprecated'), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof DeprecatedTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof DeprecatedTagValueNode, ); } @@ -209,9 +250,7 @@ public function getPropertyTagValues(string $tagName = '@property'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof PropertyTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof PropertyTagValueNode, ); } @@ -223,9 +262,7 @@ public function getPropertyReadTagValues(string $tagName = '@property-read'): ar { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof PropertyTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof PropertyTagValueNode, ); } @@ -237,9 +274,7 @@ public function getPropertyWriteTagValues(string $tagName = '@property-write'): { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof PropertyTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof PropertyTagValueNode, ); } @@ -251,9 +286,7 @@ public function getMethodTagValues(string $tagName = '@method'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof MethodTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof MethodTagValueNode, ); } @@ -265,9 +298,7 @@ public function getTypeAliasTagValues(string $tagName = '@phpstan-type'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof TypeAliasTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof TypeAliasTagValueNode, ); } @@ -279,9 +310,7 @@ public function getTypeAliasImportTagValues(string $tagName = '@phpstan-import-t { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof TypeAliasImportTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof TypeAliasImportTagValueNode, ); } @@ -293,9 +322,7 @@ public function getAssertTagValues(string $tagName = '@phpstan-assert'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof AssertTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof AssertTagValueNode, ); } @@ -307,9 +334,7 @@ public function getAssertPropertyTagValues(string $tagName = '@phpstan-assert'): { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof AssertTagPropertyValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof AssertTagPropertyValueNode, ); } @@ -321,9 +346,7 @@ public function getAssertMethodTagValues(string $tagName = '@phpstan-assert'): a { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof AssertTagMethodValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof AssertTagMethodValueNode, ); } @@ -335,9 +358,7 @@ public function getSelfOutTypeTagValues(string $tagName = '@phpstan-this-out'): { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof SelfOutTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof SelfOutTagValueNode, ); } @@ -349,9 +370,7 @@ public function getParamOutTypeTagValues(string $tagName = '@param-out'): array { return array_filter( array_column($this->getTagsByName($tagName), 'value'), - static function (PhpDocTagValueNode $value): bool { - return $value instanceof ParamOutTagValueNode; - } + static fn (PhpDocTagValueNode $value): bool => $value instanceof ParamOutTagValueNode, ); } @@ -363,7 +382,7 @@ static function (PhpDocChildNode $child): string { $s = (string) $child; return $s === '' ? '' : ' ' . $s; }, - $this->children + $this->children, ); return "/**\n *" . implode("\n *", $children) . "\n */"; } diff --git a/src/Ast/PhpDoc/PhpDocTagNode.php b/src/Ast/PhpDoc/PhpDocTagNode.php index 856cc3f1..cac2feeb 100644 --- a/src/Ast/PhpDoc/PhpDocTagNode.php +++ b/src/Ast/PhpDoc/PhpDocTagNode.php @@ -3,6 +3,7 @@ namespace PHPStan\PhpDocParser\Ast\PhpDoc; use PHPStan\PhpDocParser\Ast\NodeAttributes; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineTagValueNode; use function trim; class PhpDocTagNode implements PhpDocChildNode @@ -10,11 +11,9 @@ class PhpDocTagNode implements PhpDocChildNode use NodeAttributes; - /** @var string */ - public $name; + public string $name; - /** @var PhpDocTagValueNode */ - public $value; + public PhpDocTagValueNode $value; public function __construct(string $name, PhpDocTagValueNode $value) { @@ -25,6 +24,10 @@ public function __construct(string $name, PhpDocTagValueNode $value) public function __toString(): string { + if ($this->value instanceof DoctrineTagValueNode) { + return (string) $this->value; + } + return trim("{$this->name} {$this->value}"); } diff --git a/src/Ast/PhpDoc/PhpDocTextNode.php b/src/Ast/PhpDoc/PhpDocTextNode.php index 0bca3c99..97a96894 100644 --- a/src/Ast/PhpDoc/PhpDocTextNode.php +++ b/src/Ast/PhpDoc/PhpDocTextNode.php @@ -9,8 +9,7 @@ class PhpDocTextNode implements PhpDocChildNode use NodeAttributes; - /** @var string */ - public $text; + public string $text; public function __construct(string $text) { diff --git a/src/Ast/PhpDoc/PropertyTagValueNode.php b/src/Ast/PhpDoc/PropertyTagValueNode.php index 046003d1..cbf622f8 100644 --- a/src/Ast/PhpDoc/PropertyTagValueNode.php +++ b/src/Ast/PhpDoc/PropertyTagValueNode.php @@ -11,14 +11,12 @@ class PropertyTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; - /** @var string */ - public $propertyName; + public string $propertyName; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(TypeNode $type, string $propertyName, string $description) { diff --git a/src/Ast/PhpDoc/PureUnlessCallableIsImpureTagValueNode.php b/src/Ast/PhpDoc/PureUnlessCallableIsImpureTagValueNode.php new file mode 100644 index 00000000..1a0cff89 --- /dev/null +++ b/src/Ast/PhpDoc/PureUnlessCallableIsImpureTagValueNode.php @@ -0,0 +1,29 @@ +parameterName = $parameterName; + $this->description = $description; + } + + public function __toString(): string + { + return trim("{$this->parameterName} {$this->description}"); + } + +} diff --git a/src/Ast/PhpDoc/RequireExtendsTagValueNode.php b/src/Ast/PhpDoc/RequireExtendsTagValueNode.php new file mode 100644 index 00000000..5d8e1d22 --- /dev/null +++ b/src/Ast/PhpDoc/RequireExtendsTagValueNode.php @@ -0,0 +1,31 @@ +type = $type; + $this->description = $description; + } + + + public function __toString(): string + { + return trim("{$this->type} {$this->description}"); + } + +} diff --git a/src/Ast/PhpDoc/RequireImplementsTagValueNode.php b/src/Ast/PhpDoc/RequireImplementsTagValueNode.php new file mode 100644 index 00000000..6b7f7bae --- /dev/null +++ b/src/Ast/PhpDoc/RequireImplementsTagValueNode.php @@ -0,0 +1,31 @@ +type = $type; + $this->description = $description; + } + + + public function __toString(): string + { + return trim("{$this->type} {$this->description}"); + } + +} diff --git a/src/Ast/PhpDoc/ReturnTagValueNode.php b/src/Ast/PhpDoc/ReturnTagValueNode.php index d53c8c75..c063bf48 100644 --- a/src/Ast/PhpDoc/ReturnTagValueNode.php +++ b/src/Ast/PhpDoc/ReturnTagValueNode.php @@ -11,11 +11,10 @@ class ReturnTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(TypeNode $type, string $description) { diff --git a/src/Ast/PhpDoc/SelfOutTagValueNode.php b/src/Ast/PhpDoc/SelfOutTagValueNode.php index 83169aff..d2377620 100644 --- a/src/Ast/PhpDoc/SelfOutTagValueNode.php +++ b/src/Ast/PhpDoc/SelfOutTagValueNode.php @@ -11,11 +11,10 @@ class SelfOutTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(TypeNode $type, string $description) { diff --git a/src/Ast/PhpDoc/TemplateTagValueNode.php b/src/Ast/PhpDoc/TemplateTagValueNode.php index 1d3c70e4..ba106825 100644 --- a/src/Ast/PhpDoc/TemplateTagValueNode.php +++ b/src/Ast/PhpDoc/TemplateTagValueNode.php @@ -11,22 +11,26 @@ class TemplateTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var string */ - public $name; + /** @var non-empty-string */ + public string $name; - /** @var TypeNode|null */ - public $bound; + public ?TypeNode $bound; - /** @var TypeNode|null */ - public $default; + public ?TypeNode $default; + + public ?TypeNode $lowerBound; /** @var string (may be empty) */ - public $description; + public string $description; - public function __construct(string $name, ?TypeNode $bound, string $description, ?TypeNode $default = null) + /** + * @param non-empty-string $name + */ + public function __construct(string $name, ?TypeNode $bound, string $description, ?TypeNode $default = null, ?TypeNode $lowerBound = null) { $this->name = $name; $this->bound = $bound; + $this->lowerBound = $lowerBound; $this->default = $default; $this->description = $description; } @@ -34,9 +38,10 @@ public function __construct(string $name, ?TypeNode $bound, string $description, public function __toString(): string { - $bound = $this->bound !== null ? " of {$this->bound}" : ''; + $upperBound = $this->bound !== null ? " of {$this->bound}" : ''; + $lowerBound = $this->lowerBound !== null ? " super {$this->lowerBound}" : ''; $default = $this->default !== null ? " = {$this->default}" : ''; - return trim("{$this->name}{$bound}{$default} {$this->description}"); + return trim("{$this->name}{$upperBound}{$lowerBound}{$default} {$this->description}"); } } diff --git a/src/Ast/PhpDoc/ThrowsTagValueNode.php b/src/Ast/PhpDoc/ThrowsTagValueNode.php index 62d2aed3..dc5521b4 100644 --- a/src/Ast/PhpDoc/ThrowsTagValueNode.php +++ b/src/Ast/PhpDoc/ThrowsTagValueNode.php @@ -11,11 +11,10 @@ class ThrowsTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(TypeNode $type, string $description) { diff --git a/src/Ast/PhpDoc/TypeAliasImportTagValueNode.php b/src/Ast/PhpDoc/TypeAliasImportTagValueNode.php index ad6b85a5..d0f945d2 100644 --- a/src/Ast/PhpDoc/TypeAliasImportTagValueNode.php +++ b/src/Ast/PhpDoc/TypeAliasImportTagValueNode.php @@ -11,14 +11,11 @@ class TypeAliasImportTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var string */ - public $importedAlias; + public string $importedAlias; - /** @var IdentifierTypeNode */ - public $importedFrom; + public IdentifierTypeNode $importedFrom; - /** @var string|null */ - public $importedAs; + public ?string $importedAs = null; public function __construct(string $importedAlias, IdentifierTypeNode $importedFrom, ?string $importedAs) { @@ -31,7 +28,7 @@ public function __toString(): string { return trim( "{$this->importedAlias} from {$this->importedFrom}" - . ($this->importedAs !== null ? " as {$this->importedAs}" : '') + . ($this->importedAs !== null ? " as {$this->importedAs}" : ''), ); } diff --git a/src/Ast/PhpDoc/TypeAliasTagValueNode.php b/src/Ast/PhpDoc/TypeAliasTagValueNode.php index 4ccaaac4..ae366b50 100644 --- a/src/Ast/PhpDoc/TypeAliasTagValueNode.php +++ b/src/Ast/PhpDoc/TypeAliasTagValueNode.php @@ -11,11 +11,9 @@ class TypeAliasTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var string */ - public $alias; + public string $alias; - /** @var TypeNode */ - public $type; + public TypeNode $type; public function __construct(string $alias, TypeNode $type) { diff --git a/src/Ast/PhpDoc/TypelessParamTagValueNode.php b/src/Ast/PhpDoc/TypelessParamTagValueNode.php index 8b982954..bb99e781 100644 --- a/src/Ast/PhpDoc/TypelessParamTagValueNode.php +++ b/src/Ast/PhpDoc/TypelessParamTagValueNode.php @@ -10,19 +10,16 @@ class TypelessParamTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var bool */ - public $isReference; + public bool $isReference; - /** @var bool */ - public $isVariadic; + public bool $isVariadic; - /** @var string */ - public $parameterName; + public string $parameterName; /** @var string (may be empty) */ - public $description; + public string $description; - public function __construct(bool $isVariadic, string $parameterName, string $description, bool $isReference = false) + public function __construct(bool $isVariadic, string $parameterName, string $description, bool $isReference) { $this->isReference = $isReference; $this->isVariadic = $isVariadic; diff --git a/src/Ast/PhpDoc/UsesTagValueNode.php b/src/Ast/PhpDoc/UsesTagValueNode.php index cd573d97..b33fff60 100644 --- a/src/Ast/PhpDoc/UsesTagValueNode.php +++ b/src/Ast/PhpDoc/UsesTagValueNode.php @@ -11,11 +11,10 @@ class UsesTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var GenericTypeNode */ - public $type; + public GenericTypeNode $type; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(GenericTypeNode $type, string $description) { diff --git a/src/Ast/PhpDoc/VarTagValueNode.php b/src/Ast/PhpDoc/VarTagValueNode.php index afb941a8..5b0538c8 100644 --- a/src/Ast/PhpDoc/VarTagValueNode.php +++ b/src/Ast/PhpDoc/VarTagValueNode.php @@ -11,14 +11,13 @@ class VarTagValueNode implements PhpDocTagValueNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; /** @var string (may be empty) */ - public $variableName; + public string $variableName; /** @var string (may be empty) */ - public $description; + public string $description; public function __construct(TypeNode $type, string $variableName, string $description) { diff --git a/src/Ast/Type/ArrayShapeItemNode.php b/src/Ast/Type/ArrayShapeItemNode.php index 660c6c9d..bed62381 100644 --- a/src/Ast/Type/ArrayShapeItemNode.php +++ b/src/Ast/Type/ArrayShapeItemNode.php @@ -4,10 +4,11 @@ use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprIntegerNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprStringNode; +use PHPStan\PhpDocParser\Ast\Node; use PHPStan\PhpDocParser\Ast\NodeAttributes; use function sprintf; -class ArrayShapeItemNode implements TypeNode +class ArrayShapeItemNode implements Node { use NodeAttributes; @@ -15,11 +16,9 @@ class ArrayShapeItemNode implements TypeNode /** @var ConstExprIntegerNode|ConstExprStringNode|IdentifierTypeNode|null */ public $keyName; - /** @var bool */ - public $optional; + public bool $optional; - /** @var TypeNode */ - public $valueType; + public TypeNode $valueType; /** * @param ConstExprIntegerNode|ConstExprStringNode|IdentifierTypeNode|null $keyName @@ -39,7 +38,7 @@ public function __toString(): string '%s%s: %s', (string) $this->keyName, $this->optional ? '?' : '', - (string) $this->valueType + (string) $this->valueType, ); } diff --git a/src/Ast/Type/ArrayShapeNode.php b/src/Ast/Type/ArrayShapeNode.php index 806783f9..1d9cf850 100644 --- a/src/Ast/Type/ArrayShapeNode.php +++ b/src/Ast/Type/ArrayShapeNode.php @@ -10,36 +10,64 @@ class ArrayShapeNode implements TypeNode public const KIND_ARRAY = 'array'; public const KIND_LIST = 'list'; + public const KIND_NON_EMPTY_ARRAY = 'non-empty-array'; + public const KIND_NON_EMPTY_LIST = 'non-empty-list'; use NodeAttributes; /** @var ArrayShapeItemNode[] */ - public $items; + public array $items; - /** @var bool */ - public $sealed; + public bool $sealed; /** @var self::KIND_* */ public $kind; + public ?ArrayShapeUnsealedTypeNode $unsealedType = null; + /** * @param ArrayShapeItemNode[] $items * @param self::KIND_* $kind */ - public function __construct(array $items, bool $sealed = true, string $kind = self::KIND_ARRAY) + private function __construct( + array $items, + bool $sealed = true, + ?ArrayShapeUnsealedTypeNode $unsealedType = null, + string $kind = self::KIND_ARRAY + ) { $this->items = $items; $this->sealed = $sealed; + $this->unsealedType = $unsealedType; $this->kind = $kind; } + /** + * @param ArrayShapeItemNode[] $items + * @param self::KIND_* $kind + */ + public static function createSealed(array $items, string $kind = self::KIND_ARRAY): self + { + return new self($items, true, null, $kind); + } + + /** + * @param ArrayShapeItemNode[] $items + * @param self::KIND_* $kind + */ + public static function createUnsealed(array $items, ?ArrayShapeUnsealedTypeNode $unsealedType, string $kind = self::KIND_ARRAY): self + { + return new self($items, false, $unsealedType, $kind); + } + + public function __toString(): string { $items = $this->items; if (! $this->sealed) { - $items[] = '...'; + $items[] = '...' . $this->unsealedType; } return $this->kind . '{' . implode(', ', $items) . '}'; diff --git a/src/Ast/Type/ArrayShapeUnsealedTypeNode.php b/src/Ast/Type/ArrayShapeUnsealedTypeNode.php new file mode 100644 index 00000000..68d6b36f --- /dev/null +++ b/src/Ast/Type/ArrayShapeUnsealedTypeNode.php @@ -0,0 +1,32 @@ +valueType = $valueType; + $this->keyType = $keyType; + } + + public function __toString(): string + { + if ($this->keyType !== null) { + return sprintf('<%s, %s>', $this->keyType, $this->valueType); + } + return sprintf('<%s>', $this->valueType); + } + +} diff --git a/src/Ast/Type/ArrayTypeNode.php b/src/Ast/Type/ArrayTypeNode.php index d2031032..95c020d8 100644 --- a/src/Ast/Type/ArrayTypeNode.php +++ b/src/Ast/Type/ArrayTypeNode.php @@ -9,8 +9,7 @@ class ArrayTypeNode implements TypeNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; public function __construct(TypeNode $type) { diff --git a/src/Ast/Type/CallableTypeNode.php b/src/Ast/Type/CallableTypeNode.php index e57e5f82..0a9e3442 100644 --- a/src/Ast/Type/CallableTypeNode.php +++ b/src/Ast/Type/CallableTypeNode.php @@ -3,6 +3,7 @@ namespace PHPStan\PhpDocParser\Ast\Type; use PHPStan\PhpDocParser\Ast\NodeAttributes; +use PHPStan\PhpDocParser\Ast\PhpDoc\TemplateTagValueNode; use function implode; class CallableTypeNode implements TypeNode @@ -10,23 +11,26 @@ class CallableTypeNode implements TypeNode use NodeAttributes; - /** @var IdentifierTypeNode */ - public $identifier; + public IdentifierTypeNode $identifier; + + /** @var TemplateTagValueNode[] */ + public array $templateTypes; /** @var CallableTypeParameterNode[] */ - public $parameters; + public array $parameters; - /** @var TypeNode */ - public $returnType; + public TypeNode $returnType; /** * @param CallableTypeParameterNode[] $parameters + * @param TemplateTagValueNode[] $templateTypes */ - public function __construct(IdentifierTypeNode $identifier, array $parameters, TypeNode $returnType) + public function __construct(IdentifierTypeNode $identifier, array $parameters, TypeNode $returnType, array $templateTypes) { $this->identifier = $identifier; $this->parameters = $parameters; $this->returnType = $returnType; + $this->templateTypes = $templateTypes; } @@ -36,8 +40,11 @@ public function __toString(): string if ($returnType instanceof self) { $returnType = "({$returnType})"; } + $template = $this->templateTypes !== [] + ? '<' . implode(', ', $this->templateTypes) . '>' + : ''; $parameters = implode(', ', $this->parameters); - return "{$this->identifier}({$parameters}): {$returnType}"; + return "{$this->identifier}{$template}({$parameters}): {$returnType}"; } } diff --git a/src/Ast/Type/CallableTypeParameterNode.php b/src/Ast/Type/CallableTypeParameterNode.php index c78d4c7b..08119670 100644 --- a/src/Ast/Type/CallableTypeParameterNode.php +++ b/src/Ast/Type/CallableTypeParameterNode.php @@ -11,20 +11,16 @@ class CallableTypeParameterNode implements Node use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; - /** @var bool */ - public $isReference; + public bool $isReference; - /** @var bool */ - public $isVariadic; + public bool $isVariadic; /** @var string (may be empty) */ - public $parameterName; + public string $parameterName; - /** @var bool */ - public $isOptional; + public bool $isOptional; public function __construct(TypeNode $type, bool $isReference, bool $isVariadic, string $parameterName, bool $isOptional) { diff --git a/src/Ast/Type/ConditionalTypeForParameterNode.php b/src/Ast/Type/ConditionalTypeForParameterNode.php index fbfcae95..4c120d22 100644 --- a/src/Ast/Type/ConditionalTypeForParameterNode.php +++ b/src/Ast/Type/ConditionalTypeForParameterNode.php @@ -10,20 +10,15 @@ class ConditionalTypeForParameterNode implements TypeNode use NodeAttributes; - /** @var string */ - public $parameterName; + public string $parameterName; - /** @var TypeNode */ - public $targetType; + public TypeNode $targetType; - /** @var TypeNode */ - public $if; + public TypeNode $if; - /** @var TypeNode */ - public $else; + public TypeNode $else; - /** @var bool */ - public $negated; + public bool $negated; public function __construct(string $parameterName, TypeNode $targetType, TypeNode $if, TypeNode $else, bool $negated) { @@ -42,7 +37,7 @@ public function __toString(): string $this->negated ? 'is not' : 'is', $this->targetType, $this->if, - $this->else + $this->else, ); } diff --git a/src/Ast/Type/ConditionalTypeNode.php b/src/Ast/Type/ConditionalTypeNode.php index bfdb0db1..89c1c63f 100644 --- a/src/Ast/Type/ConditionalTypeNode.php +++ b/src/Ast/Type/ConditionalTypeNode.php @@ -10,20 +10,15 @@ class ConditionalTypeNode implements TypeNode use NodeAttributes; - /** @var TypeNode */ - public $subjectType; + public TypeNode $subjectType; - /** @var TypeNode */ - public $targetType; + public TypeNode $targetType; - /** @var TypeNode */ - public $if; + public TypeNode $if; - /** @var TypeNode */ - public $else; + public TypeNode $else; - /** @var bool */ - public $negated; + public bool $negated; public function __construct(TypeNode $subjectType, TypeNode $targetType, TypeNode $if, TypeNode $else, bool $negated) { @@ -42,7 +37,7 @@ public function __toString(): string $this->negated ? 'is not' : 'is', $this->targetType, $this->if, - $this->else + $this->else, ); } diff --git a/src/Ast/Type/ConstTypeNode.php b/src/Ast/Type/ConstTypeNode.php index 0096055b..22823e5b 100644 --- a/src/Ast/Type/ConstTypeNode.php +++ b/src/Ast/Type/ConstTypeNode.php @@ -10,8 +10,7 @@ class ConstTypeNode implements TypeNode use NodeAttributes; - /** @var ConstExprNode */ - public $constExpr; + public ConstExprNode $constExpr; public function __construct(ConstExprNode $constExpr) { diff --git a/src/Ast/Type/GenericTypeNode.php b/src/Ast/Type/GenericTypeNode.php index 44e1d16d..4e52c00a 100644 --- a/src/Ast/Type/GenericTypeNode.php +++ b/src/Ast/Type/GenericTypeNode.php @@ -16,14 +16,13 @@ class GenericTypeNode implements TypeNode use NodeAttributes; - /** @var IdentifierTypeNode */ - public $type; + public IdentifierTypeNode $type; /** @var TypeNode[] */ - public $genericTypes; + public array $genericTypes; /** @var (self::VARIANCE_*)[] */ - public $variances; + public array $variances; /** * @param TypeNode[] $genericTypes diff --git a/src/Ast/Type/IdentifierTypeNode.php b/src/Ast/Type/IdentifierTypeNode.php index 29bac308..df93fa86 100644 --- a/src/Ast/Type/IdentifierTypeNode.php +++ b/src/Ast/Type/IdentifierTypeNode.php @@ -9,8 +9,7 @@ class IdentifierTypeNode implements TypeNode use NodeAttributes; - /** @var string */ - public $name; + public string $name; public function __construct(string $name) { diff --git a/src/Ast/Type/IntersectionTypeNode.php b/src/Ast/Type/IntersectionTypeNode.php index fd761cf7..b3059cf5 100644 --- a/src/Ast/Type/IntersectionTypeNode.php +++ b/src/Ast/Type/IntersectionTypeNode.php @@ -12,7 +12,7 @@ class IntersectionTypeNode implements TypeNode use NodeAttributes; /** @var TypeNode[] */ - public $types; + public array $types; /** * @param TypeNode[] $types diff --git a/src/Ast/Type/InvalidTypeNode.php b/src/Ast/Type/InvalidTypeNode.php index 1ec47cf6..318176e7 100644 --- a/src/Ast/Type/InvalidTypeNode.php +++ b/src/Ast/Type/InvalidTypeNode.php @@ -11,7 +11,7 @@ class InvalidTypeNode implements TypeNode use NodeAttributes; /** @var mixed[] */ - private $exceptionArgs; + private array $exceptionArgs; public function __construct(ParserException $exception) { diff --git a/src/Ast/Type/NullableTypeNode.php b/src/Ast/Type/NullableTypeNode.php index 73f438cd..080a13f2 100644 --- a/src/Ast/Type/NullableTypeNode.php +++ b/src/Ast/Type/NullableTypeNode.php @@ -9,8 +9,7 @@ class NullableTypeNode implements TypeNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; public function __construct(TypeNode $type) { diff --git a/src/Ast/Type/ObjectShapeItemNode.php b/src/Ast/Type/ObjectShapeItemNode.php index 2f012406..f7aa9efb 100644 --- a/src/Ast/Type/ObjectShapeItemNode.php +++ b/src/Ast/Type/ObjectShapeItemNode.php @@ -3,10 +3,11 @@ namespace PHPStan\PhpDocParser\Ast\Type; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprStringNode; +use PHPStan\PhpDocParser\Ast\Node; use PHPStan\PhpDocParser\Ast\NodeAttributes; use function sprintf; -class ObjectShapeItemNode implements TypeNode +class ObjectShapeItemNode implements Node { use NodeAttributes; @@ -14,11 +15,9 @@ class ObjectShapeItemNode implements TypeNode /** @var ConstExprStringNode|IdentifierTypeNode */ public $keyName; - /** @var bool */ - public $optional; + public bool $optional; - /** @var TypeNode */ - public $valueType; + public TypeNode $valueType; /** * @param ConstExprStringNode|IdentifierTypeNode $keyName @@ -38,7 +37,7 @@ public function __toString(): string '%s%s: %s', (string) $this->keyName, $this->optional ? '?' : '', - (string) $this->valueType + (string) $this->valueType, ); } diff --git a/src/Ast/Type/ObjectShapeNode.php b/src/Ast/Type/ObjectShapeNode.php index f418bc30..41dc68c3 100644 --- a/src/Ast/Type/ObjectShapeNode.php +++ b/src/Ast/Type/ObjectShapeNode.php @@ -11,7 +11,7 @@ class ObjectShapeNode implements TypeNode use NodeAttributes; /** @var ObjectShapeItemNode[] */ - public $items; + public array $items; /** * @param ObjectShapeItemNode[] $items diff --git a/src/Ast/Type/OffsetAccessTypeNode.php b/src/Ast/Type/OffsetAccessTypeNode.php index 39e83dfe..4bd67d8d 100644 --- a/src/Ast/Type/OffsetAccessTypeNode.php +++ b/src/Ast/Type/OffsetAccessTypeNode.php @@ -9,11 +9,9 @@ class OffsetAccessTypeNode implements TypeNode use NodeAttributes; - /** @var TypeNode */ - public $type; + public TypeNode $type; - /** @var TypeNode */ - public $offset; + public TypeNode $offset; public function __construct(TypeNode $type, TypeNode $offset) { @@ -25,7 +23,6 @@ public function __toString(): string { if ( $this->type instanceof CallableTypeNode - || $this->type instanceof ConstTypeNode || $this->type instanceof NullableTypeNode ) { return '(' . $this->type . ')[' . $this->offset . ']'; diff --git a/src/Ast/Type/UnionTypeNode.php b/src/Ast/Type/UnionTypeNode.php index c552dab5..602cb3dd 100644 --- a/src/Ast/Type/UnionTypeNode.php +++ b/src/Ast/Type/UnionTypeNode.php @@ -12,7 +12,7 @@ class UnionTypeNode implements TypeNode use NodeAttributes; /** @var TypeNode[] */ - public $types; + public array $types; /** * @param TypeNode[] $types diff --git a/src/Lexer/Lexer.php b/src/Lexer/Lexer.php index ccae6bef..b2669131 100644 --- a/src/Lexer/Lexer.php +++ b/src/Lexer/Lexer.php @@ -2,6 +2,7 @@ namespace PHPStan\PhpDocParser\Lexer; +use PHPStan\PhpDocParser\ParserConfig; use function implode; use function preg_match_all; use const PREG_SET_ORDER; @@ -30,23 +31,27 @@ class Lexer public const TOKEN_OPEN_PHPDOC = 15; public const TOKEN_CLOSE_PHPDOC = 16; public const TOKEN_PHPDOC_TAG = 17; - public const TOKEN_FLOAT = 18; - public const TOKEN_INTEGER = 19; - public const TOKEN_SINGLE_QUOTED_STRING = 20; - public const TOKEN_DOUBLE_QUOTED_STRING = 21; - public const TOKEN_IDENTIFIER = 22; - public const TOKEN_THIS_VARIABLE = 23; - public const TOKEN_VARIABLE = 24; - public const TOKEN_HORIZONTAL_WS = 25; - public const TOKEN_PHPDOC_EOL = 26; - public const TOKEN_OTHER = 27; - public const TOKEN_END = 28; - public const TOKEN_COLON = 29; - public const TOKEN_WILDCARD = 30; - public const TOKEN_OPEN_CURLY_BRACKET = 31; - public const TOKEN_CLOSE_CURLY_BRACKET = 32; - public const TOKEN_NEGATED = 33; - public const TOKEN_ARROW = 34; + public const TOKEN_DOCTRINE_TAG = 18; + public const TOKEN_FLOAT = 19; + public const TOKEN_INTEGER = 20; + public const TOKEN_SINGLE_QUOTED_STRING = 21; + public const TOKEN_DOUBLE_QUOTED_STRING = 22; + public const TOKEN_DOCTRINE_ANNOTATION_STRING = 23; + public const TOKEN_IDENTIFIER = 24; + public const TOKEN_THIS_VARIABLE = 25; + public const TOKEN_VARIABLE = 26; + public const TOKEN_HORIZONTAL_WS = 27; + public const TOKEN_PHPDOC_EOL = 28; + public const TOKEN_OTHER = 29; + public const TOKEN_END = 30; + public const TOKEN_COLON = 31; + public const TOKEN_WILDCARD = 32; + public const TOKEN_OPEN_CURLY_BRACKET = 33; + public const TOKEN_CLOSE_CURLY_BRACKET = 34; + public const TOKEN_NEGATED = 35; + public const TOKEN_ARROW = 36; + + public const TOKEN_COMMENT = 37; public const TOKEN_LABELS = [ self::TOKEN_REFERENCE => '\'&\'', @@ -63,6 +68,7 @@ class Lexer self::TOKEN_OPEN_CURLY_BRACKET => '\'{\'', self::TOKEN_CLOSE_CURLY_BRACKET => '\'}\'', self::TOKEN_COMMA => '\',\'', + self::TOKEN_COMMENT => '\'//\'', self::TOKEN_COLON => '\':\'', self::TOKEN_VARIADIC => '\'...\'', self::TOKEN_DOUBLE_COLON => '\'::\'', @@ -72,11 +78,13 @@ class Lexer self::TOKEN_OPEN_PHPDOC => '\'/**\'', self::TOKEN_CLOSE_PHPDOC => '\'*/\'', self::TOKEN_PHPDOC_TAG => 'TOKEN_PHPDOC_TAG', + self::TOKEN_DOCTRINE_TAG => 'TOKEN_DOCTRINE_TAG', self::TOKEN_PHPDOC_EOL => 'TOKEN_PHPDOC_EOL', self::TOKEN_FLOAT => 'TOKEN_FLOAT', self::TOKEN_INTEGER => 'TOKEN_INTEGER', self::TOKEN_SINGLE_QUOTED_STRING => 'TOKEN_SINGLE_QUOTED_STRING', self::TOKEN_DOUBLE_QUOTED_STRING => 'TOKEN_DOUBLE_QUOTED_STRING', + self::TOKEN_DOCTRINE_ANNOTATION_STRING => 'TOKEN_DOCTRINE_ANNOTATION_STRING', self::TOKEN_IDENTIFIER => 'type', self::TOKEN_THIS_VARIABLE => '\'$this\'', self::TOKEN_VARIABLE => 'variable', @@ -90,8 +98,15 @@ class Lexer public const TYPE_OFFSET = 1; public const LINE_OFFSET = 2; - /** @var string|null */ - private $regexp; + private ParserConfig $config; // @phpstan-ignore property.onlyWritten + + private ?string $regexp = null; + + public function __construct(ParserConfig $config) + { + $this->config = $config; + } + /** * @return list @@ -148,6 +163,7 @@ private function generateRegexp(): string self::TOKEN_CLOSE_CURLY_BRACKET => '\\}', self::TOKEN_COMMA => ',', + self::TOKEN_COMMENT => '\/\/[^\\r\\n]*(?=\n|\r|\*/)', self::TOKEN_VARIADIC => '\\.\\.\\.', self::TOKEN_DOUBLE_COLON => '::', self::TOKEN_DOUBLE_ARROW => '=>', @@ -158,12 +174,14 @@ private function generateRegexp(): string self::TOKEN_OPEN_PHPDOC => '/\\*\\*(?=\\s)\\x20?+', self::TOKEN_CLOSE_PHPDOC => '\\*/', self::TOKEN_PHPDOC_TAG => '@(?:[a-z][a-z0-9-\\\\]+:)?[a-z][a-z0-9-\\\\]*+', + self::TOKEN_DOCTRINE_TAG => '@[a-z_\\\\][a-z0-9_\:\\\\]*[a-z_][a-z0-9_]*', self::TOKEN_PHPDOC_EOL => '\\r?+\\n[\\x09\\x20]*+(?:\\*(?!/)\\x20?+)?', - self::TOKEN_FLOAT => '(?:-?[0-9]++(_[0-9]++)*\\.[0-9]*(_[0-9]++)*+(?:e-?[0-9]++(_[0-9]++)*)?)|(?:-?[0-9]*+(_[0-9]++)*\\.[0-9]++(_[0-9]++)*(?:e-?[0-9]++(_[0-9]++)*)?)|(?:-?[0-9]++(_[0-9]++)*e-?[0-9]++(_[0-9]++)*)', - self::TOKEN_INTEGER => '-?(?:(?:0b[0-1]++(_[0-1]++)*)|(?:0o[0-7]++(_[0-7]++)*)|(?:0x[0-9a-f]++(_[0-9a-f]++)*)|(?:[0-9]++(_[0-9]++)*))', + self::TOKEN_FLOAT => '[+\-]?(?:(?:[0-9]++(_[0-9]++)*\\.[0-9]*+(_[0-9]++)*(?:e[+\-]?[0-9]++(_[0-9]++)*)?)|(?:[0-9]*+(_[0-9]++)*\\.[0-9]++(_[0-9]++)*(?:e[+\-]?[0-9]++(_[0-9]++)*)?)|(?:[0-9]++(_[0-9]++)*e[+\-]?[0-9]++(_[0-9]++)*))', + self::TOKEN_INTEGER => '[+\-]?(?:(?:0b[0-1]++(_[0-1]++)*)|(?:0o[0-7]++(_[0-7]++)*)|(?:0x[0-9a-f]++(_[0-9a-f]++)*)|(?:[0-9]++(_[0-9]++)*))', self::TOKEN_SINGLE_QUOTED_STRING => '\'(?:\\\\[^\\r\\n]|[^\'\\r\\n\\\\])*+\'', self::TOKEN_DOUBLE_QUOTED_STRING => '"(?:\\\\[^\\r\\n]|[^"\\r\\n\\\\])*+"', + self::TOKEN_DOCTRINE_ANNOTATION_STRING => '"(?:""|[^"])*+"', self::TOKEN_WILDCARD => '\\*', diff --git a/src/Parser/ConstExprParser.php b/src/Parser/ConstExprParser.php index b6db8a2c..396b8d7c 100644 --- a/src/Parser/ConstExprParser.php +++ b/src/Parser/ConstExprParser.php @@ -4,41 +4,36 @@ use PHPStan\PhpDocParser\Ast; use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\ParserConfig; use function str_replace; use function strtolower; -use function substr; class ConstExprParser { - /** @var bool */ - private $unescapeStrings; + private ParserConfig $config; - /** @var bool */ - private $quoteAwareConstExprString; + private bool $parseDoctrineStrings; - /** @var bool */ - private $useLinesAttributes; - - /** @var bool */ - private $useIndexAttributes; + public function __construct( + ParserConfig $config + ) + { + $this->config = $config; + $this->parseDoctrineStrings = false; + } /** - * @param array{lines?: bool, indexes?: bool} $usedAttributes + * @internal */ - public function __construct( - bool $unescapeStrings = false, - bool $quoteAwareConstExprString = false, - array $usedAttributes = [] - ) + public function toDoctrine(): self { - $this->unescapeStrings = $unescapeStrings; - $this->quoteAwareConstExprString = $quoteAwareConstExprString; - $this->useLinesAttributes = $usedAttributes['lines'] ?? false; - $this->useIndexAttributes = $usedAttributes['indexes'] ?? false; + $self = new self($this->config); + $self->parseDoctrineStrings = true; + return $self; } - public function parse(TokenIterator $tokens, bool $trimStrings = false): Ast\ConstExpr\ConstExprNode + public function parse(TokenIterator $tokens): Ast\ConstExpr\ConstExprNode { $startLine = $tokens->currentTokenLine(); $startIndex = $tokens->currentTokenIndex(); @@ -50,7 +45,7 @@ public function parse(TokenIterator $tokens, bool $trimStrings = false): Ast\Con $tokens, new Ast\ConstExpr\ConstExprFloatNode(str_replace('_', '', $value)), $startLine, - $startIndex + $startIndex, ); } @@ -62,41 +57,60 @@ public function parse(TokenIterator $tokens, bool $trimStrings = false): Ast\Con $tokens, new Ast\ConstExpr\ConstExprIntegerNode(str_replace('_', '', $value)), $startLine, - $startIndex + $startIndex, ); } - if ($tokens->isCurrentTokenType(Lexer::TOKEN_SINGLE_QUOTED_STRING, Lexer::TOKEN_DOUBLE_QUOTED_STRING)) { + if ($this->parseDoctrineStrings && $tokens->isCurrentTokenType(Lexer::TOKEN_DOCTRINE_ANNOTATION_STRING)) { $value = $tokens->currentTokenValue(); - $type = $tokens->currentTokenType(); - if ($trimStrings) { - if ($this->unescapeStrings) { - $value = StringUnescaper::unescapeString($value); - } else { - $value = substr($value, 1, -1); - } - } $tokens->next(); - if ($this->quoteAwareConstExprString) { + return $this->enrichWithAttributes( + $tokens, + new Ast\ConstExpr\DoctrineConstExprStringNode(Ast\ConstExpr\DoctrineConstExprStringNode::unescape($value)), + $startLine, + $startIndex, + ); + } + + if ($tokens->isCurrentTokenType(Lexer::TOKEN_SINGLE_QUOTED_STRING, Lexer::TOKEN_DOUBLE_QUOTED_STRING)) { + if ($this->parseDoctrineStrings) { + if ($tokens->isCurrentTokenType(Lexer::TOKEN_SINGLE_QUOTED_STRING)) { + throw new ParserException( + $tokens->currentTokenValue(), + $tokens->currentTokenType(), + $tokens->currentTokenOffset(), + Lexer::TOKEN_DOUBLE_QUOTED_STRING, + null, + $tokens->currentTokenLine(), + ); + } + + $value = $tokens->currentTokenValue(); + $tokens->next(); + return $this->enrichWithAttributes( $tokens, - new Ast\ConstExpr\QuoteAwareConstExprStringNode( - $value, - $type === Lexer::TOKEN_SINGLE_QUOTED_STRING - ? Ast\ConstExpr\QuoteAwareConstExprStringNode::SINGLE_QUOTED - : Ast\ConstExpr\QuoteAwareConstExprStringNode::DOUBLE_QUOTED - ), + $this->parseDoctrineString($value, $tokens), $startLine, - $startIndex + $startIndex, ); } + $value = StringUnescaper::unescapeString($tokens->currentTokenValue()); + $type = $tokens->currentTokenType(); + $tokens->next(); + return $this->enrichWithAttributes( $tokens, - new Ast\ConstExpr\ConstExprStringNode($value), + new Ast\ConstExpr\ConstExprStringNode( + $value, + $type === Lexer::TOKEN_SINGLE_QUOTED_STRING + ? Ast\ConstExpr\ConstExprStringNode::SINGLE_QUOTED + : Ast\ConstExpr\ConstExprStringNode::DOUBLE_QUOTED, + ), $startLine, - $startIndex + $startIndex, ); } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_IDENTIFIER)) { @@ -109,21 +123,21 @@ public function parse(TokenIterator $tokens, bool $trimStrings = false): Ast\Con $tokens, new Ast\ConstExpr\ConstExprTrueNode(), $startLine, - $startIndex + $startIndex, ); case 'false': return $this->enrichWithAttributes( $tokens, new Ast\ConstExpr\ConstExprFalseNode(), $startLine, - $startIndex + $startIndex, ); case 'null': return $this->enrichWithAttributes( $tokens, new Ast\ConstExpr\ConstExprNullNode(), $startLine, - $startIndex + $startIndex, ); case 'array': $tokens->consumeTokenType(Lexer::TOKEN_OPEN_PARENTHESES); @@ -165,7 +179,7 @@ public function parse(TokenIterator $tokens, bool $trimStrings = false): Ast\Con $tokens, new Ast\ConstExpr\ConstFetchNode($identifier, $classConstantName), $startLine, - $startIndex + $startIndex, ); } @@ -174,7 +188,7 @@ public function parse(TokenIterator $tokens, bool $trimStrings = false): Ast\Con $tokens, new Ast\ConstExpr\ConstFetchNode('', $identifier), $startLine, - $startIndex + $startIndex, ); } elseif ($tokens->tryConsumeTokenType(Lexer::TOKEN_OPEN_SQUARE_BRACKET)) { @@ -187,7 +201,7 @@ public function parse(TokenIterator $tokens, bool $trimStrings = false): Ast\Con $tokens->currentTokenOffset(), Lexer::TOKEN_IDENTIFIER, null, - $tokens->currentTokenLine() + $tokens->currentTokenLine(), ); } @@ -209,11 +223,28 @@ private function parseArray(TokenIterator $tokens, int $endToken, int $startInde $tokens, new Ast\ConstExpr\ConstExprArrayNode($items), $startLine, - $startIndex + $startIndex, ); } + /** + * This method is supposed to be called with TokenIterator after reading TOKEN_DOUBLE_QUOTED_STRING and shifting + * to the next token. + */ + public function parseDoctrineString(string $text, TokenIterator $tokens): Ast\ConstExpr\DoctrineConstExprStringNode + { + // Because of how Lexer works, a valid Doctrine string + // can consist of a sequence of TOKEN_DOUBLE_QUOTED_STRING and TOKEN_DOCTRINE_ANNOTATION_STRING + while ($tokens->isCurrentTokenType(Lexer::TOKEN_DOUBLE_QUOTED_STRING, Lexer::TOKEN_DOCTRINE_ANNOTATION_STRING)) { + $text .= $tokens->currentTokenValue(); + $tokens->next(); + } + + return new Ast\ConstExpr\DoctrineConstExprStringNode(Ast\ConstExpr\DoctrineConstExprStringNode::unescape($text)); + } + + private function parseArrayItem(TokenIterator $tokens): Ast\ConstExpr\ConstExprArrayItemNode { $startLine = $tokens->currentTokenLine(); @@ -234,7 +265,7 @@ private function parseArrayItem(TokenIterator $tokens): Ast\ConstExpr\ConstExprA $tokens, new Ast\ConstExpr\ConstExprArrayItemNode($key, $value), $startLine, - $startIndex + $startIndex, ); } @@ -245,22 +276,14 @@ private function parseArrayItem(TokenIterator $tokens): Ast\ConstExpr\ConstExprA */ private function enrichWithAttributes(TokenIterator $tokens, Ast\ConstExpr\ConstExprNode $node, int $startLine, int $startIndex): Ast\ConstExpr\ConstExprNode { - $endLine = $tokens->currentTokenLine(); - $endIndex = $tokens->currentTokenIndex(); - if ($this->useLinesAttributes) { + if ($this->config->useLinesAttributes) { $node->setAttribute(Ast\Attribute::START_LINE, $startLine); - $node->setAttribute(Ast\Attribute::END_LINE, $endLine); + $node->setAttribute(Ast\Attribute::END_LINE, $tokens->currentTokenLine()); } - if ($this->useIndexAttributes) { - $tokensArray = $tokens->getTokens(); - $endIndex--; - if ($tokensArray[$endIndex][Lexer::TYPE_OFFSET] === Lexer::TOKEN_HORIZONTAL_WS) { - $endIndex--; - } - + if ($this->config->useIndexAttributes) { $node->setAttribute(Ast\Attribute::START_INDEX, $startIndex); - $node->setAttribute(Ast\Attribute::END_INDEX, $endIndex); + $node->setAttribute(Ast\Attribute::END_INDEX, $tokens->endIndexOfLastRelevantToken()); } return $node; diff --git a/src/Parser/ParserException.php b/src/Parser/ParserException.php index 6ab5cc07..ae72fd9c 100644 --- a/src/Parser/ParserException.php +++ b/src/Parser/ParserException.php @@ -14,31 +14,25 @@ class ParserException extends Exception { - /** @var string */ - private $currentTokenValue; + private string $currentTokenValue; - /** @var int */ - private $currentTokenType; + private int $currentTokenType; - /** @var int */ - private $currentOffset; + private int $currentOffset; - /** @var int */ - private $expectedTokenType; + private int $expectedTokenType; - /** @var string|null */ - private $expectedTokenValue; + private ?string $expectedTokenValue; - /** @var int|null */ - private $currentTokenLine; + private ?int $currentTokenLine; public function __construct( string $currentTokenValue, int $currentTokenType, int $currentOffset, int $expectedTokenType, - ?string $expectedTokenValue = null, - ?int $currentTokenLine = null + ?string $expectedTokenValue, + ?int $currentTokenLine ) { $this->currentTokenValue = $currentTokenValue; @@ -54,7 +48,7 @@ public function __construct( Lexer::TOKEN_LABELS[$expectedTokenType], $expectedTokenValue !== null ? sprintf(' (%s)', $this->formatValue($expectedTokenValue)) : '', $currentOffset, - $currentTokenLine === null ? '' : sprintf(' on line %d', $currentTokenLine) + $currentTokenLine === null ? '' : sprintf(' on line %d', $currentTokenLine), )); } diff --git a/src/Parser/PhpDocParser.php b/src/Parser/PhpDocParser.php index a6401b65..559d8fd5 100644 --- a/src/Parser/PhpDocParser.php +++ b/src/Parser/PhpDocParser.php @@ -2,15 +2,25 @@ namespace PHPStan\PhpDocParser\Parser; +use LogicException; use PHPStan\PhpDocParser\Ast; +use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprIntegerNode; +use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprStringNode; +use PHPStan\PhpDocParser\Ast\ConstExpr\ConstFetchNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine; use PHPStan\PhpDocParser\Ast\Type\IdentifierTypeNode; use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\ParserConfig; use PHPStan\ShouldNotHappenException; use function array_key_exists; -use function array_values; use function count; +use function rtrim; +use function str_replace; use function trim; +/** + * @phpstan-import-type ValueType from Doctrine\DoctrineArgument as DoctrineValueType + */ class PhpDocParser { @@ -19,41 +29,24 @@ class PhpDocParser Lexer::TOKEN_INTERSECTION, ]; - /** @var TypeParser */ - private $typeParser; + private ParserConfig $config; - /** @var ConstExprParser */ - private $constantExprParser; + private TypeParser $typeParser; - /** @var bool */ - private $requireWhitespaceBeforeDescription; + private ConstExprParser $constantExprParser; - /** @var bool */ - private $preserveTypeAliasesWithInvalidTypes; + private ConstExprParser $doctrineConstantExprParser; - /** @var bool */ - private $useLinesAttributes; - - /** @var bool */ - private $useIndexAttributes; - - /** - * @param array{lines?: bool, indexes?: bool} $usedAttributes - */ public function __construct( + ParserConfig $config, TypeParser $typeParser, - ConstExprParser $constantExprParser, - bool $requireWhitespaceBeforeDescription = false, - bool $preserveTypeAliasesWithInvalidTypes = false, - array $usedAttributes = [] + ConstExprParser $constantExprParser ) { + $this->config = $config; $this->typeParser = $typeParser; $this->constantExprParser = $constantExprParser; - $this->requireWhitespaceBeforeDescription = $requireWhitespaceBeforeDescription; - $this->preserveTypeAliasesWithInvalidTypes = $preserveTypeAliasesWithInvalidTypes; - $this->useLinesAttributes = $usedAttributes['lines'] ?? false; - $this->useIndexAttributes = $usedAttributes['indexes'] ?? false; + $this->doctrineConstantExprParser = $constantExprParser->toDoctrine(); } @@ -65,9 +58,34 @@ public function parse(TokenIterator $tokens): Ast\PhpDoc\PhpDocNode $children = []; if (!$tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PHPDOC)) { - $children[] = $this->parseChild($tokens); - while ($tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL) && !$tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PHPDOC)) { - $children[] = $this->parseChild($tokens); + $lastChild = $this->parseChild($tokens); + $children[] = $lastChild; + while (!$tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PHPDOC)) { + if ( + $lastChild instanceof Ast\PhpDoc\PhpDocTagNode + && ( + $lastChild->value instanceof Doctrine\DoctrineTagValueNode + || $lastChild->value instanceof Ast\PhpDoc\GenericTagValueNode + ) + ) { + $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + if ($tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PHPDOC)) { + break; + } + $lastChild = $this->parseChild($tokens); + $children[] = $lastChild; + continue; + } + + if (!$tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL)) { + break; + } + if ($tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PHPDOC)) { + break; + } + + $lastChild = $this->parseChild($tokens); + $children[] = $lastChild; } } @@ -92,19 +110,30 @@ public function parse(TokenIterator $tokens): Ast\PhpDoc\PhpDocNode $tokens, new Ast\PhpDoc\InvalidTagValueNode($e->getMessage(), $e), $startLine, - $startIndex - ) + $startIndex, + ), ); $tokens->forwardToTheEnd(); + $comments = $tokens->flushComments(); + if ($comments !== []) { + throw new LogicException('Comments should already be flushed'); + } + return $this->enrichWithAttributes($tokens, new Ast\PhpDoc\PhpDocNode([$this->enrichWithAttributes($tokens, $tag, $startLine, $startIndex)]), 1, 0); } - return $this->enrichWithAttributes($tokens, new Ast\PhpDoc\PhpDocNode(array_values($children)), 1, 0); + $comments = $tokens->flushComments(); + if ($comments !== []) { + throw new LogicException('Comments should already be flushed'); + } + + return $this->enrichWithAttributes($tokens, new Ast\PhpDoc\PhpDocNode($children), 1, 0); } + /** @phpstan-impure */ private function parseChild(TokenIterator $tokens): Ast\PhpDoc\PhpDocChildNode { if ($tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_TAG)) { @@ -113,6 +142,26 @@ private function parseChild(TokenIterator $tokens): Ast\PhpDoc\PhpDocChildNode return $this->enrichWithAttributes($tokens, $this->parseTag($tokens), $startLine, $startIndex); } + if ($tokens->isCurrentTokenType(Lexer::TOKEN_DOCTRINE_TAG)) { + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + $tag = $tokens->currentTokenValue(); + $tokens->next(); + + $tagStartLine = $tokens->currentTokenLine(); + $tagStartIndex = $tokens->currentTokenIndex(); + + return $this->enrichWithAttributes($tokens, new Ast\PhpDoc\PhpDocTagNode( + $tag, + $this->enrichWithAttributes( + $tokens, + $this->parseDoctrineTagValue($tokens, $tag), + $tagStartLine, + $tagStartIndex, + ), + ), $startLine, $startIndex); + } + $startLine = $tokens->currentTokenLine(); $startIndex = $tokens->currentTokenIndex(); $text = $this->parseText($tokens); @@ -127,23 +176,14 @@ private function parseChild(TokenIterator $tokens): Ast\PhpDoc\PhpDocChildNode */ private function enrichWithAttributes(TokenIterator $tokens, Ast\Node $tag, int $startLine, int $startIndex): Ast\Node { - $endLine = $tokens->currentTokenLine(); - $endIndex = $tokens->currentTokenIndex(); - - if ($this->useLinesAttributes) { + if ($this->config->useLinesAttributes) { $tag->setAttribute(Ast\Attribute::START_LINE, $startLine); - $tag->setAttribute(Ast\Attribute::END_LINE, $endLine); + $tag->setAttribute(Ast\Attribute::END_LINE, $tokens->currentTokenLine()); } - if ($this->useIndexAttributes) { - $tokensArray = $tokens->getTokens(); - $endIndex--; - if ($tokensArray[$endIndex][Lexer::TYPE_OFFSET] === Lexer::TOKEN_HORIZONTAL_WS) { - $endIndex--; - } - + if ($this->config->useIndexAttributes) { $tag->setAttribute(Ast\Attribute::START_INDEX, $startIndex); - $tag->setAttribute(Ast\Attribute::END_INDEX, $endIndex); + $tag->setAttribute(Ast\Attribute::END_INDEX, $tokens->endIndexOfLastRelevantToken()); } return $tag; @@ -154,29 +194,134 @@ private function parseText(TokenIterator $tokens): Ast\PhpDoc\PhpDocTextNode { $text = ''; - while (!$tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL)) { - $text .= $tokens->getSkippedHorizontalWhiteSpaceIfAny() . $tokens->joinUntil(Lexer::TOKEN_PHPDOC_EOL, Lexer::TOKEN_CLOSE_PHPDOC, Lexer::TOKEN_END); + $endTokens = [Lexer::TOKEN_CLOSE_PHPDOC, Lexer::TOKEN_END]; + + $savepoint = false; - if (!$tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL)) { + // if the next token is EOL, everything below is skipped and empty string is returned + while (true) { + $tmpText = $tokens->getSkippedHorizontalWhiteSpaceIfAny() . $tokens->joinUntil(Lexer::TOKEN_PHPDOC_EOL, ...$endTokens); + $text .= $tmpText; + + // stop if we're not at EOL - meaning it's the end of PHPDoc + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL, Lexer::TOKEN_CLOSE_PHPDOC)) { break; } + if (!$savepoint) { + $tokens->pushSavePoint(); + $savepoint = true; + } elseif ($tmpText !== '') { + $tokens->dropSavePoint(); + $tokens->pushSavePoint(); + } + $tokens->pushSavePoint(); $tokens->next(); - if ($tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_TAG, Lexer::TOKEN_PHPDOC_EOL, Lexer::TOKEN_CLOSE_PHPDOC, Lexer::TOKEN_END)) { + // if we're at EOL, check what's next + // if next is a PHPDoc tag, EOL, or end of PHPDoc, stop + if ($tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_TAG, Lexer::TOKEN_DOCTRINE_TAG, ...$endTokens)) { $tokens->rollback(); break; } + // otherwise if the next is text, continue building the description string + $tokens->dropSavePoint(); - $text .= "\n"; + $text .= $tokens->getDetectedNewline() ?? "\n"; + } + + if ($savepoint) { + $tokens->rollback(); + $text = rtrim($text, $tokens->getDetectedNewline() ?? "\n"); } return new Ast\PhpDoc\PhpDocTextNode(trim($text, " \t")); } + private function parseOptionalDescriptionAfterDoctrineTag(TokenIterator $tokens): string + { + $text = ''; + + $endTokens = [Lexer::TOKEN_CLOSE_PHPDOC, Lexer::TOKEN_END]; + + $savepoint = false; + + // if the next token is EOL, everything below is skipped and empty string is returned + while (true) { + $tmpText = $tokens->getSkippedHorizontalWhiteSpaceIfAny() . $tokens->joinUntil(Lexer::TOKEN_PHPDOC_TAG, Lexer::TOKEN_DOCTRINE_TAG, Lexer::TOKEN_PHPDOC_EOL, ...$endTokens); + $text .= $tmpText; + + // stop if we're not at EOL - meaning it's the end of PHPDoc + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL, Lexer::TOKEN_CLOSE_PHPDOC)) { + if (!$tokens->isPrecededByHorizontalWhitespace()) { + return trim($text . $this->parseText($tokens)->text, " \t"); + } + if ($tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_TAG)) { + $tokens->pushSavePoint(); + $child = $this->parseChild($tokens); + if ($child instanceof Ast\PhpDoc\PhpDocTagNode) { + if ( + $child->value instanceof Ast\PhpDoc\GenericTagValueNode + || $child->value instanceof Doctrine\DoctrineTagValueNode + ) { + $tokens->rollback(); + break; + } + if ($child->value instanceof Ast\PhpDoc\InvalidTagValueNode) { + $tokens->rollback(); + $tokens->pushSavePoint(); + $tokens->next(); + if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_PARENTHESES)) { + $tokens->rollback(); + break; + } + $tokens->rollback(); + return trim($text . $this->parseText($tokens)->text, " \t"); + } + } + + $tokens->rollback(); + return trim($text . $this->parseText($tokens)->text, " \t"); + } + break; + } + + if (!$savepoint) { + $tokens->pushSavePoint(); + $savepoint = true; + } elseif ($tmpText !== '') { + $tokens->dropSavePoint(); + $tokens->pushSavePoint(); + } + + $tokens->pushSavePoint(); + $tokens->next(); + + // if we're at EOL, check what's next + // if next is a PHPDoc tag, EOL, or end of PHPDoc, stop + if ($tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_TAG, Lexer::TOKEN_DOCTRINE_TAG, ...$endTokens)) { + $tokens->rollback(); + break; + } + + // otherwise if the next is text, continue building the description string + + $tokens->dropSavePoint(); + $text .= $tokens->getDetectedNewline() ?? "\n"; + } + + if ($savepoint) { + $tokens->rollback(); + $text = rtrim($text, $tokens->getDetectedNewline() ?? "\n"); + } + + return trim($text, " \t"); + } + + public function parseTag(TokenIterator $tokens): Ast\PhpDoc\PhpDocTagNode { $tag = $tokens->currentTokenValue(); @@ -199,18 +344,42 @@ public function parseTagValue(TokenIterator $tokens, string $tag): Ast\PhpDoc\Ph case '@param': case '@phpstan-param': case '@psalm-param': + case '@phan-param': $tagValue = $this->parseParamTagValue($tokens); break; + case '@param-immediately-invoked-callable': + case '@phpstan-param-immediately-invoked-callable': + $tagValue = $this->parseParamImmediatelyInvokedCallableTagValue($tokens); + break; + + case '@param-later-invoked-callable': + case '@phpstan-param-later-invoked-callable': + $tagValue = $this->parseParamLaterInvokedCallableTagValue($tokens); + break; + + case '@param-closure-this': + case '@phpstan-param-closure-this': + $tagValue = $this->parseParamClosureThisTagValue($tokens); + break; + + case '@pure-unless-callable-is-impure': + case '@phpstan-pure-unless-callable-is-impure': + $tagValue = $this->parsePureUnlessCallableIsImpureTagValue($tokens); + break; + case '@var': case '@phpstan-var': case '@psalm-var': + case '@phan-var': $tagValue = $this->parseVarTagValue($tokens); break; case '@return': case '@phpstan-return': case '@psalm-return': + case '@phan-return': + case '@phan-real-return': $tagValue = $this->parseReturnTagValue($tokens); break; @@ -220,9 +389,20 @@ public function parseTagValue(TokenIterator $tokens, string $tag): Ast\PhpDoc\Ph break; case '@mixin': + case '@phan-mixin': $tagValue = $this->parseMixinTagValue($tokens); break; + case '@psalm-require-extends': + case '@phpstan-require-extends': + $tagValue = $this->parseRequireExtendsTagValue($tokens); + break; + + case '@psalm-require-implements': + case '@phpstan-require-implements': + $tagValue = $this->parseRequireImplementsTagValue($tokens); + break; + case '@deprecated': $tagValue = $this->parseDeprecatedTagValue($tokens); break; @@ -236,29 +416,39 @@ public function parseTagValue(TokenIterator $tokens, string $tag): Ast\PhpDoc\Ph case '@psalm-property': case '@psalm-property-read': case '@psalm-property-write': + case '@phan-property': + case '@phan-property-read': + case '@phan-property-write': $tagValue = $this->parsePropertyTagValue($tokens); break; case '@method': case '@phpstan-method': case '@psalm-method': + case '@phan-method': $tagValue = $this->parseMethodTagValue($tokens); break; case '@template': case '@phpstan-template': case '@psalm-template': + case '@phan-template': case '@template-covariant': case '@phpstan-template-covariant': case '@psalm-template-covariant': case '@template-contravariant': case '@phpstan-template-contravariant': case '@psalm-template-contravariant': - $tagValue = $this->parseTemplateTagValue($tokens, true); + $tagValue = $this->typeParser->parseTemplateTagValue( + $tokens, + fn ($tokens) => $this->parseOptionalDescription($tokens, true), + ); break; case '@extends': case '@phpstan-extends': + case '@phan-extends': + case '@phan-inherits': case '@template-extends': $tagValue = $this->parseExtendsTagValue('@extends', $tokens); break; @@ -277,6 +467,7 @@ public function parseTagValue(TokenIterator $tokens, string $tag): Ast\PhpDoc\Ph case '@phpstan-type': case '@psalm-type': + case '@phan-type': $tagValue = $this->parseTypeAliasTagValue($tokens); break; @@ -291,6 +482,9 @@ public function parseTagValue(TokenIterator $tokens, string $tag): Ast\PhpDoc\Ph case '@psalm-assert': case '@psalm-assert-if-true': case '@psalm-assert-if-false': + case '@phan-assert': + case '@phan-assert-if-true': + case '@phan-assert-if-false': $tagValue = $this->parseAssertTagValue($tokens); break; @@ -308,7 +502,11 @@ public function parseTagValue(TokenIterator $tokens, string $tag): Ast\PhpDoc\Ph break; default: - $tagValue = new Ast\PhpDoc\GenericTagValueNode($this->parseOptionalDescription($tokens)); + if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_PARENTHESES)) { + $tagValue = $this->parseDoctrineTagValue($tokens, $tag); + } else { + $tagValue = new Ast\PhpDoc\GenericTagValueNode($this->parseOptionalDescriptionAfterDoctrineTag($tokens)); + } break; } @@ -316,13 +514,319 @@ public function parseTagValue(TokenIterator $tokens, string $tag): Ast\PhpDoc\Ph } catch (ParserException $e) { $tokens->rollback(); - $tagValue = new Ast\PhpDoc\InvalidTagValueNode($this->parseOptionalDescription($tokens), $e); + $tagValue = new Ast\PhpDoc\InvalidTagValueNode($this->parseOptionalDescription($tokens, false), $e); } return $this->enrichWithAttributes($tokens, $tagValue, $startLine, $startIndex); } + private function parseDoctrineTagValue(TokenIterator $tokens, string $tag): Ast\PhpDoc\PhpDocTagValueNode + { + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + + return new Doctrine\DoctrineTagValueNode( + $this->enrichWithAttributes( + $tokens, + new Doctrine\DoctrineAnnotation($tag, $this->parseDoctrineArguments($tokens, false)), + $startLine, + $startIndex, + ), + $this->parseOptionalDescriptionAfterDoctrineTag($tokens), + ); + } + + + /** + * @return list + */ + private function parseDoctrineArguments(TokenIterator $tokens, bool $deep): array + { + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_PARENTHESES)) { + return []; + } + + if (!$deep) { + $tokens->addEndOfLineToSkippedTokens(); + } + + $arguments = []; + + try { + $tokens->consumeTokenType(Lexer::TOKEN_OPEN_PARENTHESES); + + do { + if ($tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PARENTHESES)) { + break; + } + $arguments[] = $this->parseDoctrineArgument($tokens); + } while ($tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)); + } finally { + if (!$deep) { + $tokens->removeEndOfLineFromSkippedTokens(); + } + } + + $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_PARENTHESES); + + return $arguments; + } + + + private function parseDoctrineArgument(TokenIterator $tokens): Doctrine\DoctrineArgument + { + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_IDENTIFIER)) { + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + + return $this->enrichWithAttributes( + $tokens, + new Doctrine\DoctrineArgument(null, $this->parseDoctrineArgumentValue($tokens)), + $startLine, + $startIndex, + ); + } + + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + + try { + $tokens->pushSavePoint(); + $currentValue = $tokens->currentTokenValue(); + $tokens->consumeTokenType(Lexer::TOKEN_IDENTIFIER); + + $key = $this->enrichWithAttributes( + $tokens, + new IdentifierTypeNode($currentValue), + $startLine, + $startIndex, + ); + $tokens->consumeTokenType(Lexer::TOKEN_EQUAL); + + $value = $this->parseDoctrineArgumentValue($tokens); + + $tokens->dropSavePoint(); + + return $this->enrichWithAttributes( + $tokens, + new Doctrine\DoctrineArgument($key, $value), + $startLine, + $startIndex, + ); + } catch (ParserException $e) { + $tokens->rollback(); + + return $this->enrichWithAttributes( + $tokens, + new Doctrine\DoctrineArgument(null, $this->parseDoctrineArgumentValue($tokens)), + $startLine, + $startIndex, + ); + } + } + + + /** + * @return DoctrineValueType + */ + private function parseDoctrineArgumentValue(TokenIterator $tokens) + { + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + + if ($tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_TAG, Lexer::TOKEN_DOCTRINE_TAG)) { + $name = $tokens->currentTokenValue(); + $tokens->next(); + + return $this->enrichWithAttributes( + $tokens, + new Doctrine\DoctrineAnnotation($name, $this->parseDoctrineArguments($tokens, true)), + $startLine, + $startIndex, + ); + } + + if ($tokens->tryConsumeTokenType(Lexer::TOKEN_OPEN_CURLY_BRACKET)) { + $items = []; + do { + if ($tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_CURLY_BRACKET)) { + break; + } + $items[] = $this->parseDoctrineArrayItem($tokens); + } while ($tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)); + + $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_CURLY_BRACKET); + + return $this->enrichWithAttributes( + $tokens, + new Doctrine\DoctrineArray($items), + $startLine, + $startIndex, + ); + } + + $currentTokenValue = $tokens->currentTokenValue(); + $tokens->pushSavePoint(); // because of ConstFetchNode + if ($tokens->tryConsumeTokenType(Lexer::TOKEN_IDENTIFIER)) { + $identifier = $this->enrichWithAttributes( + $tokens, + new Ast\Type\IdentifierTypeNode($currentTokenValue), + $startLine, + $startIndex, + ); + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_DOUBLE_COLON)) { + $tokens->dropSavePoint(); + return $identifier; + } + + $tokens->rollback(); // because of ConstFetchNode + } else { + $tokens->dropSavePoint(); // because of ConstFetchNode + } + + $currentTokenValue = $tokens->currentTokenValue(); + $currentTokenType = $tokens->currentTokenType(); + $currentTokenOffset = $tokens->currentTokenOffset(); + $currentTokenLine = $tokens->currentTokenLine(); + + try { + $constExpr = $this->doctrineConstantExprParser->parse($tokens); + if ($constExpr instanceof Ast\ConstExpr\ConstExprArrayNode) { + throw new ParserException( + $currentTokenValue, + $currentTokenType, + $currentTokenOffset, + Lexer::TOKEN_IDENTIFIER, + null, + $currentTokenLine, + ); + } + + return $constExpr; + } catch (LogicException $e) { + throw new ParserException( + $currentTokenValue, + $currentTokenType, + $currentTokenOffset, + Lexer::TOKEN_IDENTIFIER, + null, + $currentTokenLine, + ); + } + } + + + private function parseDoctrineArrayItem(TokenIterator $tokens): Doctrine\DoctrineArrayItem + { + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + + try { + $tokens->pushSavePoint(); + + $key = $this->parseDoctrineArrayKey($tokens); + if (!$tokens->tryConsumeTokenType(Lexer::TOKEN_EQUAL)) { + if (!$tokens->tryConsumeTokenType(Lexer::TOKEN_COLON)) { + $tokens->consumeTokenType(Lexer::TOKEN_EQUAL); // will throw exception + } + } + + $value = $this->parseDoctrineArgumentValue($tokens); + + $tokens->dropSavePoint(); + + return $this->enrichWithAttributes( + $tokens, + new Doctrine\DoctrineArrayItem($key, $value), + $startLine, + $startIndex, + ); + } catch (ParserException $e) { + $tokens->rollback(); + + return $this->enrichWithAttributes( + $tokens, + new Doctrine\DoctrineArrayItem(null, $this->parseDoctrineArgumentValue($tokens)), + $startLine, + $startIndex, + ); + } + } + + + /** + * @return ConstExprIntegerNode|ConstExprStringNode|IdentifierTypeNode|ConstFetchNode + */ + private function parseDoctrineArrayKey(TokenIterator $tokens) + { + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + + if ($tokens->isCurrentTokenType(Lexer::TOKEN_INTEGER)) { + $key = new Ast\ConstExpr\ConstExprIntegerNode(str_replace('_', '', $tokens->currentTokenValue())); + $tokens->next(); + + } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_DOCTRINE_ANNOTATION_STRING)) { + $key = $this->doctrineConstantExprParser->parseDoctrineString($tokens->currentTokenValue(), $tokens); + + $tokens->next(); + + } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_SINGLE_QUOTED_STRING)) { + $key = new Ast\ConstExpr\ConstExprStringNode(StringUnescaper::unescapeString($tokens->currentTokenValue()), Ast\ConstExpr\ConstExprStringNode::SINGLE_QUOTED); + $tokens->next(); + + } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_DOUBLE_QUOTED_STRING)) { + $value = $tokens->currentTokenValue(); + $tokens->next(); + $key = $this->doctrineConstantExprParser->parseDoctrineString($value, $tokens); + + } else { + $currentTokenValue = $tokens->currentTokenValue(); + $tokens->pushSavePoint(); // because of ConstFetchNode + if (!$tokens->tryConsumeTokenType(Lexer::TOKEN_IDENTIFIER)) { + $tokens->dropSavePoint(); + throw new ParserException( + $tokens->currentTokenValue(), + $tokens->currentTokenType(), + $tokens->currentTokenOffset(), + Lexer::TOKEN_IDENTIFIER, + null, + $tokens->currentTokenLine(), + ); + } + + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_DOUBLE_COLON)) { + $tokens->dropSavePoint(); + + return $this->enrichWithAttributes( + $tokens, + new IdentifierTypeNode($currentTokenValue), + $startLine, + $startIndex, + ); + } + + $tokens->rollback(); + $constExpr = $this->doctrineConstantExprParser->parse($tokens); + if (!$constExpr instanceof Ast\ConstExpr\ConstFetchNode) { + throw new ParserException( + $tokens->currentTokenValue(), + $tokens->currentTokenType(), + $tokens->currentTokenOffset(), + Lexer::TOKEN_IDENTIFIER, + null, + $tokens->currentTokenLine(), + ); + } + + return $constExpr; + } + + return $this->enrichWithAttributes($tokens, $key, $startLine, $startIndex); + } + + /** * @return Ast\PhpDoc\ParamTagValueNode|Ast\PhpDoc\TypelessParamTagValueNode */ @@ -339,7 +843,7 @@ private function parseParamTagValue(TokenIterator $tokens): Ast\PhpDoc\PhpDocTag $isReference = $tokens->tryConsumeTokenType(Lexer::TOKEN_REFERENCE); $isVariadic = $tokens->tryConsumeTokenType(Lexer::TOKEN_VARIADIC); $parameterName = $this->parseRequiredVariableName($tokens); - $description = $this->parseOptionalDescription($tokens); + $description = $this->parseOptionalDescription($tokens, false); if ($type !== null) { return new Ast\PhpDoc\ParamTagValueNode($type, $isVariadic, $parameterName, $description, $isReference); @@ -349,6 +853,41 @@ private function parseParamTagValue(TokenIterator $tokens): Ast\PhpDoc\PhpDocTag } + private function parseParamImmediatelyInvokedCallableTagValue(TokenIterator $tokens): Ast\PhpDoc\ParamImmediatelyInvokedCallableTagValueNode + { + $parameterName = $this->parseRequiredVariableName($tokens); + $description = $this->parseOptionalDescription($tokens, false); + + return new Ast\PhpDoc\ParamImmediatelyInvokedCallableTagValueNode($parameterName, $description); + } + + + private function parseParamLaterInvokedCallableTagValue(TokenIterator $tokens): Ast\PhpDoc\ParamLaterInvokedCallableTagValueNode + { + $parameterName = $this->parseRequiredVariableName($tokens); + $description = $this->parseOptionalDescription($tokens, false); + + return new Ast\PhpDoc\ParamLaterInvokedCallableTagValueNode($parameterName, $description); + } + + + private function parseParamClosureThisTagValue(TokenIterator $tokens): Ast\PhpDoc\ParamClosureThisTagValueNode + { + $type = $this->typeParser->parse($tokens); + $parameterName = $this->parseRequiredVariableName($tokens); + $description = $this->parseOptionalDescription($tokens, false); + + return new Ast\PhpDoc\ParamClosureThisTagValueNode($type, $parameterName, $description); + } + + private function parsePureUnlessCallableIsImpureTagValue(TokenIterator $tokens): Ast\PhpDoc\PureUnlessCallableIsImpureTagValueNode + { + $parameterName = $this->parseRequiredVariableName($tokens); + $description = $this->parseOptionalDescription($tokens, false); + + return new Ast\PhpDoc\PureUnlessCallableIsImpureTagValueNode($parameterName, $description); + } + private function parseVarTagValue(TokenIterator $tokens): Ast\PhpDoc\VarTagValueNode { $type = $this->typeParser->parse($tokens); @@ -380,9 +919,23 @@ private function parseMixinTagValue(TokenIterator $tokens): Ast\PhpDoc\MixinTagV return new Ast\PhpDoc\MixinTagValueNode($type, $description); } + private function parseRequireExtendsTagValue(TokenIterator $tokens): Ast\PhpDoc\RequireExtendsTagValueNode + { + $type = $this->typeParser->parse($tokens); + $description = $this->parseOptionalDescription($tokens, true); + return new Ast\PhpDoc\RequireExtendsTagValueNode($type, $description); + } + + private function parseRequireImplementsTagValue(TokenIterator $tokens): Ast\PhpDoc\RequireImplementsTagValueNode + { + $type = $this->typeParser->parse($tokens); + $description = $this->parseOptionalDescription($tokens, true); + return new Ast\PhpDoc\RequireImplementsTagValueNode($type, $description); + } + private function parseDeprecatedTagValue(TokenIterator $tokens): Ast\PhpDoc\DeprecatedTagValueNode { - $description = $this->parseOptionalDescription($tokens); + $description = $this->parseOptionalDescription($tokens, false); return new Ast\PhpDoc\DeprecatedTagValueNode($description); } @@ -391,17 +944,23 @@ private function parsePropertyTagValue(TokenIterator $tokens): Ast\PhpDoc\Proper { $type = $this->typeParser->parse($tokens); $parameterName = $this->parseRequiredVariableName($tokens); - $description = $this->parseOptionalDescription($tokens); + $description = $this->parseOptionalDescription($tokens, false); return new Ast\PhpDoc\PropertyTagValueNode($type, $parameterName, $description); } private function parseMethodTagValue(TokenIterator $tokens): Ast\PhpDoc\MethodTagValueNode { - $isStatic = $tokens->tryConsumeTokenValue('static'); - $startLine = $tokens->currentTokenLine(); - $startIndex = $tokens->currentTokenIndex(); - $returnTypeOrMethodName = $this->typeParser->parse($tokens); + $staticKeywordOrReturnTypeOrMethodName = $this->typeParser->parse($tokens); + + if ($staticKeywordOrReturnTypeOrMethodName instanceof Ast\Type\IdentifierTypeNode && $staticKeywordOrReturnTypeOrMethodName->name === 'static') { + $isStatic = true; + $returnTypeOrMethodName = $this->typeParser->parse($tokens); + + } else { + $isStatic = false; + $returnTypeOrMethodName = $staticKeywordOrReturnTypeOrMethodName; + } if ($tokens->isCurrentTokenType(Lexer::TOKEN_IDENTIFIER)) { $returnType = $returnTypeOrMethodName; @@ -409,9 +968,7 @@ private function parseMethodTagValue(TokenIterator $tokens): Ast\PhpDoc\MethodTa $tokens->next(); } elseif ($returnTypeOrMethodName instanceof Ast\Type\IdentifierTypeNode) { - $returnType = $isStatic - ? $this->typeParser->enrichWithAttributes($tokens, new Ast\Type\IdentifierTypeNode('static'), $startLine, $startIndex) - : null; + $returnType = $isStatic ? $staticKeywordOrReturnTypeOrMethodName : null; $methodName = $returnTypeOrMethodName->name; $isStatic = false; @@ -426,7 +983,12 @@ private function parseMethodTagValue(TokenIterator $tokens): Ast\PhpDoc\MethodTa do { $startLine = $tokens->currentTokenLine(); $startIndex = $tokens->currentTokenIndex(); - $templateTypes[] = $this->enrichWithAttributes($tokens, $this->parseTemplateTagValue($tokens, false), $startLine, $startIndex); + $templateTypes[] = $this->enrichWithAttributes( + $tokens, + $this->typeParser->parseTemplateTagValue($tokens), + $startLine, + $startIndex, + ); } while ($tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)); $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_ANGLE_BRACKET); } @@ -441,7 +1003,7 @@ private function parseMethodTagValue(TokenIterator $tokens): Ast\PhpDoc\MethodTa } $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_PARENTHESES); - $description = $this->parseOptionalDescription($tokens); + $description = $this->parseOptionalDescription($tokens, false); return new Ast\PhpDoc\MethodTagValueNode($isStatic, $returnType, $methodName, $parameters, $description, $templateTypes); } @@ -478,37 +1040,10 @@ private function parseMethodTagValueParameter(TokenIterator $tokens): Ast\PhpDoc $tokens, new Ast\PhpDoc\MethodTagValueParameterNode($parameterType, $isReference, $isVariadic, $parameterName, $defaultValue), $startLine, - $startIndex + $startIndex, ); } - private function parseTemplateTagValue(TokenIterator $tokens, bool $parseDescription): Ast\PhpDoc\TemplateTagValueNode - { - $name = $tokens->currentTokenValue(); - $tokens->consumeTokenType(Lexer::TOKEN_IDENTIFIER); - - if ($tokens->tryConsumeTokenValue('of') || $tokens->tryConsumeTokenValue('as')) { - $bound = $this->typeParser->parse($tokens); - - } else { - $bound = null; - } - - if ($tokens->tryConsumeTokenValue('=')) { - $default = $this->typeParser->parse($tokens); - } else { - $default = null; - } - - if ($parseDescription) { - $description = $this->parseOptionalDescription($tokens); - } else { - $description = ''; - } - - return new Ast\PhpDoc\TemplateTagValueNode($name, $bound, $description, $default); - } - private function parseExtendsTagValue(string $tagName, TokenIterator $tokens): Ast\PhpDoc\PhpDocTagValueNode { $startLine = $tokens->currentTokenLine(); @@ -518,10 +1053,10 @@ private function parseExtendsTagValue(string $tagName, TokenIterator $tokens): A $type = $this->typeParser->parseGeneric( $tokens, - $this->typeParser->enrichWithAttributes($tokens, $baseType, $startLine, $startIndex) + $this->typeParser->enrichWithAttributes($tokens, $baseType, $startLine, $startIndex), ); - $description = $this->parseOptionalDescription($tokens); + $description = $this->parseOptionalDescription($tokens, true); switch ($tagName) { case '@extends': @@ -540,40 +1075,34 @@ private function parseTypeAliasTagValue(TokenIterator $tokens): Ast\PhpDoc\TypeA $alias = $tokens->currentTokenValue(); $tokens->consumeTokenType(Lexer::TOKEN_IDENTIFIER); - // support psalm-type syntax + // support phan-type/psalm-type syntax $tokens->tryConsumeTokenType(Lexer::TOKEN_EQUAL); - if ($this->preserveTypeAliasesWithInvalidTypes) { - $startLine = $tokens->currentTokenLine(); - $startIndex = $tokens->currentTokenIndex(); - try { - $type = $this->typeParser->parse($tokens); - if (!$tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PHPDOC)) { - if (!$tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL)) { - throw new ParserException( - $tokens->currentTokenValue(), - $tokens->currentTokenType(), - $tokens->currentTokenOffset(), - Lexer::TOKEN_PHPDOC_EOL, - null, - $tokens->currentTokenLine() - ); - } + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + try { + $type = $this->typeParser->parse($tokens); + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PHPDOC)) { + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL)) { + throw new ParserException( + $tokens->currentTokenValue(), + $tokens->currentTokenType(), + $tokens->currentTokenOffset(), + Lexer::TOKEN_PHPDOC_EOL, + null, + $tokens->currentTokenLine(), + ); } - - return new Ast\PhpDoc\TypeAliasTagValueNode($alias, $type); - } catch (ParserException $e) { - $this->parseOptionalDescription($tokens); - return new Ast\PhpDoc\TypeAliasTagValueNode( - $alias, - $this->enrichWithAttributes($tokens, new Ast\Type\InvalidTypeNode($e), $startLine, $startIndex) - ); } - } - - $type = $this->typeParser->parse($tokens); - return new Ast\PhpDoc\TypeAliasTagValueNode($alias, $type); + return new Ast\PhpDoc\TypeAliasTagValueNode($alias, $type); + } catch (ParserException $e) { + $this->parseOptionalDescription($tokens, false); + return new Ast\PhpDoc\TypeAliasTagValueNode( + $alias, + $this->enrichWithAttributes($tokens, new Ast\Type\InvalidTypeNode($e), $startLine, $startIndex), + ); + } } private function parseTypeAliasImportTagValue(TokenIterator $tokens): Ast\PhpDoc\TypeAliasImportTagValueNode @@ -591,7 +1120,7 @@ private function parseTypeAliasImportTagValue(TokenIterator $tokens): Ast\PhpDoc $tokens, new IdentifierTypeNode($importedFrom), $identifierStartLine, - $identifierStartIndex + $identifierStartIndex, ); $importedAs = null; @@ -612,7 +1141,7 @@ private function parseAssertTagValue(TokenIterator $tokens): Ast\PhpDoc\PhpDocTa $isEquality = $tokens->tryConsumeTokenType(Lexer::TOKEN_EQUAL); $type = $this->typeParser->parse($tokens); $parameter = $this->parseAssertParameter($tokens); - $description = $this->parseOptionalDescription($tokens); + $description = $this->parseOptionalDescription($tokens, false); if (array_key_exists('method', $parameter)) { return new Ast\PhpDoc\AssertTagMethodValueNode($type, $parameter['parameter'], $parameter['method'], $isNegated, $description, $isEquality); @@ -630,15 +1159,13 @@ private function parseAssertParameter(TokenIterator $tokens): array { if ($tokens->isCurrentTokenType(Lexer::TOKEN_THIS_VARIABLE)) { $parameter = '$this'; - $requirePropertyOrMethod = true; $tokens->next(); } else { $parameter = $tokens->currentTokenValue(); - $requirePropertyOrMethod = false; $tokens->consumeTokenType(Lexer::TOKEN_VARIABLE); } - if ($requirePropertyOrMethod || $tokens->isCurrentTokenType(Lexer::TOKEN_ARROW)) { + if ($tokens->isCurrentTokenType(Lexer::TOKEN_ARROW)) { $tokens->consumeTokenType(Lexer::TOKEN_ARROW); $propertyOrMethod = $tokens->currentTokenValue(); @@ -659,7 +1186,7 @@ private function parseAssertParameter(TokenIterator $tokens): array private function parseSelfOutTagValue(TokenIterator $tokens): Ast\PhpDoc\SelfOutTagValueNode { $type = $this->typeParser->parse($tokens); - $description = $this->parseOptionalDescription($tokens); + $description = $this->parseOptionalDescription($tokens, true); return new Ast\PhpDoc\SelfOutTagValueNode($type, $description); } @@ -668,7 +1195,7 @@ private function parseParamOutTagValue(TokenIterator $tokens): Ast\PhpDoc\ParamO { $type = $this->typeParser->parse($tokens); $parameterName = $this->parseRequiredVariableName($tokens); - $description = $this->parseOptionalDescription($tokens); + $description = $this->parseOptionalDescription($tokens, false); return new Ast\PhpDoc\ParamOutTagValueNode($type, $parameterName, $description); } @@ -698,7 +1225,10 @@ private function parseRequiredVariableName(TokenIterator $tokens): string return $parameterName; } - private function parseOptionalDescription(TokenIterator $tokens, bool $limitStartToken = false): string + /** + * @param bool $limitStartToken true should be used when the description immediately follows a parsed type + */ + private function parseOptionalDescription(TokenIterator $tokens, bool $limitStartToken): string { if ($limitStartToken) { foreach (self::DISALLOWED_DESCRIPTION_START_TOKENS as $disallowedStartToken) { @@ -710,8 +1240,7 @@ private function parseOptionalDescription(TokenIterator $tokens, bool $limitStar } if ( - $this->requireWhitespaceBeforeDescription - && !$tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL, Lexer::TOKEN_CLOSE_PHPDOC, Lexer::TOKEN_END) + !$tokens->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL, Lexer::TOKEN_CLOSE_PHPDOC, Lexer::TOKEN_END) && !$tokens->isPrecededByHorizontalWhitespace() ) { $tokens->consumeTokenType(Lexer::TOKEN_HORIZONTAL_WS); // will throw exception diff --git a/src/Parser/StringUnescaper.php b/src/Parser/StringUnescaper.php index 70524055..e8e0a3d6 100644 --- a/src/Parser/StringUnescaper.php +++ b/src/Parser/StringUnescaper.php @@ -2,6 +2,7 @@ namespace PHPStan\PhpDocParser\Parser; +use PHPStan\ShouldNotHappenException; use function chr; use function hexdec; use function octdec; @@ -30,7 +31,7 @@ public static function unescapeString(string $string): string return str_replace( ['\\\\', '\\\''], ['\\', '\''], - substr($string, 1, -1) + substr($string, 1, -1), ); } @@ -56,12 +57,15 @@ static function ($matches) { return chr((int) hexdec(substr($str, 1))); } if ($str[0] === 'u') { + if (!isset($matches[2])) { + throw new ShouldNotHappenException(); + } return self::codePointToUtf8((int) hexdec($matches[2])); } return chr((int) octdec($str)); }, - $str + $str, ); } diff --git a/src/Parser/TokenIterator.php b/src/Parser/TokenIterator.php index 4348ab79..f2be3da4 100644 --- a/src/Parser/TokenIterator.php +++ b/src/Parser/TokenIterator.php @@ -3,24 +3,33 @@ namespace PHPStan\PhpDocParser\Parser; use LogicException; +use PHPStan\PhpDocParser\Ast\Comment; use PHPStan\PhpDocParser\Lexer\Lexer; use function array_pop; use function assert; use function count; use function in_array; use function strlen; +use function substr; class TokenIterator { /** @var list */ - private $tokens; + private array $tokens; - /** @var int */ - private $index; + private int $index; - /** @var int[] */ - private $savePoints = []; + /** @var list */ + private array $comments = []; + + /** @var list}> */ + private array $savePoints = []; + + /** @var list */ + private array $skippedTokenTypes = [Lexer::TOKEN_HORIZONTAL_WS]; + + private ?string $newline = null; /** * @param list $tokens @@ -30,11 +39,7 @@ public function __construct(array $tokens, int $index = 0) $this->tokens = $tokens; $this->index = $index; - if ($this->tokens[$this->index][Lexer::TYPE_OFFSET] !== Lexer::TOKEN_HORIZONTAL_WS) { - return; - } - - $this->index++; + $this->skipIrrelevantTokens(); } @@ -103,6 +108,21 @@ public function currentTokenIndex(): int } + public function endIndexOfLastRelevantToken(): int + { + $endIndex = $this->currentTokenIndex(); + $endIndex--; + while (in_array($this->tokens[$endIndex][Lexer::TYPE_OFFSET], $this->skippedTokenTypes, true)) { + if (!isset($this->tokens[$endIndex - 1])) { + break; + } + $endIndex--; + } + + return $endIndex; + } + + public function isCurrentTokenValue(string $tokenValue): bool { return $this->tokens[$this->index][Lexer::VALUE_OFFSET] === $tokenValue; @@ -130,13 +150,13 @@ public function consumeTokenType(int $tokenType): void $this->throwError($tokenType); } - $this->index++; - - if (($this->tokens[$this->index][Lexer::TYPE_OFFSET] ?? -1) !== Lexer::TOKEN_HORIZONTAL_WS) { - return; + if ($tokenType === Lexer::TOKEN_PHPDOC_EOL) { + if ($this->newline === null) { + $this->detectNewline(); + } } - $this->index++; + $this->next(); } @@ -149,13 +169,7 @@ public function consumeTokenValue(int $tokenType, string $tokenValue): void $this->throwError($tokenType, $tokenValue); } - $this->index++; - - if (($this->tokens[$this->index][Lexer::TYPE_OFFSET] ?? -1) !== Lexer::TOKEN_HORIZONTAL_WS) { - return; - } - - $this->index++; + $this->next(); } @@ -166,15 +180,20 @@ public function tryConsumeTokenValue(string $tokenValue): bool return false; } - $this->index++; - - if ($this->tokens[$this->index][Lexer::TYPE_OFFSET] === Lexer::TOKEN_HORIZONTAL_WS) { - $this->index++; - } + $this->next(); return true; } + /** + * @return list + */ + public function flushComments(): array + { + $res = $this->comments; + $this->comments = []; + return $res; + } /** @phpstan-impure */ public function tryConsumeTokenType(int $tokenType): bool @@ -183,16 +202,67 @@ public function tryConsumeTokenType(int $tokenType): bool return false; } - $this->index++; - - if ($this->tokens[$this->index][Lexer::TYPE_OFFSET] === Lexer::TOKEN_HORIZONTAL_WS) { - $this->index++; + if ($tokenType === Lexer::TOKEN_PHPDOC_EOL) { + if ($this->newline === null) { + $this->detectNewline(); + } } + $this->next(); + return true; } + /** + * @deprecated Use skipNewLineTokensAndConsumeComments instead (when parsing a type) + */ + public function skipNewLineTokens(): void + { + if (!$this->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL)) { + return; + } + + do { + $foundNewLine = $this->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + } while ($foundNewLine === true); + } + + + public function skipNewLineTokensAndConsumeComments(): void + { + if ($this->currentTokenType() === Lexer::TOKEN_COMMENT) { + $this->comments[] = new Comment($this->currentTokenValue(), $this->currentTokenLine(), $this->currentTokenIndex()); + $this->next(); + } + + if (!$this->isCurrentTokenType(Lexer::TOKEN_PHPDOC_EOL)) { + return; + } + + do { + $foundNewLine = $this->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + if ($this->currentTokenType() !== Lexer::TOKEN_COMMENT) { + continue; + } + + $this->comments[] = new Comment($this->currentTokenValue(), $this->currentTokenLine(), $this->currentTokenIndex()); + $this->next(); + } while ($foundNewLine === true); + } + + + private function detectNewline(): void + { + $value = $this->currentTokenValue(); + if (substr($value, 0, 2) === "\r\n") { + $this->newline = "\r\n"; + } elseif (substr($value, 0, 1) === "\n") { + $this->newline = "\n"; + } + } + + public function getSkippedHorizontalWhiteSpaceIfAny(): string { if ($this->index > 0 && $this->tokens[$this->index - 1][Lexer::TYPE_OFFSET] === Lexer::TOKEN_HORIZONTAL_WS) { @@ -217,12 +287,34 @@ public function joinUntil(int ...$tokenType): string public function next(): void { $this->index++; + $this->skipIrrelevantTokens(); + } + - if ($this->tokens[$this->index][Lexer::TYPE_OFFSET] !== Lexer::TOKEN_HORIZONTAL_WS) { + private function skipIrrelevantTokens(): void + { + if (!isset($this->tokens[$this->index])) { return; } - $this->index++; + while (in_array($this->tokens[$this->index][Lexer::TYPE_OFFSET], $this->skippedTokenTypes, true)) { + if (!isset($this->tokens[$this->index + 1])) { + break; + } + $this->index++; + } + } + + + public function addEndOfLineToSkippedTokens(): void + { + $this->skippedTokenTypes = [Lexer::TOKEN_HORIZONTAL_WS, Lexer::TOKEN_PHPDOC_EOL]; + } + + + public function removeEndOfLineFromSkippedTokens(): void + { + $this->skippedTokenTypes = [Lexer::TOKEN_HORIZONTAL_WS]; } /** @phpstan-impure */ @@ -235,7 +327,7 @@ public function forwardToTheEnd(): void public function pushSavePoint(): void { - $this->savePoints[] = $this->index; + $this->savePoints[] = [$this->index, $this->comments]; } @@ -247,9 +339,9 @@ public function dropSavePoint(): void public function rollback(): void { - $index = array_pop($this->savePoints); - assert($index !== null); - $this->index = $index; + $savepoint = array_pop($this->savePoints); + assert($savepoint !== null); + [$this->index, $this->comments] = $savepoint; } @@ -264,7 +356,7 @@ private function throwError(int $expectedTokenType, ?string $expectedTokenValue $this->currentTokenOffset(), $expectedTokenType, $expectedTokenValue, - $this->currentTokenLine() + $this->currentTokenLine(), ); } @@ -319,6 +411,11 @@ public function hasTokenImmediatelyAfter(int $pos, int $expectedTokenType): bool return false; } + public function getDetectedNewline(): ?string + { + return $this->newline; + } + /** * Whether the given position is immediately surrounded by parenthesis. */ diff --git a/src/Parser/TypeParser.php b/src/Parser/TypeParser.php index 4b429809..fc225854 100644 --- a/src/Parser/TypeParser.php +++ b/src/Parser/TypeParser.php @@ -4,40 +4,29 @@ use LogicException; use PHPStan\PhpDocParser\Ast; +use PHPStan\PhpDocParser\Ast\PhpDoc\TemplateTagValueNode; use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\ParserConfig; use function in_array; use function str_replace; +use function strlen; use function strpos; -use function trim; +use function substr_compare; class TypeParser { - /** @var ConstExprParser|null */ - private $constExprParser; + private ParserConfig $config; - /** @var bool */ - private $quoteAwareConstExprString; + private ConstExprParser $constExprParser; - /** @var bool */ - private $useLinesAttributes; - - /** @var bool */ - private $useIndexAttributes; - - /** - * @param array{lines?: bool, indexes?: bool} $usedAttributes - */ public function __construct( - ?ConstExprParser $constExprParser = null, - bool $quoteAwareConstExprString = false, - array $usedAttributes = [] + ParserConfig $config, + ConstExprParser $constExprParser ) { + $this->config = $config; $this->constExprParser = $constExprParser; - $this->quoteAwareConstExprString = $quoteAwareConstExprString; - $this->useLinesAttributes = $usedAttributes['lines'] ?? false; - $this->useIndexAttributes = $usedAttributes['indexes'] ?? false; } /** @phpstan-impure */ @@ -51,17 +40,44 @@ public function parse(TokenIterator $tokens): Ast\Type\TypeNode } else { $type = $this->parseAtomic($tokens); - if ($tokens->isCurrentTokenType(Lexer::TOKEN_UNION)) { - $type = $this->parseUnion($tokens, $type); + $tokens->pushSavePoint(); + $tokens->skipNewLineTokensAndConsumeComments(); + + try { + $enrichedType = $this->enrichTypeOnUnionOrIntersection($tokens, $type); - } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_INTERSECTION)) { - $type = $this->parseIntersection($tokens, $type); + } catch (ParserException $parserException) { + $enrichedType = null; + } + + if ($enrichedType !== null) { + $type = $enrichedType; + $tokens->dropSavePoint(); + + } else { + $tokens->rollback(); + $type = $this->enrichTypeOnUnionOrIntersection($tokens, $type) ?? $type; } } return $this->enrichWithAttributes($tokens, $type, $startLine, $startIndex); } + /** @phpstan-impure */ + private function enrichTypeOnUnionOrIntersection(TokenIterator $tokens, Ast\Type\TypeNode $type): ?Ast\Type\TypeNode + { + if ($tokens->isCurrentTokenType(Lexer::TOKEN_UNION)) { + return $this->parseUnion($tokens, $type); + + } + + if ($tokens->isCurrentTokenType(Lexer::TOKEN_INTERSECTION)) { + return $this->parseIntersection($tokens, $type); + } + + return null; + } + /** * @internal * @template T of Ast\Node @@ -70,23 +86,19 @@ public function parse(TokenIterator $tokens): Ast\Type\TypeNode */ public function enrichWithAttributes(TokenIterator $tokens, Ast\Node $type, int $startLine, int $startIndex): Ast\Node { - $endLine = $tokens->currentTokenLine(); - $endIndex = $tokens->currentTokenIndex(); - - if ($this->useLinesAttributes) { + if ($this->config->useLinesAttributes) { $type->setAttribute(Ast\Attribute::START_LINE, $startLine); - $type->setAttribute(Ast\Attribute::END_LINE, $endLine); + $type->setAttribute(Ast\Attribute::END_LINE, $tokens->currentTokenLine()); } - if ($this->useIndexAttributes) { - $tokensArray = $tokens->getTokens(); - $endIndex--; - if ($tokensArray[$endIndex][Lexer::TYPE_OFFSET] === Lexer::TOKEN_HORIZONTAL_WS) { - $endIndex--; - } + $comments = $tokens->flushComments(); + if ($this->config->useCommentsAttributes) { + $type->setAttribute(Ast\Attribute::COMMENTS, $comments); + } + if ($this->config->useIndexAttributes) { $type->setAttribute(Ast\Attribute::START_INDEX, $startIndex); - $type->setAttribute(Ast\Attribute::END_INDEX, $endIndex); + $type->setAttribute(Ast\Attribute::END_INDEX, $tokens->endIndexOfLastRelevantToken()); } return $type; @@ -110,7 +122,7 @@ private function subParse(TokenIterator $tokens): Ast\Type\TypeNode if ($tokens->isCurrentTokenValue('is')) { $type = $this->parseConditional($tokens, $type); } else { - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); if ($tokens->isCurrentTokenType(Lexer::TOKEN_UNION)) { $type = $this->subParseUnion($tokens, $type); @@ -132,9 +144,9 @@ private function parseAtomic(TokenIterator $tokens): Ast\Type\TypeNode $startIndex = $tokens->currentTokenIndex(); if ($tokens->tryConsumeTokenType(Lexer::TOKEN_OPEN_PARENTHESES)) { - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $type = $this->subParse($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_PARENTHESES); @@ -171,18 +183,28 @@ private function parseAtomic(TokenIterator $tokens): Ast\Type\TypeNode return $type; } - $type = $this->parseGeneric($tokens, $type); + $origType = $type; + $type = $this->tryParseCallable($tokens, $type, true); + if ($type === $origType) { + $type = $this->parseGeneric($tokens, $type); - if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_SQUARE_BRACKET)) { - $type = $this->tryParseArrayOrOffsetAccess($tokens, $type); + if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_SQUARE_BRACKET)) { + $type = $this->tryParseArrayOrOffsetAccess($tokens, $type); + } } } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_PARENTHESES)) { - $type = $this->tryParseCallable($tokens, $type); + $type = $this->tryParseCallable($tokens, $type, false); } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_SQUARE_BRACKET)) { $type = $this->tryParseArrayOrOffsetAccess($tokens, $type); - } elseif (in_array($type->name, ['array', 'list', 'object'], true) && $tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_CURLY_BRACKET) && !$tokens->isPrecededByHorizontalWhitespace()) { + } elseif (in_array($type->name, [ + Ast\Type\ArrayShapeNode::KIND_ARRAY, + Ast\Type\ArrayShapeNode::KIND_LIST, + Ast\Type\ArrayShapeNode::KIND_NON_EMPTY_ARRAY, + Ast\Type\ArrayShapeNode::KIND_NON_EMPTY_LIST, + 'object', + ], true) && $tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_CURLY_BRACKET) && !$tokens->isPrecededByHorizontalWhitespace()) { if ($type->name === 'object') { $type = $this->parseObjectShape($tokens); } else { @@ -192,7 +214,7 @@ private function parseAtomic(TokenIterator $tokens): Ast\Type\TypeNode if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_SQUARE_BRACKET)) { $type = $this->tryParseArrayOrOffsetAccess( $tokens, - $this->enrichWithAttributes($tokens, $type, $startLine, $startIndex) + $this->enrichWithAttributes($tokens, $type, $startLine, $startIndex), ); } } @@ -205,28 +227,44 @@ private function parseAtomic(TokenIterator $tokens): Ast\Type\TypeNode $tokens->dropSavePoint(); // because of ConstFetchNode } - $exception = new ParserException( - $tokens->currentTokenValue(), - $tokens->currentTokenType(), - $tokens->currentTokenOffset(), - Lexer::TOKEN_IDENTIFIER, - null, - $tokens->currentTokenLine() - ); - - if ($this->constExprParser === null) { - throw $exception; - } + $currentTokenValue = $tokens->currentTokenValue(); + $currentTokenType = $tokens->currentTokenType(); + $currentTokenOffset = $tokens->currentTokenOffset(); + $currentTokenLine = $tokens->currentTokenLine(); try { - $constExpr = $this->constExprParser->parse($tokens, true); + $constExpr = $this->constExprParser->parse($tokens); if ($constExpr instanceof Ast\ConstExpr\ConstExprArrayNode) { - throw $exception; + throw new ParserException( + $currentTokenValue, + $currentTokenType, + $currentTokenOffset, + Lexer::TOKEN_IDENTIFIER, + null, + $currentTokenLine, + ); + } + + $type = $this->enrichWithAttributes( + $tokens, + new Ast\Type\ConstTypeNode($constExpr), + $startLine, + $startIndex, + ); + if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_SQUARE_BRACKET)) { + $type = $this->tryParseArrayOrOffsetAccess($tokens, $type); } - return $this->enrichWithAttributes($tokens, new Ast\Type\ConstTypeNode($constExpr), $startLine, $startIndex); + return $type; } catch (LogicException $e) { - throw $exception; + throw new ParserException( + $currentTokenValue, + $currentTokenType, + $currentTokenOffset, + Lexer::TOKEN_IDENTIFIER, + null, + $currentTokenLine, + ); } } @@ -238,6 +276,14 @@ private function parseUnion(TokenIterator $tokens, Ast\Type\TypeNode $type): Ast while ($tokens->tryConsumeTokenType(Lexer::TOKEN_UNION)) { $types[] = $this->parseAtomic($tokens); + $tokens->pushSavePoint(); + $tokens->skipNewLineTokensAndConsumeComments(); + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_UNION)) { + $tokens->rollback(); + break; + } + + $tokens->dropSavePoint(); } return new Ast\Type\UnionTypeNode($types); @@ -250,9 +296,9 @@ private function subParseUnion(TokenIterator $tokens, Ast\Type\TypeNode $type): $types = [$type]; while ($tokens->tryConsumeTokenType(Lexer::TOKEN_UNION)) { - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $types[] = $this->parseAtomic($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); } return new Ast\Type\UnionTypeNode($types); @@ -266,6 +312,14 @@ private function parseIntersection(TokenIterator $tokens, Ast\Type\TypeNode $typ while ($tokens->tryConsumeTokenType(Lexer::TOKEN_INTERSECTION)) { $types[] = $this->parseAtomic($tokens); + $tokens->pushSavePoint(); + $tokens->skipNewLineTokensAndConsumeComments(); + if (!$tokens->isCurrentTokenType(Lexer::TOKEN_INTERSECTION)) { + $tokens->rollback(); + break; + } + + $tokens->dropSavePoint(); } return new Ast\Type\IntersectionTypeNode($types); @@ -278,9 +332,9 @@ private function subParseIntersection(TokenIterator $tokens, Ast\Type\TypeNode $ $types = [$type]; while ($tokens->tryConsumeTokenType(Lexer::TOKEN_INTERSECTION)) { - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $types[] = $this->parseAtomic($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); } return new Ast\Type\IntersectionTypeNode($types); @@ -300,15 +354,15 @@ private function parseConditional(TokenIterator $tokens, Ast\Type\TypeNode $subj $targetType = $this->parse($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $tokens->consumeTokenType(Lexer::TOKEN_NULLABLE); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $ifType = $this->parse($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $tokens->consumeTokenType(Lexer::TOKEN_COLON); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $elseType = $this->subParse($tokens); @@ -329,15 +383,15 @@ private function parseConditionalForParameter(TokenIterator $tokens, string $par $targetType = $this->parse($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $tokens->consumeTokenType(Lexer::TOKEN_NULLABLE); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $ifType = $this->parse($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $tokens->consumeTokenType(Lexer::TOKEN_COLON); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $elseType = $this->subParse($tokens); @@ -372,10 +426,16 @@ public function isHtml(TokenIterator $tokens): bool return false; } + $endTag = ''; + $endTagSearchOffset = - strlen($endTag); + while (!$tokens->isCurrentTokenType(Lexer::TOKEN_END)) { if ( - $tokens->tryConsumeTokenType(Lexer::TOKEN_OPEN_ANGLE_BRACKET) - && strpos($tokens->currentTokenValue(), '/' . $htmlTagName . '>') !== false + ( + $tokens->tryConsumeTokenType(Lexer::TOKEN_OPEN_ANGLE_BRACKET) + && strpos($tokens->currentTokenValue(), '/' . $htmlTagName . '>') !== false + ) + || substr_compare($tokens->currentTokenValue(), $endTag, $endTagSearchOffset) === 0 ) { return true; } @@ -390,42 +450,37 @@ public function isHtml(TokenIterator $tokens): bool public function parseGeneric(TokenIterator $tokens, Ast\Type\IdentifierTypeNode $baseType): Ast\Type\GenericTypeNode { $tokens->consumeTokenType(Lexer::TOKEN_OPEN_ANGLE_BRACKET); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); + $startLine = $baseType->getAttribute(Ast\Attribute::START_LINE); + $startIndex = $baseType->getAttribute(Ast\Attribute::START_INDEX); $genericTypes = []; $variances = []; - [$genericTypes[], $variances[]] = $this->parseGenericTypeArgument($tokens); + $isFirst = true; + while ( + $isFirst + || $tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA) + ) { + $tokens->skipNewLineTokensAndConsumeComments(); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); - - while ($tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)) { - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); - if ($tokens->tryConsumeTokenType(Lexer::TOKEN_CLOSE_ANGLE_BRACKET)) { - // trailing comma case - $type = new Ast\Type\GenericTypeNode($baseType, $genericTypes, $variances); - $startLine = $baseType->getAttribute(Ast\Attribute::START_LINE); - $startIndex = $baseType->getAttribute(Ast\Attribute::START_INDEX); - if ($startLine !== null && $startIndex !== null) { - $type = $this->enrichWithAttributes($tokens, $type, $startLine, $startIndex); - } - - return $type; + // trailing comma case + if (!$isFirst && $tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_ANGLE_BRACKET)) { + break; } + $isFirst = false; + [$genericTypes[], $variances[]] = $this->parseGenericTypeArgument($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); } - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); - $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_ANGLE_BRACKET); - $type = new Ast\Type\GenericTypeNode($baseType, $genericTypes, $variances); - $startLine = $baseType->getAttribute(Ast\Attribute::START_LINE); - $startIndex = $baseType->getAttribute(Ast\Attribute::START_INDEX); if ($startLine !== null && $startIndex !== null) { $type = $this->enrichWithAttributes($tokens, $type, $startLine, $startIndex); } + $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_ANGLE_BRACKET); + return $type; } @@ -457,24 +512,69 @@ public function parseGenericTypeArgument(TokenIterator $tokens): array return [$type, $variance]; } + /** + * @throws ParserException + * @param ?callable(TokenIterator): string $parseDescription + */ + public function parseTemplateTagValue( + TokenIterator $tokens, + ?callable $parseDescription = null + ): TemplateTagValueNode + { + $name = $tokens->currentTokenValue(); + $tokens->consumeTokenType(Lexer::TOKEN_IDENTIFIER); + + $upperBound = $lowerBound = null; + + if ($tokens->tryConsumeTokenValue('of') || $tokens->tryConsumeTokenValue('as')) { + $upperBound = $this->parse($tokens); + } + + if ($tokens->tryConsumeTokenValue('super')) { + $lowerBound = $this->parse($tokens); + } + + if ($tokens->tryConsumeTokenValue('=')) { + $default = $this->parse($tokens); + } else { + $default = null; + } + + if ($parseDescription !== null) { + $description = $parseDescription($tokens); + } else { + $description = ''; + } + + if ($name === '') { + throw new LogicException('Template tag name cannot be empty.'); + } + + return new Ast\PhpDoc\TemplateTagValueNode($name, $upperBound, $description, $default, $lowerBound); + } + /** @phpstan-impure */ - private function parseCallable(TokenIterator $tokens, Ast\Type\IdentifierTypeNode $identifier): Ast\Type\TypeNode + private function parseCallable(TokenIterator $tokens, Ast\Type\IdentifierTypeNode $identifier, bool $hasTemplate): Ast\Type\TypeNode { + $templates = $hasTemplate + ? $this->parseCallableTemplates($tokens) + : []; + $tokens->consumeTokenType(Lexer::TOKEN_OPEN_PARENTHESES); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $parameters = []; if (!$tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PARENTHESES)) { $parameters[] = $this->parseCallableParameter($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); while ($tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)) { - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); if ($tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_PARENTHESES)) { break; } $parameters[] = $this->parseCallableParameter($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); } } @@ -485,7 +585,52 @@ private function parseCallable(TokenIterator $tokens, Ast\Type\IdentifierTypeNod $startIndex = $tokens->currentTokenIndex(); $returnType = $this->enrichWithAttributes($tokens, $this->parseCallableReturnType($tokens), $startLine, $startIndex); - return new Ast\Type\CallableTypeNode($identifier, $parameters, $returnType); + return new Ast\Type\CallableTypeNode($identifier, $parameters, $returnType, $templates); + } + + + /** + * @return Ast\PhpDoc\TemplateTagValueNode[] + * + * @phpstan-impure + */ + private function parseCallableTemplates(TokenIterator $tokens): array + { + $tokens->consumeTokenType(Lexer::TOKEN_OPEN_ANGLE_BRACKET); + + $templates = []; + + $isFirst = true; + while ($isFirst || $tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)) { + $tokens->skipNewLineTokensAndConsumeComments(); + + // trailing comma case + if (!$isFirst && $tokens->isCurrentTokenType(Lexer::TOKEN_CLOSE_ANGLE_BRACKET)) { + break; + } + $isFirst = false; + + $templates[] = $this->parseCallableTemplateArgument($tokens); + $tokens->skipNewLineTokensAndConsumeComments(); + } + + $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_ANGLE_BRACKET); + + return $templates; + } + + + private function parseCallableTemplateArgument(TokenIterator $tokens): Ast\PhpDoc\TemplateTagValueNode + { + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + + return $this->enrichWithAttributes( + $tokens, + $this->parseTemplateTagValue($tokens), + $startLine, + $startIndex, + ); } @@ -511,7 +656,7 @@ private function parseCallableParameter(TokenIterator $tokens): Ast\Type\Callabl $tokens, new Ast\Type\CallableTypeParameterNode($type, $isReference, $isVariadic, $parameterName, $isOptional), $startLine, - $startIndex + $startIndex, ); } @@ -525,7 +670,7 @@ private function parseCallableReturnType(TokenIterator $tokens): Ast\Type\TypeNo return $this->parseNullable($tokens); } elseif ($tokens->tryConsumeTokenType(Lexer::TOKEN_OPEN_PARENTHESES)) { - $type = $this->parse($tokens); + $type = $this->subParse($tokens); $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_PARENTHESES); if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_SQUARE_BRACKET)) { $type = $this->tryParseArrayOrOffsetAccess($tokens, $type); @@ -539,7 +684,7 @@ private function parseCallableReturnType(TokenIterator $tokens): Ast\Type\TypeNo $tokens, $type, $startLine, - $startIndex + $startIndex, )); } @@ -558,15 +703,15 @@ private function parseCallableReturnType(TokenIterator $tokens): Ast\Type\TypeNo $tokens, $type, $startLine, - $startIndex - ) + $startIndex, + ), ); if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_SQUARE_BRACKET)) { $type = $this->tryParseArrayOrOffsetAccess($tokens, $this->enrichWithAttributes( $tokens, $type, $startLine, - $startIndex + $startIndex, )); } @@ -575,10 +720,16 @@ private function parseCallableReturnType(TokenIterator $tokens): Ast\Type\TypeNo $tokens, $type, $startLine, - $startIndex + $startIndex, )); - } elseif (in_array($type->name, ['array', 'list', 'object'], true) && $tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_CURLY_BRACKET) && !$tokens->isPrecededByHorizontalWhitespace()) { + } elseif (in_array($type->name, [ + Ast\Type\ArrayShapeNode::KIND_ARRAY, + Ast\Type\ArrayShapeNode::KIND_LIST, + Ast\Type\ArrayShapeNode::KIND_NON_EMPTY_ARRAY, + Ast\Type\ArrayShapeNode::KIND_NON_EMPTY_LIST, + 'object', + ], true) && $tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_CURLY_BRACKET) && !$tokens->isPrecededByHorizontalWhitespace()) { if ($type->name === 'object') { $type = $this->parseObjectShape($tokens); } else { @@ -586,7 +737,7 @@ private function parseCallableReturnType(TokenIterator $tokens): Ast\Type\TypeNo $tokens, $type, $startLine, - $startIndex + $startIndex, ), $type->name); } @@ -595,7 +746,7 @@ private function parseCallableReturnType(TokenIterator $tokens): Ast\Type\TypeNo $tokens, $type, $startLine, - $startIndex + $startIndex, )); } } @@ -609,48 +760,54 @@ private function parseCallableReturnType(TokenIterator $tokens): Ast\Type\TypeNo } } - $exception = new ParserException( - $tokens->currentTokenValue(), - $tokens->currentTokenType(), - $tokens->currentTokenOffset(), - Lexer::TOKEN_IDENTIFIER, - null, - $tokens->currentTokenLine() - ); - - if ($this->constExprParser === null) { - throw $exception; - } + $currentTokenValue = $tokens->currentTokenValue(); + $currentTokenType = $tokens->currentTokenType(); + $currentTokenOffset = $tokens->currentTokenOffset(); + $currentTokenLine = $tokens->currentTokenLine(); try { - $constExpr = $this->constExprParser->parse($tokens, true); + $constExpr = $this->constExprParser->parse($tokens); if ($constExpr instanceof Ast\ConstExpr\ConstExprArrayNode) { - throw $exception; + throw new ParserException( + $currentTokenValue, + $currentTokenType, + $currentTokenOffset, + Lexer::TOKEN_IDENTIFIER, + null, + $currentTokenLine, + ); } - $type = new Ast\Type\ConstTypeNode($constExpr); + $type = $this->enrichWithAttributes( + $tokens, + new Ast\Type\ConstTypeNode($constExpr), + $startLine, + $startIndex, + ); if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_SQUARE_BRACKET)) { - $type = $this->tryParseArrayOrOffsetAccess($tokens, $this->enrichWithAttributes( - $tokens, - $type, - $startLine, - $startIndex - )); + $type = $this->tryParseArrayOrOffsetAccess($tokens, $type); } return $type; } catch (LogicException $e) { - throw $exception; + throw new ParserException( + $currentTokenValue, + $currentTokenType, + $currentTokenOffset, + Lexer::TOKEN_IDENTIFIER, + null, + $currentTokenLine, + ); } } /** @phpstan-impure */ - private function tryParseCallable(TokenIterator $tokens, Ast\Type\IdentifierTypeNode $identifier): Ast\Type\TypeNode + private function tryParseCallable(TokenIterator $tokens, Ast\Type\IdentifierTypeNode $identifier, bool $hasTemplate): Ast\Type\TypeNode { try { $tokens->pushSavePoint(); - $type = $this->parseCallable($tokens, $identifier); + $type = $this->parseCallable($tokens, $identifier, $hasTemplate); $tokens->dropSavePoint(); } catch (ParserException $e) { @@ -685,7 +842,7 @@ private function tryParseArrayOrOffsetAccess(TokenIterator $tokens, Ast\Type\Typ $tokens, $type, $startLine, - $startIndex + $startIndex, ); } } else { @@ -698,7 +855,7 @@ private function tryParseArrayOrOffsetAccess(TokenIterator $tokens, Ast\Type\Typ $tokens, $type, $startLine, - $startIndex + $startIndex, ); } } @@ -722,29 +879,55 @@ private function parseArrayShape(TokenIterator $tokens, Ast\Type\TypeNode $type, $items = []; $sealed = true; + $unsealedType = null; + + $done = false; do { - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); if ($tokens->tryConsumeTokenType(Lexer::TOKEN_CLOSE_CURLY_BRACKET)) { - return new Ast\Type\ArrayShapeNode($items, true, $kind); + return Ast\Type\ArrayShapeNode::createSealed($items, $kind); } if ($tokens->tryConsumeTokenType(Lexer::TOKEN_VARIADIC)) { $sealed = false; + + $tokens->skipNewLineTokensAndConsumeComments(); + if ($tokens->isCurrentTokenType(Lexer::TOKEN_OPEN_ANGLE_BRACKET)) { + if ($kind === Ast\Type\ArrayShapeNode::KIND_ARRAY) { + $unsealedType = $this->parseArrayShapeUnsealedType($tokens); + } else { + $unsealedType = $this->parseListShapeUnsealedType($tokens); + } + $tokens->skipNewLineTokensAndConsumeComments(); + } + $tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA); break; } $items[] = $this->parseArrayShapeItem($tokens); + $tokens->skipNewLineTokensAndConsumeComments(); + if (!$tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)) { + $done = true; + } + if ($tokens->currentTokenType() !== Lexer::TOKEN_COMMENT) { + continue; + } - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); - } while ($tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)); + $tokens->next(); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + } while (!$done); + + $tokens->skipNewLineTokensAndConsumeComments(); $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_CURLY_BRACKET); - return new Ast\Type\ArrayShapeNode($items, $sealed, $kind); + if ($sealed) { + return Ast\Type\ArrayShapeNode::createSealed($items, $kind); + } + + return Ast\Type\ArrayShapeNode::createUnsealed($items, $unsealedType, $kind); } @@ -753,19 +936,24 @@ private function parseArrayShapeItem(TokenIterator $tokens): Ast\Type\ArrayShape { $startLine = $tokens->currentTokenLine(); $startIndex = $tokens->currentTokenIndex(); + + // parse any comments above the item + $tokens->skipNewLineTokensAndConsumeComments(); + try { $tokens->pushSavePoint(); $key = $this->parseArrayShapeKey($tokens); $optional = $tokens->tryConsumeTokenType(Lexer::TOKEN_NULLABLE); $tokens->consumeTokenType(Lexer::TOKEN_COLON); $value = $this->parse($tokens); + $tokens->dropSavePoint(); return $this->enrichWithAttributes( $tokens, new Ast\Type\ArrayShapeItemNode($key, $optional, $value), $startLine, - $startIndex + $startIndex, ); } catch (ParserException $e) { $tokens->rollback(); @@ -775,7 +963,7 @@ private function parseArrayShapeItem(TokenIterator $tokens): Ast\Type\ArrayShape $tokens, new Ast\Type\ArrayShapeItemNode(null, false, $value), $startLine, - $startIndex + $startIndex, ); } } @@ -794,19 +982,11 @@ private function parseArrayShapeKey(TokenIterator $tokens) $tokens->next(); } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_SINGLE_QUOTED_STRING)) { - if ($this->quoteAwareConstExprString) { - $key = new Ast\ConstExpr\QuoteAwareConstExprStringNode(StringUnescaper::unescapeString($tokens->currentTokenValue()), Ast\ConstExpr\QuoteAwareConstExprStringNode::SINGLE_QUOTED); - } else { - $key = new Ast\ConstExpr\ConstExprStringNode(trim($tokens->currentTokenValue(), "'")); - } + $key = new Ast\ConstExpr\ConstExprStringNode(StringUnescaper::unescapeString($tokens->currentTokenValue()), Ast\ConstExpr\ConstExprStringNode::SINGLE_QUOTED); $tokens->next(); } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_DOUBLE_QUOTED_STRING)) { - if ($this->quoteAwareConstExprString) { - $key = new Ast\ConstExpr\QuoteAwareConstExprStringNode(StringUnescaper::unescapeString($tokens->currentTokenValue()), Ast\ConstExpr\QuoteAwareConstExprStringNode::DOUBLE_QUOTED); - } else { - $key = new Ast\ConstExpr\ConstExprStringNode(trim($tokens->currentTokenValue(), '"')); - } + $key = new Ast\ConstExpr\ConstExprStringNode(StringUnescaper::unescapeString($tokens->currentTokenValue()), Ast\ConstExpr\ConstExprStringNode::DOUBLE_QUOTED); $tokens->next(); @@ -819,7 +999,64 @@ private function parseArrayShapeKey(TokenIterator $tokens) $tokens, $key, $startLine, - $startIndex + $startIndex, + ); + } + + /** + * @phpstan-impure + */ + private function parseArrayShapeUnsealedType(TokenIterator $tokens): Ast\Type\ArrayShapeUnsealedTypeNode + { + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + + $tokens->consumeTokenType(Lexer::TOKEN_OPEN_ANGLE_BRACKET); + $tokens->skipNewLineTokensAndConsumeComments(); + + $valueType = $this->parse($tokens); + $tokens->skipNewLineTokensAndConsumeComments(); + + $keyType = null; + if ($tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)) { + $tokens->skipNewLineTokensAndConsumeComments(); + + $keyType = $valueType; + $valueType = $this->parse($tokens); + $tokens->skipNewLineTokensAndConsumeComments(); + } + + $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_ANGLE_BRACKET); + + return $this->enrichWithAttributes( + $tokens, + new Ast\Type\ArrayShapeUnsealedTypeNode($valueType, $keyType), + $startLine, + $startIndex, + ); + } + + /** + * @phpstan-impure + */ + private function parseListShapeUnsealedType(TokenIterator $tokens): Ast\Type\ArrayShapeUnsealedTypeNode + { + $startLine = $tokens->currentTokenLine(); + $startIndex = $tokens->currentTokenIndex(); + + $tokens->consumeTokenType(Lexer::TOKEN_OPEN_ANGLE_BRACKET); + $tokens->skipNewLineTokensAndConsumeComments(); + + $valueType = $this->parse($tokens); + $tokens->skipNewLineTokensAndConsumeComments(); + + $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_ANGLE_BRACKET); + + return $this->enrichWithAttributes( + $tokens, + new Ast\Type\ArrayShapeUnsealedTypeNode($valueType, null), + $startLine, + $startIndex, ); } @@ -833,7 +1070,7 @@ private function parseObjectShape(TokenIterator $tokens): Ast\Type\ObjectShapeNo $items = []; do { - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); if ($tokens->tryConsumeTokenType(Lexer::TOKEN_CLOSE_CURLY_BRACKET)) { return new Ast\Type\ObjectShapeNode($items); @@ -841,10 +1078,10 @@ private function parseObjectShape(TokenIterator $tokens): Ast\Type\ObjectShapeNo $items[] = $this->parseObjectShapeItem($tokens); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); } while ($tokens->tryConsumeTokenType(Lexer::TOKEN_COMMA)); - $tokens->tryConsumeTokenType(Lexer::TOKEN_PHPDOC_EOL); + $tokens->skipNewLineTokensAndConsumeComments(); $tokens->consumeTokenType(Lexer::TOKEN_CLOSE_CURLY_BRACKET); return new Ast\Type\ObjectShapeNode($items); @@ -856,12 +1093,19 @@ private function parseObjectShapeItem(TokenIterator $tokens): Ast\Type\ObjectSha $startLine = $tokens->currentTokenLine(); $startIndex = $tokens->currentTokenIndex(); + $tokens->skipNewLineTokensAndConsumeComments(); + $key = $this->parseObjectShapeKey($tokens); $optional = $tokens->tryConsumeTokenType(Lexer::TOKEN_NULLABLE); $tokens->consumeTokenType(Lexer::TOKEN_COLON); $value = $this->parse($tokens); - return $this->enrichWithAttributes($tokens, new Ast\Type\ObjectShapeItemNode($key, $optional, $value), $startLine, $startIndex); + return $this->enrichWithAttributes( + $tokens, + new Ast\Type\ObjectShapeItemNode($key, $optional, $value), + $startLine, + $startIndex, + ); } /** @@ -874,19 +1118,11 @@ private function parseObjectShapeKey(TokenIterator $tokens) $startIndex = $tokens->currentTokenIndex(); if ($tokens->isCurrentTokenType(Lexer::TOKEN_SINGLE_QUOTED_STRING)) { - if ($this->quoteAwareConstExprString) { - $key = new Ast\ConstExpr\QuoteAwareConstExprStringNode(StringUnescaper::unescapeString($tokens->currentTokenValue()), Ast\ConstExpr\QuoteAwareConstExprStringNode::SINGLE_QUOTED); - } else { - $key = new Ast\ConstExpr\ConstExprStringNode(trim($tokens->currentTokenValue(), "'")); - } + $key = new Ast\ConstExpr\ConstExprStringNode(StringUnescaper::unescapeString($tokens->currentTokenValue()), Ast\ConstExpr\ConstExprStringNode::SINGLE_QUOTED); $tokens->next(); } elseif ($tokens->isCurrentTokenType(Lexer::TOKEN_DOUBLE_QUOTED_STRING)) { - if ($this->quoteAwareConstExprString) { - $key = new Ast\ConstExpr\QuoteAwareConstExprStringNode(StringUnescaper::unescapeString($tokens->currentTokenValue()), Ast\ConstExpr\QuoteAwareConstExprStringNode::DOUBLE_QUOTED); - } else { - $key = new Ast\ConstExpr\ConstExprStringNode(trim($tokens->currentTokenValue(), '"')); - } + $key = new Ast\ConstExpr\ConstExprStringNode(StringUnescaper::unescapeString($tokens->currentTokenValue()), Ast\ConstExpr\ConstExprStringNode::DOUBLE_QUOTED); $tokens->next(); } else { diff --git a/src/ParserConfig.php b/src/ParserConfig.php new file mode 100644 index 00000000..0c4377ad --- /dev/null +++ b/src/ParserConfig.php @@ -0,0 +1,24 @@ +useLinesAttributes = $usedAttributes['lines'] ?? false; + $this->useIndexAttributes = $usedAttributes['indexes'] ?? false; + $this->useCommentsAttributes = $usedAttributes['comments'] ?? false; + } + +} diff --git a/src/Printer/Differ.php b/src/Printer/Differ.php index ab10be59..c60fd4ad 100644 --- a/src/Printer/Differ.php +++ b/src/Printer/Differ.php @@ -180,7 +180,7 @@ private function coalesceReplacements(array $diff): array $newDiff[] = new DiffElem( DiffElem::TYPE_REPLACE, $diff[$i + $n]->old, - $diff[$j + $n]->new + $diff[$j + $n]->new, ); } } else { diff --git a/src/Printer/Printer.php b/src/Printer/Printer.php index b65d83aa..af920f8d 100644 --- a/src/Printer/Printer.php +++ b/src/Printer/Printer.php @@ -4,17 +4,26 @@ use LogicException; use PHPStan\PhpDocParser\Ast\Attribute; +use PHPStan\PhpDocParser\Ast\Comment; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprArrayNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprNode; use PHPStan\PhpDocParser\Ast\Node; use PHPStan\PhpDocParser\Ast\PhpDoc\AssertTagMethodValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\AssertTagPropertyValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\AssertTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineAnnotation; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArgument; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArray; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArrayItem; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ExtendsTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ImplementsTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\MethodTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\MethodTagValueParameterNode; use PHPStan\PhpDocParser\Ast\PhpDoc\MixinTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\ParamClosureThisTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\ParamImmediatelyInvokedCallableTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\ParamLaterInvokedCallableTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ParamOutTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ParamTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocChildNode; @@ -23,6 +32,9 @@ use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocTextNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PropertyTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\PureUnlessCallableIsImpureTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\RequireExtendsTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\RequireImplementsTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ReturnTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\SelfOutTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\TemplateTagValueNode; @@ -33,6 +45,7 @@ use PHPStan\PhpDocParser\Ast\PhpDoc\VarTagValueNode; use PHPStan\PhpDocParser\Ast\Type\ArrayShapeItemNode; use PHPStan\PhpDocParser\Ast\Type\ArrayShapeNode; +use PHPStan\PhpDocParser\Ast\Type\ArrayShapeUnsealedTypeNode; use PHPStan\PhpDocParser\Ast\Type\ArrayTypeNode; use PHPStan\PhpDocParser\Ast\Type\CallableTypeNode; use PHPStan\PhpDocParser\Ast\Type\CallableTypeParameterNode; @@ -54,6 +67,7 @@ use PHPStan\PhpDocParser\Parser\TokenIterator; use function array_keys; use function array_map; +use function assert; use function count; use function get_class; use function get_object_vars; @@ -62,6 +76,7 @@ use function is_array; use function preg_match_all; use function sprintf; +use function str_replace; use function strlen; use function strpos; use function trim; @@ -77,7 +92,7 @@ final class Printer { /** @var Differ */ - private $differ; + private Differ $differ; /** * Map From "{$class}->{$subNode}" to string that should be inserted @@ -85,16 +100,19 @@ final class Printer * * @var array */ - private $listInsertionMap = [ + private array $listInsertionMap = [ PhpDocNode::class . '->children' => "\n * ", UnionTypeNode::class . '->types' => '|', IntersectionTypeNode::class . '->types' => '&', ArrayShapeNode::class . '->items' => ', ', ObjectShapeNode::class . '->items' => ', ', CallableTypeNode::class . '->parameters' => ', ', + CallableTypeNode::class . '->templateTypes' => ', ', GenericTypeNode::class . '->genericTypes' => ', ', ConstExprArrayNode::class . '->items' => ', ', MethodTagValueNode::class . '->parameters' => ', ', + DoctrineArray::class . '->items' => ', ', + DoctrineAnnotation::class . '->arguments' => ', ', ]; /** @@ -102,14 +120,16 @@ final class Printer * * @var array */ - private $emptyListInsertionMap = [ + private array $emptyListInsertionMap = [ CallableTypeNode::class . '->parameters' => ['(', '', ''], ArrayShapeNode::class . '->items' => ['{', '', ''], ObjectShapeNode::class . '->items' => ['{', '', ''], + DoctrineArray::class . '->items' => ['{', '', ''], + DoctrineAnnotation::class . '->arguments' => ['(', '', ''], ]; /** @var array>> */ - private $parenthesesMap = [ + private array $parenthesesMap = [ CallableTypeNode::class . '->returnType' => [ CallableTypeNode::class, UnionTypeNode::class, @@ -126,13 +146,12 @@ final class Printer CallableTypeNode::class, UnionTypeNode::class, IntersectionTypeNode::class, - ConstTypeNode::class, NullableTypeNode::class, ], ]; /** @var array>> */ - private $parenthesesListMap = [ + private array $parenthesesListMap = [ IntersectionTypeNode::class . '->types' => [ IntersectionTypeNode::class, UnionTypeNode::class, @@ -162,7 +181,7 @@ public function printFormatPreserving(PhpDocNode $node, PhpDocNode $originalNode $originalTokens, $tokenIndex, PhpDocNode::class, - 'children' + 'children', ); if ($result !== null) { return $result . $originalTokens->getContentBetween($tokenIndex, $originalTokens->getTokenCount()); @@ -179,13 +198,17 @@ function (PhpDocChildNode $child): string { $s = $this->print($child); return $s === '' ? '' : ' ' . $s; }, - $node->children + $node->children, )) . "\n */"; } if ($node instanceof PhpDocTextNode) { return $node->text; } if ($node instanceof PhpDocTagNode) { + if ($node->value instanceof DoctrineTagValueNode) { + return $this->print($node->value); + } + return trim(sprintf('%s %s', $node->name, $this->print($node->value))); } if ($node instanceof PhpDocTagValueNode) { @@ -211,6 +234,48 @@ function (PhpDocChildNode $child): string { $isOptional = $node->isOptional ? '=' : ''; return trim("{$type}{$isReference}{$isVariadic}{$node->parameterName}") . $isOptional; } + if ($node instanceof ArrayShapeUnsealedTypeNode) { + if ($node->keyType !== null) { + return sprintf('<%s, %s>', $this->printType($node->keyType), $this->printType($node->valueType)); + } + return sprintf('<%s>', $this->printType($node->valueType)); + } + if ($node instanceof DoctrineAnnotation) { + return (string) $node; + } + if ($node instanceof DoctrineArgument) { + return (string) $node; + } + if ($node instanceof DoctrineArray) { + return (string) $node; + } + if ($node instanceof DoctrineArrayItem) { + return (string) $node; + } + if ($node instanceof ArrayShapeItemNode) { + if ($node->keyName !== null) { + return sprintf( + '%s%s: %s', + $this->print($node->keyName), + $node->optional ? '?' : '', + $this->printType($node->valueType), + ); + } + + return $this->printType($node->valueType); + } + if ($node instanceof ObjectShapeItemNode) { + if ($node->keyName !== null) { + return sprintf( + '%s%s: %s', + $this->print($node->keyName), + $node->optional ? '?' : '', + $this->printType($node->valueType), + ); + } + + return $this->printType($node->valueType); + } throw new LogicException(sprintf('Unknown node type %s', get_class($node))); } @@ -245,19 +310,23 @@ private function printTagValue(PhpDocTagValueNode $node): string if ($node instanceof MethodTagValueNode) { $static = $node->isStatic ? 'static ' : ''; $returnType = $node->returnType !== null ? $this->printType($node->returnType) . ' ' : ''; - $parameters = implode(', ', array_map(function (MethodTagValueParameterNode $parameter): string { - return $this->print($parameter); - }, $node->parameters)); + $parameters = implode(', ', array_map(fn (MethodTagValueParameterNode $parameter): string => $this->print($parameter), $node->parameters)); $description = $node->description !== '' ? " {$node->description}" : ''; - $templateTypes = count($node->templateTypes) > 0 ? '<' . implode(', ', array_map(function (TemplateTagValueNode $templateTag): string { - return $this->print($templateTag); - }, $node->templateTypes)) . '>' : ''; + $templateTypes = count($node->templateTypes) > 0 ? '<' . implode(', ', array_map(fn (TemplateTagValueNode $templateTag): string => $this->print($templateTag), $node->templateTypes)) . '>' : ''; return "{$static}{$returnType}{$node->methodName}{$templateTypes}({$parameters}){$description}"; } if ($node instanceof MixinTagValueNode) { $type = $this->printType($node->type); return trim("{$type} {$node->description}"); } + if ($node instanceof RequireExtendsTagValueNode) { + $type = $this->printType($node->type); + return trim("{$type} {$node->description}"); + } + if ($node instanceof RequireImplementsTagValueNode) { + $type = $this->printType($node->type); + return trim("{$type} {$node->description}"); + } if ($node instanceof ParamOutTagValueNode) { $type = $this->printType($node->type); return trim("{$type} {$node->parameterName} {$node->description}"); @@ -268,6 +337,18 @@ private function printTagValue(PhpDocTagValueNode $node): string $type = $this->printType($node->type); return trim("{$type} {$reference}{$variadic}{$node->parameterName} {$node->description}"); } + if ($node instanceof ParamImmediatelyInvokedCallableTagValueNode) { + return trim("{$node->parameterName} {$node->description}"); + } + if ($node instanceof ParamLaterInvokedCallableTagValueNode) { + return trim("{$node->parameterName} {$node->description}"); + } + if ($node instanceof ParamClosureThisTagValueNode) { + return trim("{$node->type} {$node->parameterName} {$node->description}"); + } + if ($node instanceof PureUnlessCallableIsImpureTagValueNode) { + return trim("{$node->parameterName} {$node->description}"); + } if ($node instanceof PropertyTagValueNode) { $type = $this->printType($node->type); return trim("{$type} {$node->propertyName} {$node->description}"); @@ -281,9 +362,10 @@ private function printTagValue(PhpDocTagValueNode $node): string return trim($type . ' ' . $node->description); } if ($node instanceof TemplateTagValueNode) { - $bound = $node->bound !== null ? ' of ' . $this->printType($node->bound) : ''; + $upperBound = $node->bound !== null ? ' of ' . $this->printType($node->bound) : ''; + $lowerBound = $node->lowerBound !== null ? ' super ' . $this->printType($node->lowerBound) : ''; $default = $node->default !== null ? ' = ' . $this->printType($node->default) : ''; - return trim("{$node->name}{$bound}{$default} {$node->description}"); + return trim("{$node->name}{$upperBound}{$lowerBound}{$default} {$node->description}"); } if ($node instanceof ThrowsTagValueNode) { $type = $this->printType($node->type); @@ -292,7 +374,7 @@ private function printTagValue(PhpDocTagValueNode $node): string if ($node instanceof TypeAliasImportTagValueNode) { return trim( "{$node->importedAlias} from " . $this->printType($node->importedFrom) - . ($node->importedAs !== null ? " as {$node->importedAs}" : '') + . ($node->importedAs !== null ? " as {$node->importedAs}" : ''), ); } if ($node instanceof TypeAliasTagValueNode) { @@ -314,28 +396,14 @@ private function printTagValue(PhpDocTagValueNode $node): string private function printType(TypeNode $node): string { if ($node instanceof ArrayShapeNode) { - $items = array_map(function (ArrayShapeItemNode $item): string { - return $this->printType($item); - }, $node->items); + $items = array_map(fn (ArrayShapeItemNode $item): string => $this->print($item), $node->items); if (! $node->sealed) { - $items[] = '...'; + $items[] = '...' . ($node->unsealedType === null ? '' : $this->print($node->unsealedType)); } return $node->kind . '{' . implode(', ', $items) . '}'; } - if ($node instanceof ArrayShapeItemNode) { - if ($node->keyName !== null) { - return sprintf( - '%s%s: %s', - $this->print($node->keyName), - $node->optional ? '?' : '', - $this->printType($node->valueType) - ); - } - - return $this->printType($node->valueType); - } if ($node instanceof ArrayTypeNode) { return $this->printOffsetAccessType($node->type) . '[]'; } @@ -345,10 +413,11 @@ private function printType(TypeNode $node): string } else { $returnType = $this->printType($node->returnType); } - $parameters = implode(', ', array_map(function (CallableTypeParameterNode $parameterNode): string { - return $this->print($parameterNode); - }, $node->parameters)); - return "{$node->identifier}({$parameters}): {$returnType}"; + $template = $node->templateTypes !== [] + ? '<' . implode(', ', array_map(fn (TemplateTagValueNode $templateNode): string => $this->print($templateNode), $node->templateTypes)) . '>' + : ''; + $parameters = implode(', ', array_map(fn (CallableTypeParameterNode $parameterNode): string => $this->print($parameterNode), $node->parameters)); + return "{$node->identifier}{$template}({$parameters}): {$returnType}"; } if ($node instanceof ConditionalTypeForParameterNode) { return sprintf( @@ -357,7 +426,7 @@ private function printType(TypeNode $node): string $node->negated ? 'is not' : 'is', $this->printType($node->targetType), $this->printType($node->if), - $this->printType($node->else) + $this->printType($node->else), ); } if ($node instanceof ConditionalTypeNode) { @@ -367,7 +436,7 @@ private function printType(TypeNode $node): string $node->negated ? 'is not' : 'is', $this->printType($node->targetType), $this->printType($node->if), - $this->printType($node->else) + $this->printType($node->else), ); } if ($node instanceof ConstTypeNode) { @@ -420,24 +489,10 @@ private function printType(TypeNode $node): string return '?' . $this->printType($node->type); } if ($node instanceof ObjectShapeNode) { - $items = array_map(function (ObjectShapeItemNode $item): string { - return $this->printType($item); - }, $node->items); + $items = array_map(fn (ObjectShapeItemNode $item): string => $this->print($item), $node->items); return 'object{' . implode(', ', $items) . '}'; } - if ($node instanceof ObjectShapeItemNode) { - if ($node->keyName !== null) { - return sprintf( - '%s%s: %s', - $this->print($node->keyName), - $node->optional ? '?' : '', - $this->printType($node->valueType) - ); - } - - return $this->printType($node->valueType); - } if ($node instanceof OffsetAccessTypeNode) { return $this->printOffsetAccessType($node->type) . '[' . $this->printType($node->offset) . ']'; } @@ -459,7 +514,6 @@ private function printOffsetAccessType(TypeNode $type): string $type instanceof CallableTypeNode || $type instanceof UnionTypeNode || $type instanceof IntersectionTypeNode - || $type instanceof ConstTypeNode || $type instanceof NullableTypeNode ) { return $this->wrapInParentheses($type); @@ -491,24 +545,35 @@ private function printArrayFormatPreserving(array $nodes, array $originalNodes, [$isMultiline, $beforeAsteriskIndent, $afterAsteriskIndent] = $this->isMultiline($tokenIndex, $originalNodes, $originalTokens); if ($insertStr === "\n * ") { - $insertStr = sprintf("\n%s*%s", $beforeAsteriskIndent, $afterAsteriskIndent); + $insertStr = sprintf('%s%s*%s', $originalTokens->getDetectedNewline() ?? "\n", $beforeAsteriskIndent, $afterAsteriskIndent); } foreach ($diff as $i => $diffElem) { $diffType = $diffElem->type; - $newNode = $diffElem->new; - $originalNode = $diffElem->old; + $arrItem = $diffElem->new; + $origArrayItem = $diffElem->old; if ($diffType === DiffElem::TYPE_KEEP || $diffType === DiffElem::TYPE_REPLACE) { $beforeFirstKeepOrReplace = false; - if (!$newNode instanceof Node || !$originalNode instanceof Node) { + if (!$arrItem instanceof Node || !$origArrayItem instanceof Node) { return null; } - $itemStartPos = $originalNode->getAttribute(Attribute::START_INDEX); - $itemEndPos = $originalNode->getAttribute(Attribute::END_INDEX); + + /** @var int $itemStartPos */ + $itemStartPos = $origArrayItem->getAttribute(Attribute::START_INDEX); + + /** @var int $itemEndPos */ + $itemEndPos = $origArrayItem->getAttribute(Attribute::END_INDEX); + if ($itemStartPos < 0 || $itemEndPos < 0 || $itemStartPos < $tokenIndex) { throw new LogicException(); } + $comments = $arrItem->getAttribute(Attribute::COMMENTS) ?? []; + $origComments = $origArrayItem->getAttribute(Attribute::COMMENTS) ?? []; + + $commentStartPos = count($origComments) > 0 ? $origComments[0]->startIndex : $itemStartPos; + assert($commentStartPos >= 0); + $result .= $originalTokens->getContentBetween($tokenIndex, $itemStartPos); if (count($delayedAdd) > 0) { @@ -518,13 +583,22 @@ private function printArrayFormatPreserving(array $nodes, array $originalNodes, if ($parenthesesNeeded) { $result .= '('; } + + if ($insertNewline) { + $delayedAddComments = $delayedAddNode->getAttribute(Attribute::COMMENTS) ?? []; + if (count($delayedAddComments) > 0) { + $result .= $this->printComments($delayedAddComments, $beforeAsteriskIndent, $afterAsteriskIndent); + $result .= sprintf('%s%s*%s', $originalTokens->getDetectedNewline() ?? "\n", $beforeAsteriskIndent, $afterAsteriskIndent); + } + } + $result .= $this->printNodeFormatPreserving($delayedAddNode, $originalTokens); if ($parenthesesNeeded) { $result .= ')'; } if ($insertNewline) { - $result .= $insertStr . sprintf("\n%s*%s", $beforeAsteriskIndent, $afterAsteriskIndent); + $result .= $insertStr . sprintf('%s%s*%s', $originalTokens->getDetectedNewline() ?? "\n", $beforeAsteriskIndent, $afterAsteriskIndent); } else { $result .= $insertStr; } @@ -534,14 +608,21 @@ private function printArrayFormatPreserving(array $nodes, array $originalNodes, } $parenthesesNeeded = isset($this->parenthesesListMap[$mapKey]) - && in_array(get_class($newNode), $this->parenthesesListMap[$mapKey], true) - && !in_array(get_class($originalNode), $this->parenthesesListMap[$mapKey], true); + && in_array(get_class($arrItem), $this->parenthesesListMap[$mapKey], true) + && !in_array(get_class($origArrayItem), $this->parenthesesListMap[$mapKey], true); $addParentheses = $parenthesesNeeded && !$originalTokens->hasParentheses($itemStartPos, $itemEndPos); if ($addParentheses) { $result .= '('; } - $result .= $this->printNodeFormatPreserving($newNode, $originalTokens); + if ($comments !== $origComments) { + if (count($comments) > 0) { + $result .= $this->printComments($comments, $beforeAsteriskIndent, $afterAsteriskIndent); + $result .= sprintf('%s%s*%s', $originalTokens->getDetectedNewline() ?? "\n", $beforeAsteriskIndent, $afterAsteriskIndent); + } + } + + $result .= $this->printNodeFormatPreserving($arrItem, $originalTokens); if ($addParentheses) { $result .= ')'; } @@ -551,35 +632,42 @@ private function printArrayFormatPreserving(array $nodes, array $originalNodes, if ($insertStr === null) { return null; } - if (!$newNode instanceof Node) { + if (!$arrItem instanceof Node) { return null; } - if ($insertStr === ', ' && $isMultiline) { + if ($insertStr === ', ' && $isMultiline || count($arrItem->getAttribute(Attribute::COMMENTS) ?? []) > 0) { $insertStr = ','; $insertNewline = true; } if ($beforeFirstKeepOrReplace) { // Will be inserted at the next "replace" or "keep" element - $delayedAdd[] = $newNode; + $delayedAdd[] = $arrItem; continue; } + /** @var int $itemEndPos */ $itemEndPos = $tokenIndex - 1; if ($insertNewline) { - $result .= $insertStr . sprintf("\n%s*%s", $beforeAsteriskIndent, $afterAsteriskIndent); + $comments = $arrItem->getAttribute(Attribute::COMMENTS) ?? []; + $result .= $insertStr; + if (count($comments) > 0) { + $result .= sprintf('%s%s*%s', $originalTokens->getDetectedNewline() ?? "\n", $beforeAsteriskIndent, $afterAsteriskIndent); + $result .= $this->printComments($comments, $beforeAsteriskIndent, $afterAsteriskIndent); + } + $result .= sprintf('%s%s*%s', $originalTokens->getDetectedNewline() ?? "\n", $beforeAsteriskIndent, $afterAsteriskIndent); } else { $result .= $insertStr; } $parenthesesNeeded = isset($this->parenthesesListMap[$mapKey]) - && in_array(get_class($newNode), $this->parenthesesListMap[$mapKey], true); + && in_array(get_class($arrItem), $this->parenthesesListMap[$mapKey], true); if ($parenthesesNeeded) { $result .= '('; } - $result .= $this->printNodeFormatPreserving($newNode, $originalTokens); + $result .= $this->printNodeFormatPreserving($arrItem, $originalTokens); if ($parenthesesNeeded) { $result .= ')'; } @@ -587,12 +675,15 @@ private function printArrayFormatPreserving(array $nodes, array $originalNodes, $tokenIndex = $itemEndPos + 1; } elseif ($diffType === DiffElem::TYPE_REMOVE) { - if (!$originalNode instanceof Node) { + if (!$origArrayItem instanceof Node) { return null; } - $itemStartPos = $originalNode->getAttribute(Attribute::START_INDEX); - $itemEndPos = $originalNode->getAttribute(Attribute::END_INDEX); + /** @var int $itemStartPos */ + $itemStartPos = $origArrayItem->getAttribute(Attribute::START_INDEX); + + /** @var int $itemEndPos */ + $itemEndPos = $origArrayItem->getAttribute(Attribute::END_INDEX); if ($itemStartPos < 0 || $itemEndPos < 0) { throw new LogicException(); } @@ -637,7 +728,7 @@ private function printArrayFormatPreserving(array $nodes, array $originalNodes, if (!$first) { $result .= $insertStr; if ($insertNewline) { - $result .= sprintf("\n%s*%s", $beforeAsteriskIndent, $afterAsteriskIndent); + $result .= sprintf('%s%s*%s', $originalTokens->getDetectedNewline() ?? "\n", $beforeAsteriskIndent, $afterAsteriskIndent); } } @@ -651,7 +742,21 @@ private function printArrayFormatPreserving(array $nodes, array $originalNodes, } /** - * @param Node[] $nodes + * @param list $comments + */ + private function printComments(array $comments, string $beforeAsteriskIndent, string $afterAsteriskIndent): string + { + $formattedComments = []; + + foreach ($comments as $comment) { + $formattedComments[] = str_replace("\n", "\n" . $beforeAsteriskIndent . '*' . $afterAsteriskIndent, $comment->getReformattedText()); + } + + return implode("\n$beforeAsteriskIndent*$afterAsteriskIndent", $formattedComments); + } + + /** + * @param array $nodes * @return array{bool, string, string} */ private function isMultiline(int $initialIndex, array $nodes, TokenIterator $originalTokens): array @@ -679,7 +784,7 @@ private function isMultiline(int $initialIndex, array $nodes, TokenIterator $ori $c = preg_match_all('~\n(?[\\x09\\x20]*)\*(?\\x20*)~', $allText, $matches, PREG_SET_ORDER); if ($c === 0) { - return [$isMultiline, '', '']; + return [$isMultiline, ' ', ' ']; } $before = ''; @@ -695,6 +800,9 @@ private function isMultiline(int $initialIndex, array $nodes, TokenIterator $ori $after = $match['after']; } + $before = strlen($before) === 0 ? ' ' : $before; + $after = strlen($after) === 0 ? ' ' : $after; + return [$isMultiline, $before, $after]; } @@ -741,7 +849,7 @@ private function printNodeFormatPreserving(Node $node, TokenIterator $originalTo $originalTokens, $pos, $class, - $subNodeName + $subNodeName, ); if ($listResult === null) { @@ -770,6 +878,10 @@ private function printNodeFormatPreserving(Node $node, TokenIterator $originalTo throw new LogicException(); } + if ($subEndPos < $subStartPos) { + return $this->print($node); + } + if ($subNode === null) { return $this->print($node); } diff --git a/tests/PHPStan/Ast/Attributes/AttributesTest.php b/tests/PHPStan/Ast/Attributes/AttributesTest.php index a20747ec..f513b1fa 100644 --- a/tests/PHPStan/Ast/Attributes/AttributesTest.php +++ b/tests/PHPStan/Ast/Attributes/AttributesTest.php @@ -8,20 +8,22 @@ use PHPStan\PhpDocParser\Parser\PhpDocParser; use PHPStan\PhpDocParser\Parser\TokenIterator; use PHPStan\PhpDocParser\Parser\TypeParser; +use PHPStan\PhpDocParser\ParserConfig; use PHPUnit\Framework\TestCase; final class AttributesTest extends TestCase { - /** @var PhpDocNode */ - private $phpDocNode; + private PhpDocNode $phpDocNode; protected function setUp(): void { parent::setUp(); - $lexer = new Lexer(); - $constExprParser = new ConstExprParser(); - $phpDocParser = new PhpDocParser(new TypeParser($constExprParser), $constExprParser); + + $config = new ParserConfig([]); + $lexer = new Lexer($config); + $constExprParser = new ConstExprParser($config); + $phpDocParser = new PhpDocParser($config, new TypeParser($config, $constExprParser), $constExprParser); $input = '/** @var string */'; $tokens = new TokenIterator($lexer->tokenize($input)); diff --git a/tests/PHPStan/Ast/NodeTraverserTest.php b/tests/PHPStan/Ast/NodeTraverserTest.php index 74089ed2..49f3646b 100644 --- a/tests/PHPStan/Ast/NodeTraverserTest.php +++ b/tests/PHPStan/Ast/NodeTraverserTest.php @@ -163,7 +163,7 @@ public function testMerge(): void $this->assertEquals( [$strStart, $strR1, $strR2, $strEnd], - $traverser->traverse([$strStart, $strMiddle, $strEnd]) + $traverser->traverse([$strStart, $strMiddle, $strEnd]), ); } @@ -393,7 +393,7 @@ public function provideTestInvalidReturn(): array ['enterNode', $expr, new ConstExprIntegerNode('42')], ]); $visitor8 = new NodeVisitorForTesting([ - ['enterNode', $num, new ReturnTagValueNode(new ConstTypeNode(new ConstExprStringNode('foo')), '')], + ['enterNode', $num, new ReturnTagValueNode(new ConstTypeNode(new ConstExprStringNode('foo', ConstExprStringNode::SINGLE_QUOTED)), '')], ]); return [ diff --git a/tests/PHPStan/Ast/NodeVisitorForTesting.php b/tests/PHPStan/Ast/NodeVisitorForTesting.php index f3ca0ac0..fa5dd26d 100644 --- a/tests/PHPStan/Ast/NodeVisitorForTesting.php +++ b/tests/PHPStan/Ast/NodeVisitorForTesting.php @@ -15,13 +15,12 @@ class NodeVisitorForTesting implements NodeVisitor { /** @var list */ - public $trace = []; + public array $trace = []; /** @var list> */ - private $returns; + private array $returns; - /** @var int */ - private $returnsPos; + private int $returnsPos; /** * @param list> $returns diff --git a/tests/PHPStan/Ast/ToString/ConstExprToStringTest.php b/tests/PHPStan/Ast/ToString/ConstExprToStringTest.php index fe8be67c..b7568879 100644 --- a/tests/PHPStan/Ast/ToString/ConstExprToStringTest.php +++ b/tests/PHPStan/Ast/ToString/ConstExprToStringTest.php @@ -44,16 +44,17 @@ public static function provideConstExprCases(): Generator ['false', new ConstExprFalseNode()], ['8', new ConstExprIntegerNode('8')], ['21.37', new ConstExprFloatNode('21.37')], - ['foo', new ConstExprStringNode('foo')], + ['\'foo\'', new ConstExprStringNode('foo', ConstExprStringNode::SINGLE_QUOTED)], + ['"foo"', new ConstExprStringNode('foo', ConstExprStringNode::DOUBLE_QUOTED)], ['FooBar', new ConstFetchNode('', 'FooBar')], ['Foo\\Bar::Baz', new ConstFetchNode('Foo\\Bar', 'Baz')], ['[]', new ConstExprArrayNode([])], [ - '[foo, 4 => foo, bar => baz]', + "['foo', 4 => 'foo', 'bar' => 'baz']", new ConstExprArrayNode([ - new ConstExprArrayItemNode(null, new ConstExprStringNode('foo')), - new ConstExprArrayItemNode(new ConstExprIntegerNode('4'), new ConstExprStringNode('foo')), - new ConstExprArrayItemNode(new ConstExprStringNode('bar'), new ConstExprStringNode('baz')), + new ConstExprArrayItemNode(null, new ConstExprStringNode('foo', ConstExprStringNode::SINGLE_QUOTED)), + new ConstExprArrayItemNode(new ConstExprIntegerNode('4'), new ConstExprStringNode('foo', ConstExprStringNode::SINGLE_QUOTED)), + new ConstExprArrayItemNode(new ConstExprStringNode('bar', ConstExprStringNode::SINGLE_QUOTED), new ConstExprStringNode('baz', ConstExprStringNode::SINGLE_QUOTED)), ]), ], ]; diff --git a/tests/PHPStan/Ast/ToString/PhpDocToStringTest.php b/tests/PHPStan/Ast/ToString/PhpDocToStringTest.php index 01ba2f3c..47a25db3 100644 --- a/tests/PHPStan/Ast/ToString/PhpDocToStringTest.php +++ b/tests/PHPStan/Ast/ToString/PhpDocToStringTest.php @@ -3,12 +3,18 @@ namespace PHPStan\PhpDocParser\Ast\ToString; use Generator; +use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprIntegerNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprStringNode; use PHPStan\PhpDocParser\Ast\Node; use PHPStan\PhpDocParser\Ast\PhpDoc\AssertTagMethodValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\AssertTagPropertyValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\AssertTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\DeprecatedTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineAnnotation; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArgument; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArray; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArrayItem; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ExtendsTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\GenericTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ImplementsTagValueNode; @@ -22,6 +28,8 @@ use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocTagNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocTextNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PropertyTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\RequireExtendsTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\RequireImplementsTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ReturnTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\SelfOutTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\TemplateTagValueNode; @@ -63,6 +71,7 @@ public function testFullPhpDocPrinter(string $expected, Node $node): void * @dataProvider provideMethodCases * @dataProvider provideClassCases * @dataProvider provideAssertionCases + * @dataProvider provideDoctrineCases */ public function testTagValueNodeToString(string $expected, Node $node): void { @@ -74,6 +83,7 @@ public function testTagValueNodeToString(string $expected, Node $node): void * @dataProvider provideMethodCases * @dataProvider provideClassCases * @dataProvider provideAssertionCases + * @dataProvider provideDoctrineCases */ public function testTagValueNodePrinter(string $expected, Node $node): void { @@ -127,7 +137,7 @@ public static function provideOtherCases(): Generator '#desc', new InvalidTagValueNode( '#desc', - new ParserException('#desc', Lexer::TOKEN_OTHER, 11, Lexer::TOKEN_IDENTIFIER) + new ParserException('#desc', Lexer::TOKEN_OTHER, 11, Lexer::TOKEN_IDENTIFIER, null, null), ), ]; @@ -145,12 +155,15 @@ public static function provideOtherCases(): Generator $baz = new IdentifierTypeNode('Foo\\Baz'); yield from [ - ['TValue', new TemplateTagValueNode('TValue', null, '', null)], - ['TValue of Foo\\Bar', new TemplateTagValueNode('TValue', $bar, '', null)], + ['TValue', new TemplateTagValueNode('TValue', null, '')], + ['TValue of Foo\\Bar', new TemplateTagValueNode('TValue', $bar, '')], + ['TValue super Foo\\Bar', new TemplateTagValueNode('TValue', null, '', null, $bar)], ['TValue = Foo\\Bar', new TemplateTagValueNode('TValue', null, '', $bar)], ['TValue of Foo\\Bar = Foo\\Baz', new TemplateTagValueNode('TValue', $bar, '', $baz)], - ['TValue Description.', new TemplateTagValueNode('TValue', null, 'Description.', null)], + ['TValue Description.', new TemplateTagValueNode('TValue', null, 'Description.')], ['TValue of Foo\\Bar = Foo\\Baz Description.', new TemplateTagValueNode('TValue', $bar, 'Description.', $baz)], + ['TValue super Foo\\Bar = Foo\\Baz Description.', new TemplateTagValueNode('TValue', null, 'Description.', $baz, $bar)], + ['TValue of Foo\\Bar super Foo\\Baz Description.', new TemplateTagValueNode('TValue', $bar, 'Description.', null, $baz)], ]; } @@ -205,6 +218,15 @@ public static function provideClassCases(): Generator ['Foo\\Bar Baz', new MixinTagValueNode(new IdentifierTypeNode('Foo\\Bar'), 'Baz')], ]; + yield from [ + ['PHPUnit\\TestCase', new RequireExtendsTagValueNode(new IdentifierTypeNode('PHPUnit\\TestCase'), '')], + ['Foo\\Bar Baz', new RequireExtendsTagValueNode(new IdentifierTypeNode('Foo\\Bar'), 'Baz')], + ]; + yield from [ + ['PHPUnit\\TestCase', new RequireImplementsTagValueNode(new IdentifierTypeNode('PHPUnit\\TestCase'), '')], + ['Foo\\Bar Baz', new RequireImplementsTagValueNode(new IdentifierTypeNode('Foo\\Bar'), 'Baz')], + ]; + yield from [ ['Foo array', new TypeAliasTagValueNode('Foo', $arrayOfStrings)], ['Test from Foo\Bar', new TypeAliasImportTagValueNode('Test', $bar, null)], @@ -261,8 +283,8 @@ public static function provideClassCases(): Generator new MethodTagValueParameterNode($string, true, false, 'foo', null), ], [ - 'string &foo = bar', - new MethodTagValueParameterNode($string, true, false, 'foo', new ConstExprStringNode('bar')), + 'string &foo = \'bar\'', + new MethodTagValueParameterNode($string, true, false, 'foo', new ConstExprStringNode('bar', ConstExprStringNode::SINGLE_QUOTED)), ], [ '&...foo', @@ -351,4 +373,78 @@ public static function provideAssertionCases(): Generator ]; } + /** + * @return iterable + */ + public static function provideDoctrineCases(): iterable + { + yield [ + '@ORM\Entity()', + new PhpDocTagNode('@ORM\Entity', new DoctrineTagValueNode( + new DoctrineAnnotation('@ORM\Entity', []), + '', + )), + ]; + + yield [ + '@ORM\Entity() test', + new PhpDocTagNode('@ORM\Entity', new DoctrineTagValueNode( + new DoctrineAnnotation('@ORM\Entity', []), + 'test', + )), + ]; + + yield [ + '@ORM\Entity(1, b=2)', + new DoctrineTagValueNode( + new DoctrineAnnotation('@ORM\Entity', [ + new DoctrineArgument(null, new ConstExprIntegerNode('1')), + new DoctrineArgument(new IdentifierTypeNode('b'), new ConstExprIntegerNode('2')), + ]), + '', + ), + ]; + + yield [ + '{}', + new DoctrineArray([]), + ]; + + yield [ + '{1, \'a\'=2}', + new DoctrineArray([ + new DoctrineArrayItem(null, new ConstExprIntegerNode('1')), + new DoctrineArrayItem(new ConstExprStringNode('a', ConstExprStringNode::SINGLE_QUOTED), new ConstExprIntegerNode('2')), + ]), + ]; + + yield [ + '1', + new DoctrineArrayItem(null, new ConstExprIntegerNode('1')), + ]; + + yield [ + '\'a\'=2', + new DoctrineArrayItem(new ConstExprStringNode('a', ConstExprStringNode::SINGLE_QUOTED), new ConstExprIntegerNode('2')), + ]; + + yield [ + '@ORM\Entity(1, b=2)', + new DoctrineAnnotation('@ORM\Entity', [ + new DoctrineArgument(null, new ConstExprIntegerNode('1')), + new DoctrineArgument(new IdentifierTypeNode('b'), new ConstExprIntegerNode('2')), + ]), + ]; + + yield [ + '1', + new DoctrineArgument(null, new ConstExprIntegerNode('1')), + ]; + + yield [ + 'b=2', + new DoctrineArgument(new IdentifierTypeNode('b'), new ConstExprIntegerNode('2')), + ]; + } + } diff --git a/tests/PHPStan/Ast/ToString/TypeToStringTest.php b/tests/PHPStan/Ast/ToString/TypeToStringTest.php index 9cf7ebf6..6716e433 100644 --- a/tests/PHPStan/Ast/ToString/TypeToStringTest.php +++ b/tests/PHPStan/Ast/ToString/TypeToStringTest.php @@ -58,31 +58,31 @@ public static function provideArrayCases(): Generator ]; yield from [ - ['array{}', new ArrayShapeNode([])], - ['array{...}', new ArrayShapeNode([], false)], + ['array{}', ArrayShapeNode::createSealed([])], + ['array{...}', ArrayShapeNode::createUnsealed([], null)], [ 'array{string, int, ...}', - new ArrayShapeNode([ + ArrayShapeNode::createUnsealed([ new ArrayShapeItemNode(null, false, new IdentifierTypeNode('string')), new ArrayShapeItemNode(null, false, new IdentifierTypeNode('int')), - ], false), + ], null), ], [ - 'array{foo: Foo, bar?: Bar, 1: Baz}', - new ArrayShapeNode([ - new ArrayShapeItemNode(new ConstExprStringNode('foo'), false, new IdentifierTypeNode('Foo')), - new ArrayShapeItemNode(new ConstExprStringNode('bar'), true, new IdentifierTypeNode('Bar')), + 'array{\'foo\': Foo, \'bar\'?: Bar, 1: Baz}', + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new ConstExprStringNode('foo', ConstExprStringNode::SINGLE_QUOTED), false, new IdentifierTypeNode('Foo')), + new ArrayShapeItemNode(new ConstExprStringNode('bar', ConstExprStringNode::SINGLE_QUOTED), true, new IdentifierTypeNode('Bar')), new ArrayShapeItemNode(new ConstExprIntegerNode('1'), false, new IdentifierTypeNode('Baz')), ]), ], - ['list{}', new ArrayShapeNode([], true, 'list')], - ['list{...}', new ArrayShapeNode([], false, 'list')], + ['list{}', ArrayShapeNode::createSealed([], 'list')], + ['list{...}', ArrayShapeNode::createUnsealed([], null, 'list')], [ 'list{string, int, ...}', - new ArrayShapeNode([ + ArrayShapeNode::createUnsealed([ new ArrayShapeItemNode(null, false, new IdentifierTypeNode('string')), new ArrayShapeItemNode(null, false, new IdentifierTypeNode('int')), - ], false, 'list'), + ], null, 'list'), ], ]; } @@ -92,34 +92,34 @@ public static function provideCallableCases(): Generator yield from [ [ '\\Closure(): string', - new CallableTypeNode(new IdentifierTypeNode('\Closure'), [], new IdentifierTypeNode('string')), + new CallableTypeNode(new IdentifierTypeNode('\Closure'), [], new IdentifierTypeNode('string'), []), ], [ 'callable(int, int $foo): void', new CallableTypeNode(new IdentifierTypeNode('callable'), [ new CallableTypeParameterNode(new IdentifierTypeNode('int'), false, false, '', false), new CallableTypeParameterNode(new IdentifierTypeNode('int'), false, false, '$foo', false), - ], new IdentifierTypeNode('void')), + ], new IdentifierTypeNode('void'), []), ], [ 'callable(int=, int $foo=): void', new CallableTypeNode(new IdentifierTypeNode('callable'), [ new CallableTypeParameterNode(new IdentifierTypeNode('int'), false, false, '', true), new CallableTypeParameterNode(new IdentifierTypeNode('int'), false, false, '$foo', true), - ], new IdentifierTypeNode('void')), + ], new IdentifierTypeNode('void'), []), ], [ 'callable(int &, int &$foo): void', new CallableTypeNode(new IdentifierTypeNode('callable'), [ new CallableTypeParameterNode(new IdentifierTypeNode('int'), true, false, '', false), new CallableTypeParameterNode(new IdentifierTypeNode('int'), true, false, '$foo', false), - ], new IdentifierTypeNode('void')), + ], new IdentifierTypeNode('void'), []), ], [ 'callable(string ...$foo): void', new CallableTypeNode(new IdentifierTypeNode('callable'), [ new CallableTypeParameterNode(new IdentifierTypeNode('string'), false, true, '$foo', false), - ], new IdentifierTypeNode('void')), + ], new IdentifierTypeNode('void'), []), ], ]; } @@ -136,7 +136,7 @@ public static function provideGenericCases(): Generator new GenericTypeNode( new IdentifierTypeNode('array'), [new IdentifierTypeNode('string'), new IdentifierTypeNode('int')], - [GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_BIVARIANT] + [GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_BIVARIANT], ), ], [ @@ -144,7 +144,7 @@ public static function provideGenericCases(): Generator new GenericTypeNode( new IdentifierTypeNode('Foo\\Bar'), [new IdentifierTypeNode('string'), new IdentifierTypeNode('int')], - [GenericTypeNode::VARIANCE_COVARIANT, GenericTypeNode::VARIANCE_CONTRAVARIANT] + [GenericTypeNode::VARIANCE_COVARIANT, GenericTypeNode::VARIANCE_CONTRAVARIANT], ), ], ]; @@ -160,7 +160,7 @@ public static function provideConditionalCases(): Generator new IdentifierTypeNode('int'), new GenericTypeNode(new IdentifierTypeNode('list'), [new IdentifierTypeNode('int')]), new GenericTypeNode(new IdentifierTypeNode('list'), [new IdentifierTypeNode('string')]), - false + false, ), ], [ @@ -170,7 +170,7 @@ public static function provideConditionalCases(): Generator new IdentifierTypeNode('array'), new IdentifierTypeNode('int'), new ArrayTypeNode(new IdentifierTypeNode('int')), - true + true, ), ], [ @@ -180,7 +180,7 @@ public static function provideConditionalCases(): Generator new IdentifierTypeNode('Exception'), new IdentifierTypeNode('never'), new IdentifierTypeNode('string'), - false + false, ), ], [ @@ -190,7 +190,7 @@ public static function provideConditionalCases(): Generator new IdentifierTypeNode('Exception'), new IdentifierTypeNode('string'), new IdentifierTypeNode('never'), - true + true, ), ], ]; diff --git a/tests/PHPStan/Parser/ConstExprParserTest.php b/tests/PHPStan/Parser/ConstExprParserTest.php index a8fb1b52..18922e86 100644 --- a/tests/PHPStan/Parser/ConstExprParserTest.php +++ b/tests/PHPStan/Parser/ConstExprParserTest.php @@ -16,22 +16,22 @@ use PHPStan\PhpDocParser\Ast\ConstExpr\ConstFetchNode; use PHPStan\PhpDocParser\Ast\NodeTraverser; use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\ParserConfig; use PHPUnit\Framework\TestCase; class ConstExprParserTest extends TestCase { - /** @var Lexer */ - private $lexer; + private Lexer $lexer; - /** @var ConstExprParser */ - private $constExprParser; + private ConstExprParser $constExprParser; protected function setUp(): void { parent::setUp(); - $this->lexer = new Lexer(); - $this->constExprParser = new ConstExprParser(true); + $config = new ParserConfig([]); + $this->lexer = new Lexer($config); + $this->constExprParser = new ConstExprParser($config); } @@ -65,16 +65,15 @@ public function testParse(string $input, ConstExprNode $expectedExpr, int $nextT * @dataProvider provideStringNodeParseData * @dataProvider provideArrayNodeParseData * @dataProvider provideFetchNodeParseData - * - * @dataProvider provideWithTrimStringsStringNodeParseData */ public function testVerifyAttributes(string $input): void { $tokens = new TokenIterator($this->lexer->tokenize($input)); - $constExprParser = new ConstExprParser(true, true, [ + $config = new ParserConfig([ 'lines' => true, 'indexes' => true, ]); + $constExprParser = new ConstExprParser($config); $visitor = new NodeCollectingVisitor(); $traverser = new NodeTraverser([$visitor]); $traverser->traverse([$constExprParser->parse($tokens)]); @@ -152,6 +151,16 @@ public function provideIntegerNodeParseData(): Iterator new ConstExprIntegerNode('123'), ]; + yield [ + '+123', + new ConstExprIntegerNode('+123'), + ]; + + yield [ + '-123', + new ConstExprIntegerNode('-123'), + ]; + yield [ '0b0110101', new ConstExprIntegerNode('0b0110101'), @@ -191,6 +200,11 @@ public function provideIntegerNodeParseData(): Iterator '-0X7_Fb_4', new ConstExprIntegerNode('-0X7Fb4'), ]; + + yield [ + '18_446_744_073_709_551_616', // 64-bit unsigned long + 1, larger than PHP_INT_MAX + new ConstExprIntegerNode('18446744073709551616'), + ]; } @@ -227,8 +241,13 @@ public function provideFloatNodeParseData(): Iterator ]; yield [ - '-123', - new ConstExprIntegerNode('-123'), + '+123.5', + new ConstExprFloatNode('+123.5'), + ]; + + yield [ + '-123.', + new ConstExprFloatNode('-123.'), ]; yield [ @@ -260,6 +279,31 @@ public function provideFloatNodeParseData(): Iterator '-1_2.3_4e5_6', new ConstExprFloatNode('-12.34e56'), ]; + + yield [ + '123.4e+8', + new ConstExprFloatNode('123.4e+8'), + ]; + + yield [ + '.4e+8', + new ConstExprFloatNode('.4e+8'), + ]; + + yield [ + '123E+80', + new ConstExprFloatNode('123E+80'), + ]; + + yield [ + '8.2023437675747321', // greater precision than 64-bit double + new ConstExprFloatNode('8.2023437675747321'), + ]; + + yield [ + '-0.0', + new ConstExprFloatNode('-0.0'), + ]; } @@ -267,22 +311,32 @@ public function provideStringNodeParseData(): Iterator { yield [ '"foo"', - new ConstExprStringNode('"foo"'), + new ConstExprStringNode('foo', ConstExprStringNode::DOUBLE_QUOTED), ]; yield [ '"Foo \\n\\"\\r Bar"', - new ConstExprStringNode('"Foo \\n\\"\\r Bar"'), + new ConstExprStringNode("Foo \n\"\r Bar", ConstExprStringNode::DOUBLE_QUOTED), ]; yield [ '\'bar\'', - new ConstExprStringNode('\'bar\''), + new ConstExprStringNode('bar', ConstExprStringNode::SINGLE_QUOTED), ]; yield [ '\'Foo \\\' Bar\'', - new ConstExprStringNode('\'Foo \\\' Bar\''), + new ConstExprStringNode('Foo \' Bar', ConstExprStringNode::SINGLE_QUOTED), + ]; + + yield [ + '"\u{1f601}"', + new ConstExprStringNode("\u{1f601}", ConstExprStringNode::DOUBLE_QUOTED), + ]; + + yield [ + '"\u{ffffffff}"', + new ConstExprStringNode("\u{fffd}", ConstExprStringNode::DOUBLE_QUOTED), ]; } @@ -299,7 +353,7 @@ public function provideArrayNodeParseData(): Iterator new ConstExprArrayNode([ new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('123') + new ConstExprIntegerNode('123'), ), ]), ]; @@ -309,15 +363,15 @@ public function provideArrayNodeParseData(): Iterator new ConstExprArrayNode([ new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('1') + new ConstExprIntegerNode('1'), ), new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('2') + new ConstExprIntegerNode('2'), ), new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('3') + new ConstExprIntegerNode('3'), ), ]), ]; @@ -327,15 +381,15 @@ public function provideArrayNodeParseData(): Iterator new ConstExprArrayNode([ new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('1') + new ConstExprIntegerNode('1'), ), new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('2') + new ConstExprIntegerNode('2'), ), new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('3') + new ConstExprIntegerNode('3'), ), ]), ]; @@ -345,7 +399,7 @@ public function provideArrayNodeParseData(): Iterator new ConstExprArrayNode([ new ConstExprArrayItemNode( new ConstExprIntegerNode('1'), - new ConstExprIntegerNode('2') + new ConstExprIntegerNode('2'), ), ]), ]; @@ -355,11 +409,11 @@ public function provideArrayNodeParseData(): Iterator new ConstExprArrayNode([ new ConstExprArrayItemNode( new ConstExprIntegerNode('1'), - new ConstExprIntegerNode('2') + new ConstExprIntegerNode('2'), ), new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('3') + new ConstExprIntegerNode('3'), ), ]), ]; @@ -369,20 +423,20 @@ public function provideArrayNodeParseData(): Iterator new ConstExprArrayNode([ new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('1') + new ConstExprIntegerNode('1'), ), new ConstExprArrayItemNode( null, new ConstExprArrayNode([ new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('2') + new ConstExprIntegerNode('2'), ), new ConstExprArrayItemNode( null, - new ConstExprIntegerNode('3') + new ConstExprIntegerNode('3'), ), - ]) + ]), ), ]), ]; @@ -412,50 +466,4 @@ public function provideFetchNodeParseData(): Iterator ]; } - /** - * @dataProvider provideWithTrimStringsStringNodeParseData - */ - public function testParseWithTrimStrings(string $input, ConstExprNode $expectedExpr, int $nextTokenType = Lexer::TOKEN_END): void - { - $tokens = new TokenIterator($this->lexer->tokenize($input)); - $exprNode = $this->constExprParser->parse($tokens, true); - - $this->assertSame((string) $expectedExpr, (string) $exprNode); - $this->assertEquals($expectedExpr, $exprNode); - $this->assertSame($nextTokenType, $tokens->currentTokenType()); - } - - public function provideWithTrimStringsStringNodeParseData(): Iterator - { - yield [ - '"foo"', - new ConstExprStringNode('foo'), - ]; - - yield [ - '"Foo \\n\\"\\r Bar"', - new ConstExprStringNode("Foo \n\"\r Bar"), - ]; - - yield [ - '\'bar\'', - new ConstExprStringNode('bar'), - ]; - - yield [ - '\'Foo \\\' Bar\'', - new ConstExprStringNode('Foo \' Bar'), - ]; - - yield [ - '"\u{1f601}"', - new ConstExprStringNode("\u{1f601}"), - ]; - - yield [ - '"\u{ffffffff}"', - new ConstExprStringNode("\u{fffd}"), - ]; - } - } diff --git a/tests/PHPStan/Parser/Doctrine/ApiResource.php b/tests/PHPStan/Parser/Doctrine/ApiResource.php new file mode 100644 index 00000000..9134705f --- /dev/null +++ b/tests/PHPStan/Parser/Doctrine/ApiResource.php @@ -0,0 +1,28 @@ + + * + * @Annotation + * @Target({"CLASS"}) + */ +final class ApiResource +{ + + public string $shortName; + + public string $description; + + public string $iri; + + public array $itemOperations; + + public array $collectionOperations; + + public array $attributes = []; + +} diff --git a/tests/PHPStan/Parser/Doctrine/X.php b/tests/PHPStan/Parser/Doctrine/X.php new file mode 100644 index 00000000..03d6fda5 --- /dev/null +++ b/tests/PHPStan/Parser/Doctrine/X.php @@ -0,0 +1,17 @@ +lexer = new Lexer(); - $this->typeParser = new TypeParser(new ConstExprParser()); - $this->constExprParser = new ConstExprParser(); + $config = new ParserConfig([]); + $this->lexer = new Lexer($config); + $this->typeParser = new TypeParser($config, new ConstExprParser($config)); + $this->constExprParser = new ConstExprParser($config); } /** @@ -44,7 +46,7 @@ public function testTypeParser(string $input): void $this->assertSame( Lexer::TOKEN_END, $tokens->currentTokenType(), - sprintf('Failed to parse input %s', $input) + sprintf('Failed to parse input %s', $input), ); } @@ -64,7 +66,7 @@ public function testConstExprParser(string $input): void $this->assertSame( Lexer::TOKEN_END, $tokens->currentTokenType(), - sprintf('Failed to parse input %s', $input) + sprintf('Failed to parse input %s', $input), ); } diff --git a/tests/PHPStan/Parser/NodeCollectingVisitor.php b/tests/PHPStan/Parser/NodeCollectingVisitor.php index aa01e559..dbce9dfc 100644 --- a/tests/PHPStan/Parser/NodeCollectingVisitor.php +++ b/tests/PHPStan/Parser/NodeCollectingVisitor.php @@ -9,7 +9,7 @@ class NodeCollectingVisitor extends AbstractNodeVisitor { /** @var list */ - public $nodes = []; + public array $nodes = []; public function enterNode(Node $node) { diff --git a/tests/PHPStan/Parser/PhpDocParserTest.php b/tests/PHPStan/Parser/PhpDocParserTest.php index 0d533f70..abe69b8e 100644 --- a/tests/PHPStan/Parser/PhpDocParserTest.php +++ b/tests/PHPStan/Parser/PhpDocParserTest.php @@ -2,6 +2,7 @@ namespace PHPStan\PhpDocParser\Parser; +use Doctrine\Common\Annotations\DocParser; use Iterator; use PHPStan\PhpDocParser\Ast\Attribute; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprArrayItemNode; @@ -9,12 +10,18 @@ use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprIntegerNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprStringNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstFetchNode; +use PHPStan\PhpDocParser\Ast\ConstExpr\DoctrineConstExprStringNode; use PHPStan\PhpDocParser\Ast\Node; use PHPStan\PhpDocParser\Ast\NodeTraverser; use PHPStan\PhpDocParser\Ast\PhpDoc\AssertTagMethodValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\AssertTagPropertyValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\AssertTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\DeprecatedTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineAnnotation; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArgument; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArray; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArrayItem; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ExtendsTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\GenericTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ImplementsTagValueNode; @@ -22,12 +29,18 @@ use PHPStan\PhpDocParser\Ast\PhpDoc\MethodTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\MethodTagValueParameterNode; use PHPStan\PhpDocParser\Ast\PhpDoc\MixinTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\ParamClosureThisTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\ParamImmediatelyInvokedCallableTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\ParamLaterInvokedCallableTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ParamOutTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ParamTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocTagNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocTextNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PropertyTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\PureUnlessCallableIsImpureTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\RequireExtendsTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\RequireImplementsTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ReturnTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\SelfOutTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\TemplateTagValueNode; @@ -47,40 +60,36 @@ use PHPStan\PhpDocParser\Ast\Type\ConstTypeNode; use PHPStan\PhpDocParser\Ast\Type\GenericTypeNode; use PHPStan\PhpDocParser\Ast\Type\IdentifierTypeNode; +use PHPStan\PhpDocParser\Ast\Type\IntersectionTypeNode; use PHPStan\PhpDocParser\Ast\Type\InvalidTypeNode; use PHPStan\PhpDocParser\Ast\Type\NullableTypeNode; +use PHPStan\PhpDocParser\Ast\Type\ObjectShapeItemNode; +use PHPStan\PhpDocParser\Ast\Type\ObjectShapeNode; use PHPStan\PhpDocParser\Ast\Type\OffsetAccessTypeNode; use PHPStan\PhpDocParser\Ast\Type\UnionTypeNode; use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\ParserConfig; use PHPUnit\Framework\TestCase; use function count; use function sprintf; +use const DIRECTORY_SEPARATOR; use const PHP_EOL; class PhpDocParserTest extends TestCase { - /** @var Lexer */ - private $lexer; + private Lexer $lexer; - /** @var PhpDocParser */ - private $phpDocParser; - - /** @var PhpDocParser */ - private $phpDocParserWithRequiredWhitespaceBeforeDescription; - - /** @var PhpDocParser */ - private $phpDocParserWithPreserveTypeAliasesWithInvalidTypes; + private PhpDocParser $phpDocParser; protected function setUp(): void { parent::setUp(); - $this->lexer = new Lexer(); - $constExprParser = new ConstExprParser(); - $typeParser = new TypeParser($constExprParser); - $this->phpDocParser = new PhpDocParser($typeParser, $constExprParser); - $this->phpDocParserWithRequiredWhitespaceBeforeDescription = new PhpDocParser($typeParser, $constExprParser, true); - $this->phpDocParserWithPreserveTypeAliasesWithInvalidTypes = new PhpDocParser($typeParser, $constExprParser, true, true); + $config = new ParserConfig([]); + $this->lexer = new Lexer($config); + $constExprParser = new ConstExprParser($config); + $typeParser = new TypeParser($config, $constExprParser); + $this->phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); } @@ -88,11 +97,17 @@ protected function setUp(): void * @dataProvider provideTagsWithNumbers * @dataProvider provideSpecializedTags * @dataProvider provideParamTagsData + * @dataProvider provideParamImmediatelyInvokedCallableTagsData + * @dataProvider provideParamLaterInvokedCallableTagsData * @dataProvider provideTypelessParamTagsData + * @dataProvider provideParamClosureThisTagsData + * @dataProvider providePureUnlessCallableIsImpureTagsData * @dataProvider provideVarTagsData * @dataProvider provideReturnTagsData * @dataProvider provideThrowsTagsData * @dataProvider provideMixinTagsData + * @dataProvider provideRequireExtendsTagsData + * @dataProvider provideRequireImplementsTagsData * @dataProvider provideDeprecatedTagsData * @dataProvider providePropertyTagsData * @dataProvider provideMethodTagsData @@ -108,34 +123,22 @@ protected function setUp(): void * @dataProvider provideTagsWithBackslash * @dataProvider provideSelfOutTagsData * @dataProvider provideParamOutTagsData + * @dataProvider provideDoctrineData + * @dataProvider provideDoctrineWithoutDoctrineCheckData + * @dataProvider provideCommentLikeDescriptions + * @dataProvider provideInlineTags */ public function testParse( string $label, string $input, - PhpDocNode $expectedPhpDocNode, - ?PhpDocNode $withRequiredWhitespaceBeforeDescriptionExpectedPhpDocNode = null, - ?PhpDocNode $withPreserveTypeAliasesWithInvalidTypesExpectedPhpDocNode = null + PhpDocNode $expectedPhpDocNode ): void { $this->executeTestParse( $this->phpDocParser, $label, $input, - $expectedPhpDocNode - ); - - $this->executeTestParse( - $this->phpDocParserWithRequiredWhitespaceBeforeDescription, - $label, - $input, - $withRequiredWhitespaceBeforeDescriptionExpectedPhpDocNode ?? $expectedPhpDocNode - ); - - $this->executeTestParse( - $this->phpDocParserWithPreserveTypeAliasesWithInvalidTypes, - $label, - $input, - $withPreserveTypeAliasesWithInvalidTypesExpectedPhpDocNode ?? $withRequiredWhitespaceBeforeDescriptionExpectedPhpDocNode ?? $expectedPhpDocNode + $expectedPhpDocNode, ); } @@ -146,8 +149,8 @@ private function executeTestParse(PhpDocParser $phpDocParser, string $label, str $actualPhpDocNode = $phpDocParser->parse($tokens); $this->assertEquals($expectedPhpDocNode, $actualPhpDocNode, $label); - $this->assertSame((string) $expectedPhpDocNode, (string) $actualPhpDocNode); - $this->assertSame(Lexer::TOKEN_END, $tokens->currentTokenType()); + $this->assertSame((string) $expectedPhpDocNode, (string) $actualPhpDocNode, $label); + $this->assertSame(Lexer::TOKEN_END, $tokens->currentTokenType(), $label); } @@ -163,8 +166,9 @@ public function provideParamTagsData(): Iterator new IdentifierTypeNode('Foo'), false, '$foo', - '' - ) + '', + false, + ), ), ]), ]; @@ -179,8 +183,9 @@ public function provideParamTagsData(): Iterator new IdentifierTypeNode('Foo'), false, '$foo', - 'optional description' - ) + 'optional description', + false, + ), ), ]), ]; @@ -195,8 +200,9 @@ public function provideParamTagsData(): Iterator new IdentifierTypeNode('Foo'), true, '$foo', - '' - ) + '', + false, + ), ), ]), ]; @@ -211,8 +217,9 @@ public function provideParamTagsData(): Iterator new IdentifierTypeNode('Foo'), true, '$foo', - 'optional description' - ) + 'optional description', + false, + ), ), ]), ]; @@ -228,8 +235,8 @@ public function provideParamTagsData(): Iterator false, '$foo', '', - true - ) + true, + ), ), ]), ]; @@ -245,8 +252,8 @@ public function provideParamTagsData(): Iterator false, '$foo', 'optional description', - true - ) + true, + ), ), ]), ]; @@ -262,8 +269,8 @@ public function provideParamTagsData(): Iterator true, '$foo', '', - true - ) + true, + ), ), ]), ]; @@ -279,8 +286,8 @@ public function provideParamTagsData(): Iterator true, '$foo', 'optional description', - true - ) + true, + ), ), ]), ]; @@ -295,8 +302,9 @@ public function provideParamTagsData(): Iterator new ConstTypeNode(new ConstFetchNode('self', '*')), false, '$foo', - 'optional description' - ) + 'optional description', + false, + ), ), ]), ]; @@ -315,9 +323,9 @@ public function provideParamTagsData(): Iterator 11, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -336,9 +344,9 @@ public function provideParamTagsData(): Iterator 11, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -357,9 +365,9 @@ public function provideParamTagsData(): Iterator 16, Lexer::TOKEN_CLOSE_PARENTHESES, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -378,9 +386,9 @@ public function provideParamTagsData(): Iterator 16, Lexer::TOKEN_CLOSE_PARENTHESES, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -399,9 +407,9 @@ public function provideParamTagsData(): Iterator 19, Lexer::TOKEN_CLOSE_ANGLE_BRACKET, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -420,9 +428,9 @@ public function provideParamTagsData(): Iterator 16, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -441,9 +449,9 @@ public function provideParamTagsData(): Iterator 15, Lexer::TOKEN_VARIABLE, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -464,9 +472,9 @@ public function provideParamTagsData(): Iterator 21, Lexer::TOKEN_VARIABLE, null, - 2 - ) - ) + 2, + ), + ), ), ]), ]; @@ -485,9 +493,9 @@ public function provideParamTagsData(): Iterator 15, Lexer::TOKEN_VARIABLE, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -512,8 +520,9 @@ public function provideParamTagsData(): Iterator false, '$parameters', '{' . PHP_EOL . - ' Optional. Parameters for filtering the list of user assignments. Default empty array.' - ) + ' Optional. Parameters for filtering the list of user assignments. Default empty array.', + false, + ), ), new PhpDocTextNode(''), new PhpDocTagNode('@type', new GenericTagValueNode('bool $is_active Pass `true` to only return active user assignments and `false` to' . PHP_EOL . @@ -536,8 +545,9 @@ public function provideTypelessParamTagsData(): Iterator new TypelessParamTagValueNode( false, '$foo', - 'description' - ) + 'description', + false, + ), ), ]), ]; @@ -552,8 +562,8 @@ public function provideTypelessParamTagsData(): Iterator false, '$foo', 'description', - true - ) + true, + ), ), ]), ]; @@ -567,8 +577,9 @@ public function provideTypelessParamTagsData(): Iterator new TypelessParamTagValueNode( true, '$foo', - 'description' - ) + 'description', + false, + ), ), ]), ]; @@ -583,8 +594,8 @@ public function provideTypelessParamTagsData(): Iterator true, '$foo', 'description', - true - ) + true, + ), ), ]), ]; @@ -599,8 +610,149 @@ public function provideTypelessParamTagsData(): Iterator false, '$foo', '', - false - ) + false, + ), + ), + ]), + ]; + } + + public function provideParamImmediatelyInvokedCallableTagsData(): Iterator + { + yield [ + 'OK', + '/** @param-immediately-invoked-callable $foo */', + new PhpDocNode([ + new PhpDocTagNode( + '@param-immediately-invoked-callable', + new ParamImmediatelyInvokedCallableTagValueNode( + '$foo', + '', + ), + ), + ]), + ]; + + yield [ + 'OK with description', + '/** @param-immediately-invoked-callable $foo test two three */', + new PhpDocNode([ + new PhpDocTagNode( + '@param-immediately-invoked-callable', + new ParamImmediatelyInvokedCallableTagValueNode( + '$foo', + 'test two three', + ), + ), + ]), + ]; + } + + public function provideParamLaterInvokedCallableTagsData(): Iterator + { + yield [ + 'OK', + '/** @param-later-invoked-callable $foo */', + new PhpDocNode([ + new PhpDocTagNode( + '@param-later-invoked-callable', + new ParamLaterInvokedCallableTagValueNode( + '$foo', + '', + ), + ), + ]), + ]; + + yield [ + 'OK with description', + '/** @param-later-invoked-callable $foo test two three */', + new PhpDocNode([ + new PhpDocTagNode( + '@param-later-invoked-callable', + new ParamLaterInvokedCallableTagValueNode( + '$foo', + 'test two three', + ), + ), + ]), + ]; + } + + public function provideParamClosureThisTagsData(): Iterator + { + yield [ + 'OK', + '/** @param-closure-this Foo $a */', + new PhpDocNode([ + new PhpDocTagNode( + '@param-closure-this', + new ParamClosureThisTagValueNode( + new IdentifierTypeNode('Foo'), + '$a', + '', + ), + ), + ]), + ]; + + yield [ + 'OK with prefix', + '/** @phpstan-param-closure-this Foo $a */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-param-closure-this', + new ParamClosureThisTagValueNode( + new IdentifierTypeNode('Foo'), + '$a', + '', + ), + ), + ]), + ]; + + yield [ + 'OK with description', + '/** @param-closure-this Foo $a test */', + new PhpDocNode([ + new PhpDocTagNode( + '@param-closure-this', + new ParamClosureThisTagValueNode( + new IdentifierTypeNode('Foo'), + '$a', + 'test', + ), + ), + ]), + ]; + } + + public function providePureUnlessCallableIsImpureTagsData(): Iterator + { + yield [ + 'OK', + '/** @pure-unless-callable-is-impure $foo */', + new PhpDocNode([ + new PhpDocTagNode( + '@pure-unless-callable-is-impure', + new PureUnlessCallableIsImpureTagValueNode( + '$foo', + '', + ), + ), + ]), + ]; + + yield [ + 'OK with description', + '/** @pure-unless-callable-is-impure $foo test two three */', + new PhpDocNode([ + new PhpDocTagNode( + '@pure-unless-callable-is-impure', + new PureUnlessCallableIsImpureTagValueNode( + '$foo', + 'test two three', + ), ), ]), ]; @@ -617,8 +769,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '', - '' - ) + '', + ), ), ]), ]; @@ -632,8 +784,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '$foo', - '' - ) + '', + ), ), ]), ]; @@ -647,8 +799,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '$foo', - '' - ) + '', + ), ), ]), ]; @@ -662,8 +814,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '', - 'optional description' - ) + 'optional description', + ), ), ]), ]; @@ -676,11 +828,11 @@ public function provideVarTagsData(): Iterator '@var', new VarTagValueNode( new ArrayTypeNode( - new IdentifierTypeNode('callable') + new IdentifierTypeNode('callable'), ), '', - 'function (Configurator $sender, DI\Compiler $compiler); Occurs after the compiler is created' - ) + 'function (Configurator $sender, DI\Compiler $compiler); Occurs after the compiler is created', + ), ), ]), ]; @@ -694,8 +846,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '', - '@inject' - ) + '@inject', + ), ), ]), ]; @@ -709,8 +861,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '', - '(Bar)' - ) + '(Bar)', + ), ), ]), ]; @@ -724,8 +876,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '$foo', - 'optional description' - ) + 'optional description', + ), ), ]), ]; @@ -739,8 +891,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '$this', - '' - ) + '', + ), ), ]), ]; @@ -754,8 +906,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '$this', - '' - ) + '', + ), ), ]), ]; @@ -769,8 +921,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '$this', - 'Testing' - ) + 'Testing', + ), ), ]), ]; @@ -784,8 +936,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '$this', - 'Testing' - ) + 'Testing', + ), ), ]), ]; @@ -799,18 +951,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '$foo', - '#desc' - ) - ), - ]), - new PhpDocNode([ - new PhpDocTagNode( - '@var', - new VarTagValueNode( - new IdentifierTypeNode('Foo'), - '$foo', - '#desc' - ) + '#desc', + ), ), ]), ]; @@ -824,8 +966,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new ConstTypeNode(new ConstFetchNode('self', '*')), '$foo', - 'optional description' - ) + 'optional description', + ), ), ]), ]; @@ -839,8 +981,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '$var', - '' - ) + '', + ), ), ]), ]; @@ -854,8 +996,8 @@ public function provideVarTagsData(): Iterator new VarTagValueNode( new IdentifierTypeNode('Foo'), '', - '' - ) + '', + ), ), ]), ]; @@ -874,9 +1016,9 @@ public function provideVarTagsData(): Iterator 9, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -895,9 +1037,9 @@ public function provideVarTagsData(): Iterator 9, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -916,9 +1058,9 @@ public function provideVarTagsData(): Iterator 14, Lexer::TOKEN_CLOSE_PARENTHESES, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -937,9 +1079,9 @@ public function provideVarTagsData(): Iterator 14, Lexer::TOKEN_CLOSE_PARENTHESES, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -958,9 +1100,9 @@ public function provideVarTagsData(): Iterator 17, Lexer::TOKEN_CLOSE_ANGLE_BRACKET, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -979,9 +1121,9 @@ public function provideVarTagsData(): Iterator 14, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -989,23 +1131,6 @@ public function provideVarTagsData(): Iterator yield [ 'invalid object shape', '/** @psalm-type PARTSTRUCTURE_PARAM = objecttt{attribute:string, value?:string} */', - new PhpDocNode([ - new PhpDocTagNode( - '@psalm-type', - new InvalidTagValueNode( - 'Unexpected token "{", expected \'*/\' at offset 46 on line 1', - new ParserException( - '{', - Lexer::TOKEN_OPEN_CURLY_BRACKET, - 46, - Lexer::TOKEN_CLOSE_PHPDOC, - null, - 1 - ) - ) - ), - ]), - null, new PhpDocNode([ new PhpDocTagNode( '@psalm-type', @@ -1018,10 +1143,10 @@ public function provideVarTagsData(): Iterator 46, Lexer::TOKEN_PHPDOC_EOL, null, - 1 - ) - ) - ) + 1, + ), + ), + ), ), ]), ]; @@ -1039,8 +1164,8 @@ public function providePropertyTagsData(): Iterator new PropertyTagValueNode( new IdentifierTypeNode('Foo'), '$foo', - '' - ) + '', + ), ), ]), ]; @@ -1054,8 +1179,8 @@ public function providePropertyTagsData(): Iterator new PropertyTagValueNode( new IdentifierTypeNode('Foo'), '$foo', - 'optional description' - ) + 'optional description', + ), ), ]), ]; @@ -1074,9 +1199,9 @@ public function providePropertyTagsData(): Iterator 14, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1095,9 +1220,9 @@ public function providePropertyTagsData(): Iterator 14, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1116,9 +1241,9 @@ public function providePropertyTagsData(): Iterator 19, Lexer::TOKEN_CLOSE_PARENTHESES, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1137,9 +1262,9 @@ public function providePropertyTagsData(): Iterator 19, Lexer::TOKEN_CLOSE_PARENTHESES, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1158,9 +1283,9 @@ public function providePropertyTagsData(): Iterator 22, Lexer::TOKEN_CLOSE_ANGLE_BRACKET, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1179,9 +1304,9 @@ public function providePropertyTagsData(): Iterator 19, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1200,9 +1325,9 @@ public function providePropertyTagsData(): Iterator 18, Lexer::TOKEN_VARIABLE, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1221,9 +1346,9 @@ public function providePropertyTagsData(): Iterator 18, Lexer::TOKEN_VARIABLE, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1240,8 +1365,8 @@ public function provideReturnTagsData(): Iterator '@return', new ReturnTagValueNode( new IdentifierTypeNode('Foo'), - '' - ) + '', + ), ), ]), ]; @@ -1254,8 +1379,8 @@ public function provideReturnTagsData(): Iterator '@return', new ReturnTagValueNode( new IdentifierTypeNode('Foo'), - 'optional description' - ) + 'optional description', + ), ), ]), ]; @@ -1268,8 +1393,8 @@ public function provideReturnTagsData(): Iterator '@return', new ReturnTagValueNode( new IdentifierTypeNode('Foo'), - '[Bar]' - ) + '[Bar]', + ), ), ]), ]; @@ -1283,10 +1408,24 @@ public function provideReturnTagsData(): Iterator new ReturnTagValueNode( new OffsetAccessTypeNode( new IdentifierTypeNode('Foo'), - new IdentifierTypeNode('Bar') + new IdentifierTypeNode('Bar'), ), - '' - ) + '', + ), + ), + ]), + ]; + + yield [ + 'OK with HTML description', + '/** @return MongoCollection

Returns a collection object representing the new collection.

*/', + new PhpDocNode([ + new PhpDocTagNode( + '@return', + new ReturnTagValueNode( + new IdentifierTypeNode('MongoCollection'), + '

Returns a collection object representing the new collection.

', + ), ), ]), ]; @@ -1305,9 +1444,9 @@ public function provideReturnTagsData(): Iterator 12, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1326,9 +1465,9 @@ public function provideReturnTagsData(): Iterator 12, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1347,9 +1486,9 @@ public function provideReturnTagsData(): Iterator 18, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1368,9 +1507,9 @@ public function provideReturnTagsData(): Iterator 18, Lexer::TOKEN_OTHER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1389,9 +1528,9 @@ public function provideReturnTagsData(): Iterator 18, Lexer::TOKEN_OTHER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1410,9 +1549,9 @@ public function provideReturnTagsData(): Iterator 24, Lexer::TOKEN_CLOSE_ANGLE_BRACKET, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1433,11 +1572,11 @@ public function provideReturnTagsData(): Iterator ], [ GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ]), - '' - ) + '', + ), ), ]), ]; @@ -1450,8 +1589,8 @@ public function provideReturnTagsData(): Iterator '@return', new ReturnTagValueNode( new ConstTypeNode(new ConstFetchNode('self', '*')), - 'example description' - ) + 'example description', + ), ), ]), ]; @@ -1468,10 +1607,10 @@ public function provideReturnTagsData(): Iterator new IdentifierTypeNode('Bar'), new IdentifierTypeNode('never'), new IdentifierTypeNode('int'), - false + false, ), - '' - ) + '', + ), ), ]), ]; @@ -1488,10 +1627,10 @@ public function provideReturnTagsData(): Iterator new IdentifierTypeNode('Bar'), new IdentifierTypeNode('never'), new IdentifierTypeNode('int'), - true + true, ), - '' - ) + '', + ), ), ]), ]; @@ -1514,12 +1653,12 @@ public function provideReturnTagsData(): Iterator new ConstTypeNode(new ConstFetchNode('self', 'TYPE_INT')), new IdentifierTypeNode('int'), new IdentifierTypeNode('bool'), - false + false, ), - false + false, ), - '' - ) + '', + ), ), ]), ]; @@ -1532,8 +1671,8 @@ public function provideReturnTagsData(): Iterator '@return', new ReturnTagValueNode( new IdentifierTypeNode('Foo'), - 'is not Bar ? never : int' - ) + 'is not Bar ? never : int', + ), ), ]), ]; @@ -1550,10 +1689,10 @@ public function provideReturnTagsData(): Iterator new IdentifierTypeNode('Bar'), new IdentifierTypeNode('never'), new IdentifierTypeNode('int'), - false + false, ), - '' - ) + '', + ), ), ]), ]; @@ -1570,10 +1709,10 @@ public function provideReturnTagsData(): Iterator new IdentifierTypeNode('Bar'), new IdentifierTypeNode('never'), new IdentifierTypeNode('int'), - true + true, ), - '' - ) + '', + ), ), ]), ]; @@ -1596,12 +1735,12 @@ public function provideReturnTagsData(): Iterator new ConstTypeNode(new ConstFetchNode('self', 'TYPE_INT')), new IdentifierTypeNode('int'), new IdentifierTypeNode('bool'), - false + false, ), - false + false, ), - '' - ) + '', + ), ), ]), ]; @@ -1620,9 +1759,9 @@ public function provideReturnTagsData(): Iterator 12, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1642,20 +1781,21 @@ public function provideReturnTagsData(): Iterator false, true, '$u', - false + false, ), new CallableTypeParameterNode( new IdentifierTypeNode('string'), false, false, '', - false + false, ), ], - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), + [], ), - '' - ) + '', + ), ), ]), ]; @@ -1663,15 +1803,6 @@ public function provideReturnTagsData(): Iterator yield [ 'invalid variadic callable', '/** @return \Closure(...int, string): string */', - new PhpDocNode([ - new PhpDocTagNode( - '@return', - new ReturnTagValueNode( - new IdentifierTypeNode('\Closure'), - '(...int, string): string' - ) - ), - ]), new PhpDocNode([ new PhpDocTagNode( '@return', @@ -1683,9 +1814,72 @@ public function provideReturnTagsData(): Iterator 20, Lexer::TOKEN_HORIZONTAL_WS, null, - 1 - ) - ) + 1, + ), + ), + ), + ]), + ]; + + yield [ + 'valid CallableTypeNode without space after "callable"', + '/** @return callable(int, string): void */', + new PhpDocNode([ + new PhpDocTagNode( + '@return', + new ReturnTagValueNode( + new CallableTypeNode(new IdentifierTypeNode('callable'), [ + new CallableTypeParameterNode(new IdentifierTypeNode('int'), false, false, '', false), + new CallableTypeParameterNode(new IdentifierTypeNode('string'), false, false, '', false), + ], new IdentifierTypeNode('void'), []), + '', + ), + ), + ]), + ]; + + yield [ + 'valid CallableTypeNode with space after "callable"', + '/** @return callable (int, string): void */', + new PhpDocNode([ + new PhpDocTagNode( + '@return', + new ReturnTagValueNode( + new CallableTypeNode(new IdentifierTypeNode('callable'), [ + new CallableTypeParameterNode(new IdentifierTypeNode('int'), false, false, '', false), + new CallableTypeParameterNode(new IdentifierTypeNode('string'), false, false, '', false), + ], new IdentifierTypeNode('void'), []), + '', + ), + ), + ]), + ]; + + yield [ + 'valid IdentifierTypeNode with space after "callable" turns the rest to description', + '/** @return callable (int, string) */', + new PhpDocNode([ + new PhpDocTagNode( + '@return', + new ReturnTagValueNode(new IdentifierTypeNode('callable'), '(int, string)'), + ), + ]), + ]; + + yield [ + 'invalid CallableTypeNode without space after "callable"', + '/** @return callable(int, string) */', + new PhpDocNode([ + new PhpDocTagNode( + '@return', + new InvalidTagValueNode('callable(int, string)', new ParserException( + '(', + 4, + 20, + 27, + null, + 1, + )), ), ]), ]; @@ -1702,8 +1896,8 @@ public function provideThrowsTagsData(): Iterator '@throws', new ThrowsTagValueNode( new IdentifierTypeNode('Foo'), - '' - ) + '', + ), ), ]), ]; @@ -1716,8 +1910,8 @@ public function provideThrowsTagsData(): Iterator '@throws', new ThrowsTagValueNode( new IdentifierTypeNode('Foo'), - 'optional description' - ) + 'optional description', + ), ), ]), ]; @@ -1730,8 +1924,8 @@ public function provideThrowsTagsData(): Iterator '@throws', new ThrowsTagValueNode( new IdentifierTypeNode('Foo'), - '[Bar]' - ) + '[Bar]', + ), ), ]), ]; @@ -1750,9 +1944,9 @@ public function provideThrowsTagsData(): Iterator 12, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1771,9 +1965,9 @@ public function provideThrowsTagsData(): Iterator 18, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1789,8 +1983,8 @@ public function provideMixinTagsData(): Iterator '@mixin', new MixinTagValueNode( new IdentifierTypeNode('Foo'), - '' - ) + '', + ), ), ]), ]; @@ -1803,8 +1997,8 @@ public function provideMixinTagsData(): Iterator '@mixin', new MixinTagValueNode( new IdentifierTypeNode('Foo'), - 'optional description' - ) + 'optional description', + ), ), ]), ]; @@ -1817,8 +2011,8 @@ public function provideMixinTagsData(): Iterator '@mixin', new MixinTagValueNode( new IdentifierTypeNode('Foo'), - '[Bar]' - ) + '[Bar]', + ), ), ]), ]; @@ -1837,9 +2031,9 @@ public function provideMixinTagsData(): Iterator 11, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1858,9 +2052,9 @@ public function provideMixinTagsData(): Iterator 17, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -1877,8 +2071,140 @@ public function provideMixinTagsData(): Iterator ], [ GenericTypeNode::VARIANCE_INVARIANT, ]), - '' - ) + '', + ), + ), + ]), + ]; + } + + public function provideRequireExtendsTagsData(): Iterator + { + yield [ + 'OK without description', + '/** @phpstan-require-extends Foo */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-require-extends', + new RequireExtendsTagValueNode( + new IdentifierTypeNode('Foo'), + '', + ), + ), + ]), + ]; + + yield [ + 'OK with description', + '/** @phpstan-require-extends Foo optional description */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-require-extends', + new RequireExtendsTagValueNode( + new IdentifierTypeNode('Foo'), + 'optional description', + ), + ), + ]), + ]; + + yield [ + 'OK with psalm-prefix description', + '/** @psalm-require-extends Foo optional description */', + new PhpDocNode([ + new PhpDocTagNode( + '@psalm-require-extends', + new RequireExtendsTagValueNode( + new IdentifierTypeNode('Foo'), + 'optional description', + ), + ), + ]), + ]; + + yield [ + 'invalid without type and description', + '/** @phpstan-require-extends */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-require-extends', + new InvalidTagValueNode( + '', + new ParserException( + '*/', + Lexer::TOKEN_CLOSE_PHPDOC, + 29, + Lexer::TOKEN_IDENTIFIER, + null, + 1, + ), + ), + ), + ]), + ]; + } + + public function provideRequireImplementsTagsData(): Iterator + { + yield [ + 'OK without description', + '/** @phpstan-require-implements Foo */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-require-implements', + new RequireImplementsTagValueNode( + new IdentifierTypeNode('Foo'), + '', + ), + ), + ]), + ]; + + yield [ + 'OK with description', + '/** @phpstan-require-implements Foo optional description */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-require-implements', + new RequireImplementsTagValueNode( + new IdentifierTypeNode('Foo'), + 'optional description', + ), + ), + ]), + ]; + + yield [ + 'OK with psalm-prefix description', + '/** @psalm-require-implements Foo optional description */', + new PhpDocNode([ + new PhpDocTagNode( + '@psalm-require-implements', + new RequireImplementsTagValueNode( + new IdentifierTypeNode('Foo'), + 'optional description', + ), + ), + ]), + ]; + + yield [ + 'invalid without type and description', + '/** @phpstan-require-implements */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-require-implements', + new InvalidTagValueNode( + '', + new ParserException( + '*/', + Lexer::TOKEN_CLOSE_PHPDOC, + 32, + Lexer::TOKEN_IDENTIFIER, + null, + 1, + ), + ), ), ]), ]; @@ -1892,7 +2218,7 @@ public function provideDeprecatedTagsData(): Iterator new PhpDocNode([ new PhpDocTagNode( '@deprecated', - new DeprecatedTagValueNode('') + new DeprecatedTagValueNode(''), ), ]), ]; @@ -1903,7 +2229,7 @@ public function provideDeprecatedTagsData(): Iterator new PhpDocNode([ new PhpDocTagNode( '@deprecated', - new DeprecatedTagValueNode('text string') + new DeprecatedTagValueNode('text string'), ), ]), ]; @@ -1916,12 +2242,12 @@ public function provideDeprecatedTagsData(): Iterator new PhpDocNode([ new PhpDocTagNode( '@deprecated', - new DeprecatedTagValueNode('text first') + new DeprecatedTagValueNode('text first'), ), new PhpDocTextNode(''), new PhpDocTagNode( '@deprecated', - new DeprecatedTagValueNode('text second') + new DeprecatedTagValueNode('text second'), ), ]), ]; @@ -1934,11 +2260,11 @@ public function provideDeprecatedTagsData(): Iterator new PhpDocNode([ new PhpDocTagNode( '@deprecated', - new DeprecatedTagValueNode('text first') + new DeprecatedTagValueNode('text first'), ), new PhpDocTagNode( '@deprecated', - new DeprecatedTagValueNode('text second') + new DeprecatedTagValueNode('text second'), ), ]), ]; @@ -1955,7 +2281,7 @@ public function provideDeprecatedTagsData(): Iterator new DeprecatedTagValueNode('in Drupal 8.6.0 and will be removed before Drupal 9.0.0. In Drupal 9 there will be no way to set the status and in Drupal 8 this ability has been removed because mb_*() functions are supplied using - Symfony\'s polyfill.') + Symfony\'s polyfill.'), ), ]), ]; @@ -1976,7 +2302,7 @@ public function provideDeprecatedTagsData(): Iterator new PhpDocTextNode(''), new PhpDocTagNode( '@author', - new GenericTagValueNode('Foo Baz ') + new GenericTagValueNode('Foo Baz '), ), new PhpDocTextNode(''), new PhpDocTagNode( @@ -1984,7 +2310,7 @@ public function provideDeprecatedTagsData(): Iterator new DeprecatedTagValueNode('in Drupal 8.6.0 and will be removed before Drupal 9.0.0. In Drupal 9 there will be no way to set the status and in Drupal 8 this ability has been removed because mb_*() functions are supplied using - Symfony\'s polyfill.') + Symfony\'s polyfill.'), ), ]), ]; @@ -2003,8 +2329,9 @@ public function provideMethodTagsData(): Iterator null, 'foo', [], - '' - ) + '', + [], + ), ), ]), ]; @@ -2020,8 +2347,9 @@ public function provideMethodTagsData(): Iterator new IdentifierTypeNode('Foo'), 'foo', [], - '' - ) + '', + [], + ), ), ]), ]; @@ -2037,8 +2365,9 @@ public function provideMethodTagsData(): Iterator new IdentifierTypeNode('static'), 'foo', [], - '' - ) + '', + [], + ), ), ]), ]; @@ -2054,8 +2383,9 @@ public function provideMethodTagsData(): Iterator new IdentifierTypeNode('Foo'), 'foo', [], - '' - ) + '', + [], + ), ), ]), ]; @@ -2071,8 +2401,9 @@ public function provideMethodTagsData(): Iterator new IdentifierTypeNode('static'), 'foo', [], - '' - ) + '', + [], + ), ), ]), ]; @@ -2088,8 +2419,9 @@ public function provideMethodTagsData(): Iterator new IdentifierTypeNode('Foo'), 'foo', [], - 'optional description' - ) + 'optional description', + [], + ), ), ]), ]; @@ -2110,11 +2442,12 @@ public function provideMethodTagsData(): Iterator false, false, '$a', - null + null, ), ], - '' - ) + '', + [], + ), ), ]), ]; @@ -2135,11 +2468,12 @@ public function provideMethodTagsData(): Iterator false, false, '$a', - null + null, ), ], - '' - ) + '', + [], + ), ), ]), ]; @@ -2160,11 +2494,12 @@ public function provideMethodTagsData(): Iterator true, false, '$a', - null + null, ), ], - '' - ) + '', + [], + ), ), ]), ]; @@ -2185,11 +2520,12 @@ public function provideMethodTagsData(): Iterator false, true, '$a', - null + null, ), ], - '' - ) + '', + [], + ), ), ]), ]; @@ -2210,11 +2546,12 @@ public function provideMethodTagsData(): Iterator true, true, '$a', - null + null, ), ], - '' - ) + '', + [], + ), ), ]), ]; @@ -2235,11 +2572,12 @@ public function provideMethodTagsData(): Iterator false, false, '$a', - new ConstExprIntegerNode('123') + new ConstExprIntegerNode('123'), ), ], - '' - ) + '', + [], + ), ), ]), ]; @@ -2260,11 +2598,12 @@ public function provideMethodTagsData(): Iterator true, true, '$a', - new ConstExprIntegerNode('123') + new ConstExprIntegerNode('123'), ), ], - '' - ) + '', + [], + ), ), ]), ]; @@ -2285,25 +2624,26 @@ public function provideMethodTagsData(): Iterator false, false, '$a', - null + null, ), new MethodTagValueParameterNode( null, false, false, '$b', - null + null, ), new MethodTagValueParameterNode( null, false, false, '$c', - null + null, ), ], - '' - ) + '', + [], + ), ), ]), ]; @@ -2322,9 +2662,9 @@ public function provideMethodTagsData(): Iterator 16, Lexer::TOKEN_OPEN_PARENTHESES, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -2343,9 +2683,9 @@ public function provideMethodTagsData(): Iterator 23, Lexer::TOKEN_OPEN_PARENTHESES, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -2364,9 +2704,9 @@ public function provideMethodTagsData(): Iterator 17, Lexer::TOKEN_VARIABLE, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -2392,7 +2732,7 @@ public function provideMethodTagsData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), false, false, @@ -2400,13 +2740,13 @@ public function provideMethodTagsData(): Iterator new ConstExprArrayNode([ new ConstExprArrayItemNode( null, - new ConstExprStringNode('\'a\'') + new ConstExprStringNode('a', ConstExprStringNode::SINGLE_QUOTED), ), new ConstExprArrayItemNode( null, - new ConstExprStringNode('\'b\'') + new ConstExprStringNode('b', ConstExprStringNode::SINGLE_QUOTED), ), - ]) + ]), ), ], '', @@ -2415,10 +2755,10 @@ public function provideMethodTagsData(): Iterator 'T', null, '', - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), - ] - ) + ], + ), ), ]), ]; @@ -2439,21 +2779,21 @@ public function provideMethodTagsData(): Iterator false, false, '$t1', - null + null, ), new MethodTagValueParameterNode( new IdentifierTypeNode('T2'), false, false, '$t2', - null + null, ), new MethodTagValueParameterNode( new IdentifierTypeNode('T3'), false, false, '$t3', - null + null, ), ], '', @@ -2461,8 +2801,29 @@ public function provideMethodTagsData(): Iterator new TemplateTagValueNode('T1', null, ''), new TemplateTagValueNode('T2', new IdentifierTypeNode('Bar'), ''), new TemplateTagValueNode('T3', new IdentifierTypeNode('Baz'), ''), - ] - ) + ], + ), + ), + ]), + ]; + + yield [ + 'OK non-static with return type that starts with static type', + '/** @method static|null foo() */', + new PhpDocNode([ + new PhpDocTagNode( + '@method', + new MethodTagValueNode( + false, + new UnionTypeNode([ + new IdentifierTypeNode('static'), + new IdentifierTypeNode('null'), + ]), + 'foo', + [], + '', + [], + ), ), ]), ]; @@ -2482,7 +2843,7 @@ public function provideSingleLinePhpDocData(): Iterator '/** /**/', new PhpDocNode([ new PhpDocTextNode( - '/*' + '/*', ), ]), ]; @@ -2492,7 +2853,7 @@ public function provideSingleLinePhpDocData(): Iterator '/** text */', new PhpDocNode([ new PhpDocTextNode( - 'text' + 'text', ), ]), ]; @@ -2502,7 +2863,7 @@ public function provideSingleLinePhpDocData(): Iterator '/** text @foo bar */', new PhpDocNode([ new PhpDocTextNode( - 'text @foo bar' + 'text @foo bar', ), ]), ]; @@ -2513,7 +2874,7 @@ public function provideSingleLinePhpDocData(): Iterator new PhpDocNode([ new PhpDocTagNode( '@foo', - new GenericTagValueNode('') + new GenericTagValueNode(''), ), ]), ]; @@ -2524,7 +2885,7 @@ public function provideSingleLinePhpDocData(): Iterator new PhpDocNode([ new PhpDocTagNode( '@foo', - new GenericTagValueNode('lorem ipsum') + new GenericTagValueNode('lorem ipsum'), ), ]), ]; @@ -2535,7 +2896,11 @@ public function provideSingleLinePhpDocData(): Iterator new PhpDocNode([ new PhpDocTagNode( '@foo', - new GenericTagValueNode('lorem @bar ipsum') + new GenericTagValueNode('lorem'), + ), + new PhpDocTagNode( + '@bar', + new GenericTagValueNode('ipsum'), ), ]), ]; @@ -2547,8 +2912,8 @@ public function provideSingleLinePhpDocData(): Iterator new PhpDocTagNode( '@varFoo', new GenericTagValueNode( - '$foo' - ) + '$foo', + ), ), ]), ]; @@ -2563,11 +2928,9 @@ public function provideSingleLinePhpDocData(): Iterator new PhpDocNode([ new PhpDocTagNode( '@example', - new GenericTagValueNode('') - ), - new PhpDocTextNode( - 'entity_managers:' . PHP_EOL . - ' default:' + new GenericTagValueNode(PHP_EOL . + ' entity_managers:' . PHP_EOL . + ' default:'), ), ]), ]; @@ -2584,11 +2947,12 @@ public function provideSingleLinePhpDocData(): Iterator [ new CallableTypeParameterNode(new IdentifierTypeNode('int'), false, false, '', false), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), '', - '' - ) + '', + ), ), ]), ]; @@ -2602,8 +2966,8 @@ public function provideSingleLinePhpDocData(): Iterator new VarTagValueNode( new IdentifierTypeNode('callable'), '', - '(int)' - ) + '(int)', + ), ), ]), ]; @@ -2611,16 +2975,6 @@ public function provideSingleLinePhpDocData(): Iterator yield [ 'callable with incomplete signature without return type', '/** @var callable(int) */', - new PhpDocNode([ - new PhpDocTagNode( - '@var', - new VarTagValueNode( - new IdentifierTypeNode('callable'), - '', - '(int)' - ) - ), - ]), new PhpDocNode([ new PhpDocTagNode( '@var', @@ -2632,20 +2986,20 @@ public function provideSingleLinePhpDocData(): Iterator 17, Lexer::TOKEN_HORIZONTAL_WS, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; } /** - * @return array + * @return iterable> */ - public function provideMultiLinePhpDocData(): array + public function provideMultiLinePhpDocData(): iterable { - return [ + yield from [ [ 'multi-line with two tags', '/** @@ -2659,8 +3013,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('Foo'), false, '$foo', - '1st multi world description' - ) + '1st multi world description', + false, + ), ), new PhpDocTagNode( '@param', @@ -2668,8 +3023,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('Bar'), false, '$bar', - '2nd multi world description' - ) + '2nd multi world description', + false, + ), ), ]), ], @@ -2688,8 +3044,9 @@ public function provideMultiLinePhpDocData(): array false, '$foo', '1st multi world description -some text in the middle' - ) +some text in the middle', + false, + ), ), new PhpDocTagNode( '@param', @@ -2697,8 +3054,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('Bar'), false, '$bar', - '2nd multi world description' - ) + '2nd multi world description', + false, + ), ), ]), ], @@ -2727,26 +3085,29 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('Foo'), false, '$foo', - '1st multi world description with empty lines' - ) + '1st multi world description with empty lines + + +some text in the middle', + false, + ), ), new PhpDocTextNode(''), new PhpDocTextNode(''), - new PhpDocTextNode('some text in the middle'), - new PhpDocTextNode(''), - new PhpDocTextNode(''), new PhpDocTagNode( '@param', new ParamTagValueNode( new IdentifierTypeNode('Bar'), false, '$bar', - '2nd multi world description with empty lines' - ) + '2nd multi world description with empty lines + + +test', + false, + ), ), - new PhpDocTextNode(''), - new PhpDocTextNode(''), - new PhpDocTextNode('test'), + ]), ], [ @@ -2774,8 +3135,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('int'), false, '$foo', - '@param string $bar' - ) + '@param string $bar', + false, + ), ), ]), ], @@ -2833,18 +3195,19 @@ public function provideMultiLinePhpDocData(): array false, false, '$a', - null + null, ), new MethodTagValueParameterNode( new IdentifierTypeNode('int'), false, false, '$b', - null + null, ), ], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -2858,18 +3221,19 @@ public function provideMultiLinePhpDocData(): array false, false, '$a', - null + null, ), new MethodTagValueParameterNode( null, false, false, '$b', - null + null, ), ], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -2881,8 +3245,9 @@ public function provideMultiLinePhpDocData(): array ]), 'getFooOrBar', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -2891,8 +3256,9 @@ public function provideMultiLinePhpDocData(): array null, 'methodWithNoReturnType', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -2906,18 +3272,19 @@ public function provideMultiLinePhpDocData(): array false, false, '$a', - null + null, ), new MethodTagValueParameterNode( new IdentifierTypeNode('int'), false, false, '$b', - null + null, ), ], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -2931,18 +3298,19 @@ public function provideMultiLinePhpDocData(): array false, false, '$a', - null + null, ), new MethodTagValueParameterNode( null, false, false, '$b', - null + null, ), ], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -2954,8 +3322,9 @@ public function provideMultiLinePhpDocData(): array ]), 'getFooOrBarStatically', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -2964,8 +3333,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('static'), 'methodWithNoReturnTypeStatically', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -2979,18 +3349,19 @@ public function provideMultiLinePhpDocData(): array false, false, '$a', - null + null, ), new MethodTagValueParameterNode( new IdentifierTypeNode('int'), false, false, '$b', - null + null, ), ], - 'Get an integer with a description.' - ) + 'Get an integer with a description.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3004,18 +3375,19 @@ public function provideMultiLinePhpDocData(): array false, false, '$a', - null + null, ), new MethodTagValueParameterNode( null, false, false, '$b', - null + null, ), ], - 'Do something with a description.' - ) + 'Do something with a description.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3027,8 +3399,9 @@ public function provideMultiLinePhpDocData(): array ]), 'getFooOrBarWithDescription', [], - 'Get a Foo or a Bar with a description.' - ) + 'Get a Foo or a Bar with a description.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3037,8 +3410,9 @@ public function provideMultiLinePhpDocData(): array null, 'methodWithNoReturnTypeWithDescription', [], - 'Do something with a description but what, who knows!' - ) + 'Do something with a description but what, who knows!', + [], + ), ), new PhpDocTagNode( '@method', @@ -3052,18 +3426,19 @@ public function provideMultiLinePhpDocData(): array false, false, '$a', - null + null, ), new MethodTagValueParameterNode( new IdentifierTypeNode('int'), false, false, '$b', - null + null, ), ], - 'Get an integer with a description statically.' - ) + 'Get an integer with a description statically.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3077,18 +3452,19 @@ public function provideMultiLinePhpDocData(): array false, false, '$a', - null + null, ), new MethodTagValueParameterNode( null, false, false, '$b', - null + null, ), ], - 'Do something with a description statically.' - ) + 'Do something with a description statically.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3100,8 +3476,9 @@ public function provideMultiLinePhpDocData(): array ]), 'getFooOrBarStaticallyWithDescription', [], - 'Get a Foo or a Bar with a description statically.' - ) + 'Get a Foo or a Bar with a description statically.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3110,8 +3487,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('static'), 'methodWithNoReturnTypeStaticallyWithDescription', [], - 'Do something with a description statically, but what, who knows!' - ) + 'Do something with a description statically, but what, who knows!', + [], + ), ), new PhpDocTagNode( '@method', @@ -3120,8 +3498,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('bool'), 'aStaticMethodThatHasAUniqueReturnTypeInThisClass', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3130,8 +3509,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('string'), 'aStaticMethodThatHasAUniqueReturnTypeInThisClassWithDescription', [], - 'A Description.' - ) + 'A Description.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3140,8 +3520,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('int'), 'getIntegerNoParams', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3150,8 +3531,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('void'), 'doSomethingNoParams', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3163,8 +3545,9 @@ public function provideMultiLinePhpDocData(): array ]), 'getFooOrBarNoParams', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3173,8 +3556,9 @@ public function provideMultiLinePhpDocData(): array null, 'methodWithNoReturnTypeNoParams', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3183,8 +3567,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('int'), 'getIntegerStaticallyNoParams', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3193,8 +3578,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('void'), 'doSomethingStaticallyNoParams', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3206,8 +3592,9 @@ public function provideMultiLinePhpDocData(): array ]), 'getFooOrBarStaticallyNoParams', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3216,8 +3603,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('static'), 'methodWithNoReturnTypeStaticallyNoParams', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3226,8 +3614,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('int'), 'getIntegerWithDescriptionNoParams', [], - 'Get an integer with a description.' - ) + 'Get an integer with a description.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3236,8 +3625,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('void'), 'doSomethingWithDescriptionNoParams', [], - 'Do something with a description.' - ) + 'Do something with a description.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3249,8 +3639,9 @@ public function provideMultiLinePhpDocData(): array ]), 'getFooOrBarWithDescriptionNoParams', [], - 'Get a Foo or a Bar with a description.' - ) + 'Get a Foo or a Bar with a description.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3259,8 +3650,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('int'), 'getIntegerStaticallyWithDescriptionNoParams', [], - 'Get an integer with a description statically.' - ) + 'Get an integer with a description statically.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3269,8 +3661,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('void'), 'doSomethingStaticallyWithDescriptionNoParams', [], - 'Do something with a description statically.' - ) + 'Do something with a description statically.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3282,8 +3675,9 @@ public function provideMultiLinePhpDocData(): array ]), 'getFooOrBarStaticallyWithDescriptionNoParams', [], - 'Get a Foo or a Bar with a description statically.' - ) + 'Get a Foo or a Bar with a description statically.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3295,8 +3689,9 @@ public function provideMultiLinePhpDocData(): array ]), 'aStaticMethodThatHasAUniqueReturnTypeInThisClassNoParams', [], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3308,8 +3703,9 @@ public function provideMultiLinePhpDocData(): array ]), 'aStaticMethodThatHasAUniqueReturnTypeInThisClassWithDescriptionNoParams', [], - 'A Description.' - ) + 'A Description.', + [], + ), ), new PhpDocTagNode( '@method', @@ -3323,11 +3719,12 @@ public function provideMultiLinePhpDocData(): array false, false, '$args', - null + null, ), ], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3341,18 +3738,19 @@ public function provideMultiLinePhpDocData(): array true, true, '$angle', - new ConstExprArrayNode([]) + new ConstExprArrayNode([]), ), new MethodTagValueParameterNode( null, false, false, '$backgroundColor', - null + null, ), ], - '' - ) + '', + [], + ), ), new PhpDocTagNode( '@method', @@ -3361,8 +3759,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('Foo'), 'overridenMethod', [], - '' - ) + '', + [], + ), ), ]), ], @@ -3376,11 +3775,11 @@ public function provideMultiLinePhpDocData(): array new PhpDocNode([ new PhpDocTagNode( '@template', - new TemplateTagValueNode('TKey', new IdentifierTypeNode('array-key'), '') + new TemplateTagValueNode('TKey', new IdentifierTypeNode('array-key'), ''), ), new PhpDocTagNode( '@template', - new TemplateTagValueNode('TValue', null, '') + new TemplateTagValueNode('TValue', null, ''), ), new PhpDocTagNode( '@method', @@ -3397,11 +3796,12 @@ public function provideMultiLinePhpDocData(): array false, false, '$v', - null + null, ), ], - 'find index of $v' - ) + 'find index of $v', + [], + ), ), ]), ], @@ -3423,11 +3823,11 @@ public function provideMultiLinePhpDocData(): array new PhpDocNode([ new PhpDocTagNode( '@template', - new TemplateTagValueNode('TRandKey', new IdentifierTypeNode('array-key'), '') + new TemplateTagValueNode('TRandKey', new IdentifierTypeNode('array-key'), ''), ), new PhpDocTagNode( '@template', - new TemplateTagValueNode('TRandVal', null, '') + new TemplateTagValueNode('TRandVal', null, ''), ), new PhpDocTagNode( '@template', @@ -3443,7 +3843,7 @@ public function provideMultiLinePhpDocData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new GenericTypeNode( new IdentifierTypeNode('XIterator'), @@ -3454,7 +3854,7 @@ public function provideMultiLinePhpDocData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new GenericTypeNode( new IdentifierTypeNode('Traversable'), @@ -3465,11 +3865,11 @@ public function provideMultiLinePhpDocData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ]), - '' - ) + '', + ), ), new PhpDocTextNode(''), new PhpDocTagNode( @@ -3478,8 +3878,9 @@ public function provideMultiLinePhpDocData(): array new IdentifierTypeNode('TRandList'), false, '$list', - '' - ) + '', + false, + ), ), new PhpDocTextNode(''), new PhpDocTagNode( @@ -3497,7 +3898,7 @@ public function provideMultiLinePhpDocData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new ConditionalTypeNode( new IdentifierTypeNode('TRandList'), @@ -3511,7 +3912,7 @@ public function provideMultiLinePhpDocData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new UnionTypeNode([ new GenericTypeNode( @@ -3523,7 +3924,7 @@ public function provideMultiLinePhpDocData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new GenericTypeNode( new IdentifierTypeNode('LimitIterator'), @@ -3534,229 +3935,723 @@ public function provideMultiLinePhpDocData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ]), - false + false, ), - false + false, ), - '' - ) + '', + ), ), ]), ], ]; - } - public function provideTemplateTagsData(): Iterator - { yield [ - 'OK without bound and description', - '/** @template T */', + 'Empty lines before end', + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' */', new PhpDocNode([ - new PhpDocTagNode( - '@template', - new TemplateTagValueNode( - 'T', - null, - '' - ) - ), + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', '', false)), + new PhpDocTextNode(''), + new PhpDocTextNode(''), ]), ]; yield [ - 'OK without bound', - '/** @template T the value type*/', + 'Empty lines before end 2', + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' * test' . PHP_EOL . + ' */', new PhpDocNode([ + new PhpDocTextNode('Real description'), new PhpDocTagNode( - '@template', - new TemplateTagValueNode( - 'T', - null, - 'the value type' - ) + '@param', + new ParamTagValueNode( + new IdentifierTypeNode('int'), + false, + '$a', + PHP_EOL + . PHP_EOL + . PHP_EOL + . 'test', + false, + ), ), ]), ]; yield [ - 'OK without description', - '/** @template T of DateTime */', + 'Real-world test case multiline PHPDoc', + '/**' . PHP_EOL . + ' *' . PHP_EOL . + ' * MultiLine' . PHP_EOL . + ' * description' . PHP_EOL . + ' * @param bool $a' . PHP_EOL . + ' *' . PHP_EOL . + ' * @return void' . PHP_EOL . + ' *' . PHP_EOL . + ' * @throws \Exception' . PHP_EOL . + ' *' . PHP_EOL . + ' */', new PhpDocNode([ - new PhpDocTagNode( - '@template', - new TemplateTagValueNode( - 'T', - new IdentifierTypeNode('DateTime'), - '' - ) + new PhpDocTextNode( + PHP_EOL . + 'MultiLine' . PHP_EOL . + 'description', ), + new PhpDocTagNode('@param', new ParamTagValueNode( + new IdentifierTypeNode('bool'), + false, + '$a', + '', + false, + )), + new PhpDocTextNode(''), + new PhpDocTagNode('@return', new ReturnTagValueNode( + new IdentifierTypeNode('void'), + '', + )), + new PhpDocTextNode(''), + new PhpDocTagNode('@throws', new ThrowsTagValueNode( + new IdentifierTypeNode('\Exception'), + '', + )), + new PhpDocTextNode(''), ]), ]; yield [ - 'OK without description', - '/** @template T as DateTime */', + 'Multiline PHPDoc with new line across generic type declaration', + '/**' . PHP_EOL . + ' * @param array,' . PHP_EOL . + ' * }> $a' . PHP_EOL . + ' */', new PhpDocNode([ - new PhpDocTagNode( - '@template', - new TemplateTagValueNode( - 'T', - new IdentifierTypeNode('DateTime'), - '' - ) - ), + new PhpDocTagNode('@param', new ParamTagValueNode( + new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('string'), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar'), true, new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('string'), + new UnionTypeNode([ + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar1'), true, new IdentifierTypeNode('true')), + ]), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('foo2'), false, new IdentifierTypeNode('true')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar1'), true, new IdentifierTypeNode('true')), + ]), + ]), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], + )), + ]), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], + ), + false, + '$a', + '', + false, + )), ]), ]; yield [ - 'OK with bound and description', - '/** @template T of DateTime the value type */', + 'Multiline PHPDoc with new line within type declaration', + '/**' . PHP_EOL . + ' * @param array,' . PHP_EOL . + ' * }> $a' . PHP_EOL . + ' */', new PhpDocNode([ - new PhpDocTagNode( - '@template', - new TemplateTagValueNode( - 'T', - new IdentifierTypeNode('DateTime'), - 'the value type' - ) - ), + new PhpDocTagNode('@param', new ParamTagValueNode( + new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('string'), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar'), true, new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('string'), + new UnionTypeNode([ + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar1'), true, new IdentifierTypeNode('true')), + ]), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('foo2'), false, new IdentifierTypeNode('true')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar1'), true, new IdentifierTypeNode('true')), + ]), + ]), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], + )), + ]), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], + ), + false, + '$a', + '', + false, + )), ]), ]; yield [ - 'OK with bound and description', - '/** @template T as DateTime the value type */', + 'Multiline PHPDoc with new line within type declaration including usage of braces', + '/**' . PHP_EOL . + ' * @phpstan-type FactoriesConfigurationType = array<' . PHP_EOL . + ' * string,' . PHP_EOL . + ' * (class-string|Factory\FactoryInterface)' . PHP_EOL . + ' * |callable(ContainerInterface,?string,array|null):object' . PHP_EOL . + ' * >' . PHP_EOL . + ' */', new PhpDocNode([ - new PhpDocTagNode( - '@template', - new TemplateTagValueNode( - 'T', - new IdentifierTypeNode('DateTime'), - 'the value type' - ) - ), + new PhpDocTagNode('@phpstan-type', new TypeAliasTagValueNode( + 'FactoriesConfigurationType', + new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('string'), + new UnionTypeNode([ + new UnionTypeNode([ + new GenericTypeNode( + new IdentifierTypeNode('class-string'), + [new IdentifierTypeNode('Factory\\FactoryInterface')], + [GenericTypeNode::VARIANCE_INVARIANT], + ), + new IdentifierTypeNode('Factory\\FactoryInterface'), + ]), + new CallableTypeNode( + new IdentifierTypeNode('callable'), + [ + new CallableTypeParameterNode(new IdentifierTypeNode('ContainerInterface'), false, false, '', false), + new CallableTypeParameterNode( + new NullableTypeNode( + new IdentifierTypeNode('string'), + ), + false, + false, + '', + false, + ), + new CallableTypeParameterNode( + new UnionTypeNode([ + new GenericTypeNode( + new IdentifierTypeNode('array'), + [new IdentifierTypeNode('mixed')], + [GenericTypeNode::VARIANCE_INVARIANT], + ), + new IdentifierTypeNode('null'), + ]), + false, + false, + '', + false, + ), + ], + new IdentifierTypeNode('object'), + [], + ), + ]), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], + ), + )), ]), ]; yield [ - 'invalid without bound and description', - '/** @template */', + 'Multiline PHPDoc with multiple new line within union type declaration', + '/**' . PHP_EOL . + ' * @param array,' . PHP_EOL . + ' * }> $a' . PHP_EOL . + ' */', new PhpDocNode([ - new PhpDocTagNode( - '@template', - new InvalidTagValueNode( - '', - new ParserException( - '*/', - Lexer::TOKEN_CLOSE_PHPDOC, - 14, - Lexer::TOKEN_IDENTIFIER, - null, - 1 - ) - ) - ), + new PhpDocTagNode('@param', new ParamTagValueNode( + new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('string'), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar'), true, new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('string'), + new UnionTypeNode([ + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar1'), true, new IdentifierTypeNode('true')), + ]), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('foo2'), false, new IdentifierTypeNode('true')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar1'), true, new IdentifierTypeNode('true')), + ]), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo2'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('foo3'), false, new IdentifierTypeNode('bool')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar2'), true, new IdentifierTypeNode('true')), + ]), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('foo3'), false, new IdentifierTypeNode('true')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar3'), true, new IdentifierTypeNode('false')), + ]), + ]), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], + )), + ]), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], + ), + false, + '$a', + '', + false, + )), ]), ]; yield [ - 'invalid without bound and with description', - '/** @template #desc */', + 'Multiline PHPDoc with multiple new line within intersection type declaration', + '/**' . PHP_EOL . + ' * @param array,' . PHP_EOL . + ' * }> $a' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@param', new ParamTagValueNode( + new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('string'), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar'), true, new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('string'), + new IntersectionTypeNode([ + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar1'), true, new IdentifierTypeNode('true')), + ]), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('foo2'), false, new IdentifierTypeNode('true')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar1'), true, new IdentifierTypeNode('true')), + ]), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo2'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('foo3'), false, new IdentifierTypeNode('bool')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar2'), true, new IdentifierTypeNode('true')), + ]), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(new IdentifierTypeNode('foo1'), false, new IdentifierTypeNode('int')), + new ArrayShapeItemNode(new IdentifierTypeNode('foo3'), false, new IdentifierTypeNode('true')), + new ArrayShapeItemNode(new IdentifierTypeNode('bar3'), true, new IdentifierTypeNode('false')), + ]), + ]), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], + )), + ]), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], + ), + false, + '$a', + '', + false, + )), + ]), + ]; + + yield [ + 'Multiline PHPDoc with multiple new line being invalid due to union and intersection type declaration', + '/**' . PHP_EOL . + ' * @param array,' . PHP_EOL . + ' * }> $a' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@param', new InvalidTagValueNode( + 'array,' . PHP_EOL . + '}> $a', + new ParserException( + '?', + Lexer::TOKEN_NULLABLE, + DIRECTORY_SEPARATOR === '\\' ? 65 : 62, + Lexer::TOKEN_CLOSE_CURLY_BRACKET, + null, + 4, + ), + )), + ]), + ]; + + /** + * @return object{ + * a: int, + * + * b: int, + * } + */ + + yield [ + 'Multiline PHPDoc with new line within object type declaration', + '/**' . PHP_EOL . + ' * @return object{' . PHP_EOL . + ' * a: int,' . PHP_EOL . + ' *' . PHP_EOL . + ' * b: int,' . PHP_EOL . + ' * }' . PHP_EOL . + ' */', new PhpDocNode([ new PhpDocTagNode( - '@template', - new InvalidTagValueNode( - '#desc', - new ParserException( - '#desc', - Lexer::TOKEN_OTHER, - 14, - Lexer::TOKEN_IDENTIFIER, - null, - 1 - ) - ) + '@return', + new ReturnTagValueNode( + new ObjectShapeNode( + [ + new ObjectShapeItemNode( + new IdentifierTypeNode('a'), + false, + new IdentifierTypeNode('int'), + ), + new ObjectShapeItemNode( + new IdentifierTypeNode('b'), + false, + new IdentifierTypeNode('int'), + ), + ], + ), + '', + ), ), ]), ]; + } + public function provideTemplateTagsData(): Iterator + { yield [ - 'OK with covariance', - '/** @template-covariant T */', + 'OK without bound and description', + '/** @template T */', new PhpDocNode([ new PhpDocTagNode( - '@template-covariant', + '@template', new TemplateTagValueNode( 'T', null, - '' - ) + '', + ), ), ]), ]; yield [ - 'OK with contravariance', - '/** @template-contravariant T */', + 'OK without bound', + '/** @template T the value type*/', new PhpDocNode([ new PhpDocTagNode( - '@template-contravariant', + '@template', new TemplateTagValueNode( 'T', null, - '' - ) + 'the value type', + ), ), ]), ]; yield [ - 'OK with default', - '/** @template T = string */', + 'OK without description', + '/** @template T of DateTime */', new PhpDocNode([ new PhpDocTagNode( '@template', new TemplateTagValueNode( 'T', - null, + new IdentifierTypeNode('DateTime'), '', - new IdentifierTypeNode('string') - ) + ), ), ]), ]; yield [ - 'OK with default and description', - '/** @template T = string the value type */', + 'OK without description', + '/** @template T as DateTime */', new PhpDocNode([ new PhpDocTagNode( '@template', new TemplateTagValueNode( 'T', - null, - 'the value type', - new IdentifierTypeNode('string') - ) + new IdentifierTypeNode('DateTime'), + '', + ), ), ]), ]; yield [ - 'OK with bound and default and description', - '/** @template T of string = \'\' the value type */', + 'OK with upper bound and description', + '/** @template T of DateTime the value type */', new PhpDocNode([ new PhpDocTagNode( '@template', new TemplateTagValueNode( 'T', - new IdentifierTypeNode('string'), + new IdentifierTypeNode('DateTime'), 'the value type', - new ConstTypeNode(new ConstExprStringNode('')) - ) + ), + ), + ]), + ]; + + yield [ + 'OK with lower bound and description', + '/** @template T super DateTimeImmutable the value type */', + new PhpDocNode([ + new PhpDocTagNode( + '@template', + new TemplateTagValueNode( + 'T', + null, + 'the value type', + null, + new IdentifierTypeNode('DateTimeImmutable'), + ), + ), + ]), + ]; + + yield [ + 'OK with both bounds and description', + '/** @template T of DateTimeInterface super DateTimeImmutable the value type */', + new PhpDocNode([ + new PhpDocTagNode( + '@template', + new TemplateTagValueNode( + 'T', + new IdentifierTypeNode('DateTimeInterface'), + 'the value type', + null, + new IdentifierTypeNode('DateTimeImmutable'), + ), + ), + ]), + ]; + + yield [ + 'invalid without bounds and description', + '/** @template */', + new PhpDocNode([ + new PhpDocTagNode( + '@template', + new InvalidTagValueNode( + '', + new ParserException( + '*/', + Lexer::TOKEN_CLOSE_PHPDOC, + 14, + Lexer::TOKEN_IDENTIFIER, + null, + 1, + ), + ), + ), + ]), + ]; + + yield [ + 'invalid without bound and with description', + '/** @template #desc */', + new PhpDocNode([ + new PhpDocTagNode( + '@template', + new InvalidTagValueNode( + '#desc', + new ParserException( + '#desc', + Lexer::TOKEN_OTHER, + 14, + Lexer::TOKEN_IDENTIFIER, + null, + 1, + ), + ), + ), + ]), + ]; + + yield [ + 'OK with covariance', + '/** @template-covariant T */', + new PhpDocNode([ + new PhpDocTagNode( + '@template-covariant', + new TemplateTagValueNode( + 'T', + null, + '', + ), + ), + ]), + ]; + + yield [ + 'OK with contravariance', + '/** @template-contravariant T */', + new PhpDocNode([ + new PhpDocTagNode( + '@template-contravariant', + new TemplateTagValueNode( + 'T', + null, + '', + ), + ), + ]), + ]; + + yield [ + 'OK with default', + '/** @template T = string */', + new PhpDocNode([ + new PhpDocTagNode( + '@template', + new TemplateTagValueNode( + 'T', + null, + '', + new IdentifierTypeNode('string'), + ), + ), + ]), + ]; + + yield [ + 'OK with default and description', + '/** @template T = string the value type */', + new PhpDocNode([ + new PhpDocTagNode( + '@template', + new TemplateTagValueNode( + 'T', + null, + 'the value type', + new IdentifierTypeNode('string'), + ), + ), + ]), + ]; + + yield [ + 'OK with bound and default and description', + '/** @template T of string = \'\' the value type */', + new PhpDocNode([ + new PhpDocTagNode( + '@template', + new TemplateTagValueNode( + 'T', + new IdentifierTypeNode('string'), + 'the value type', + new ConstTypeNode(new ConstExprStringNode('', ConstExprStringNode::SINGLE_QUOTED)), + ), ), ]), ]; @@ -3778,10 +4673,10 @@ public function provideExtendsTagsData(): Iterator ], [ GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), - '' - ) + '', + ), ), ]), ]; @@ -3802,10 +4697,10 @@ public function provideExtendsTagsData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), - '' - ) + '', + ), ), ]), ]; @@ -3826,10 +4721,10 @@ public function provideExtendsTagsData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), - '' - ) + '', + ), ), ]), ]; @@ -3850,10 +4745,10 @@ public function provideExtendsTagsData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), - '' - ) + '', + ), ), ]), ]; @@ -3868,10 +4763,10 @@ public function provideExtendsTagsData(): Iterator new GenericTypeNode( new IdentifierTypeNode('Foo'), [new IdentifierTypeNode('A')], - [GenericTypeNode::VARIANCE_INVARIANT] + [GenericTypeNode::VARIANCE_INVARIANT], ), - 'extends foo' - ) + 'extends foo', + ), ), ]), ]; @@ -3890,9 +4785,9 @@ public function provideExtendsTagsData(): Iterator 13, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -3911,9 +4806,9 @@ public function provideExtendsTagsData(): Iterator 17, Lexer::TOKEN_OPEN_ANGLE_BRACKET, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -3926,8 +4821,8 @@ public function provideExtendsTagsData(): Iterator '@return', new ReturnTagValueNode( new IdentifierTypeNode('class-string'), - '' - ) + '', + ), ), ]), ]; @@ -3940,8 +4835,8 @@ public function provideExtendsTagsData(): Iterator '@return', new ReturnTagValueNode( new IdentifierTypeNode('class-string'), - 'some description' - ) + 'some description', + ), ), ]), ]; @@ -3956,8 +4851,9 @@ public function provideExtendsTagsData(): Iterator new IdentifierTypeNode('class-string'), false, '$test', - '' - ) + '', + false, + ), ), ]), ]; @@ -3972,8 +4868,9 @@ public function provideExtendsTagsData(): Iterator new IdentifierTypeNode('class-string'), false, '$test', - 'some description' - ) + 'some description', + false, + ), ), ]), ]; @@ -3992,8 +4889,8 @@ public function provideTypeAliasTagsData(): Iterator new UnionTypeNode([ new IdentifierTypeNode('string'), new IdentifierTypeNode('int'), - ]) - ) + ]), + ), ), ]), ]; @@ -4009,8 +4906,8 @@ public function provideTypeAliasTagsData(): Iterator new UnionTypeNode([ new IdentifierTypeNode('string'), new IdentifierTypeNode('int'), - ]) - ) + ]), + ), ), ]), ]; @@ -4018,23 +4915,6 @@ public function provideTypeAliasTagsData(): Iterator yield [ 'invalid without type', '/** @phpstan-type TypeAlias */', - new PhpDocNode([ - new PhpDocTagNode( - '@phpstan-type', - new InvalidTagValueNode( - 'TypeAlias', - new ParserException( - '*/', - Lexer::TOKEN_CLOSE_PHPDOC, - 28, - Lexer::TOKEN_IDENTIFIER, - null, - 1 - ) - ) - ), - ]), - null, new PhpDocNode([ new PhpDocTagNode( '@phpstan-type', @@ -4046,9 +4926,9 @@ public function provideTypeAliasTagsData(): Iterator 28, Lexer::TOKEN_IDENTIFIER, null, - 1 - )) - ) + 1, + )), + ), ), ]), ]; @@ -4058,23 +4938,6 @@ public function provideTypeAliasTagsData(): Iterator '/** * @phpstan-type TypeAlias */', - new PhpDocNode([ - new PhpDocTagNode( - '@phpstan-type', - new InvalidTagValueNode( - 'TypeAlias', - new ParserException( - "\n\t\t\t ", - Lexer::TOKEN_PHPDOC_EOL, - 34, - Lexer::TOKEN_IDENTIFIER, - null, - 2 - ) - ) - ), - ]), - null, new PhpDocNode([ new PhpDocTagNode( '@phpstan-type', @@ -4086,9 +4949,9 @@ public function provideTypeAliasTagsData(): Iterator 34, Lexer::TOKEN_IDENTIFIER, null, - 2 - )) - ) + 2, + )), + ), ), ]), ]; @@ -4099,30 +4962,6 @@ public function provideTypeAliasTagsData(): Iterator * @phpstan-type TypeAlias * @mixin T */', - new PhpDocNode([ - new PhpDocTagNode( - '@phpstan-type', - new InvalidTagValueNode( - 'TypeAlias', - new ParserException( - "\n\t\t\t * ", - Lexer::TOKEN_PHPDOC_EOL, - 34, - Lexer::TOKEN_IDENTIFIER, - null, - 2 - ) - ) - ), - new PhpDocTagNode( - '@mixin', - new MixinTagValueNode( - new IdentifierTypeNode('T'), - '' - ) - ), - ]), - null, new PhpDocNode([ new PhpDocTagNode( '@phpstan-type', @@ -4134,16 +4973,16 @@ public function provideTypeAliasTagsData(): Iterator 34, Lexer::TOKEN_IDENTIFIER, null, - 2 - )) - ) + 2, + )), + ), ), new PhpDocTagNode( '@mixin', new MixinTagValueNode( new IdentifierTypeNode('T'), - '' - ) + '', + ), ), ]), ]; @@ -4154,30 +4993,13 @@ public function provideTypeAliasTagsData(): Iterator * @phpstan-type Foo array{} * @phpstan-type InvalidFoo what{} */', - new PhpDocNode([ - new PhpDocTagNode( - '@phpstan-type', - new InvalidTagValueNode( - "Unexpected token \"{\", expected '*/' at offset 65 on line 3", - new ParserException( - '{', - Lexer::TOKEN_OPEN_CURLY_BRACKET, - 65, - Lexer::TOKEN_CLOSE_PHPDOC, - null, - 3 - ) - ) - ), - ]), - null, new PhpDocNode([ new PhpDocTagNode( '@phpstan-type', new TypeAliasTagValueNode( 'Foo', - new ArrayShapeNode([]) - ) + ArrayShapeNode::createSealed([]), + ), ), new PhpDocTagNode( '@phpstan-type', @@ -4189,9 +5011,9 @@ public function provideTypeAliasTagsData(): Iterator 65, Lexer::TOKEN_PHPDOC_EOL, null, - 3 - )) - ) + 3, + )), + ), ), ]), ]; @@ -4203,30 +5025,13 @@ public function provideTypeAliasTagsData(): Iterator * @phpstan-type InvalidFoo what{} * @phpstan-type Bar array{} */', - new PhpDocNode([ - new PhpDocTagNode( - '@phpstan-type', - new InvalidTagValueNode( - "Unexpected token \"{\", expected '*/' at offset 65 on line 3", - new ParserException( - '{', - Lexer::TOKEN_OPEN_CURLY_BRACKET, - 65, - Lexer::TOKEN_CLOSE_PHPDOC, - null, - 3 - ) - ) - ), - ]), - null, new PhpDocNode([ new PhpDocTagNode( '@phpstan-type', new TypeAliasTagValueNode( 'Foo', - new ArrayShapeNode([]) - ) + ArrayShapeNode::createSealed([]), + ), ), new PhpDocTagNode( '@phpstan-type', @@ -4238,16 +5043,16 @@ public function provideTypeAliasTagsData(): Iterator 65, Lexer::TOKEN_PHPDOC_EOL, null, - 3 - )) - ) + 3, + )), + ), ), new PhpDocTagNode( '@phpstan-type', new TypeAliasTagValueNode( 'Bar', - new ArrayShapeNode([]) - ) + ArrayShapeNode::createSealed([]), + ), ), ]), ]; @@ -4266,9 +5071,9 @@ public function provideTypeAliasTagsData(): Iterator 18, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -4285,8 +5090,8 @@ public function provideTypeAliasImportTagsData(): Iterator new TypeAliasImportTagValueNode( 'TypeAlias', new IdentifierTypeNode('AnotherClass'), - null - ) + null, + ), ), ]), ]; @@ -4300,8 +5105,8 @@ public function provideTypeAliasImportTagsData(): Iterator new TypeAliasImportTagValueNode( 'TypeAlias', new IdentifierTypeNode('AnotherClass'), - 'DifferentAlias' - ) + 'DifferentAlias', + ), ), ]), ]; @@ -4320,10 +5125,10 @@ public function provideTypeAliasImportTagsData(): Iterator 40, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) - ), + 1, + ), + ), + ), ]), ]; @@ -4341,9 +5146,9 @@ public function provideTypeAliasImportTagsData(): Iterator 52, Lexer::TOKEN_CLOSE_PHPDOC, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -4362,9 +5167,9 @@ public function provideTypeAliasImportTagsData(): Iterator 35, Lexer::TOKEN_IDENTIFIER, 'from', - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -4383,9 +5188,9 @@ public function provideTypeAliasImportTagsData(): Iterator 35, Lexer::TOKEN_IDENTIFIER, 'from', - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -4404,9 +5209,9 @@ public function provideTypeAliasImportTagsData(): Iterator 25, Lexer::TOKEN_IDENTIFIER, null, - 1 - ) - ) + 1, + ), + ), ), ]), ]; @@ -4424,8 +5229,9 @@ public function provideAssertTagsData(): Iterator new IdentifierTypeNode('Type'), '$var', false, - '' - ) + '', + false, + ), ), ]), ]; @@ -4440,8 +5246,9 @@ public function provideAssertTagsData(): Iterator new IdentifierTypeNode('Type'), '$var', false, - '' - ) + '', + false, + ), ), ]), ]; @@ -4456,8 +5263,9 @@ public function provideAssertTagsData(): Iterator new IdentifierTypeNode('Type'), '$var', false, - 'assert Type to $var' - ) + 'assert Type to $var', + false, + ), ), ]), ]; @@ -4475,8 +5283,9 @@ public function provideAssertTagsData(): Iterator ]), '$var', false, - '' - ) + '', + false, + ), ), ]), ]; @@ -4492,8 +5301,9 @@ public function provideAssertTagsData(): Iterator '$var', 'method', false, - '' - ) + '', + false, + ), ), ]), ]; @@ -4509,29 +5319,68 @@ public function provideAssertTagsData(): Iterator '$var', 'property', false, - '' - ) + '', + false, + ), ), ]), ]; yield [ - 'invalid $this', + 'OK $this', '/** @phpstan-assert Type $this */', new PhpDocNode([ new PhpDocTagNode( '@phpstan-assert', - new InvalidTagValueNode( - 'Type $this', - new ParserException( - '*/', - Lexer::TOKEN_CLOSE_PHPDOC, - 31, - Lexer::TOKEN_ARROW, - null, - 1 - ) - ) + new AssertTagValueNode( + new IdentifierTypeNode('Type'), + '$this', + false, + '', + false, + ), + ), + ]), + ]; + + yield [ + 'OK $this with description', + '/** @phpstan-assert Type $this assert Type to $this */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-assert', + new AssertTagValueNode( + new IdentifierTypeNode('Type'), + '$this', + false, + 'assert Type to $this', + false, + ), + ), + ]), + ]; + + yield [ + 'OK $this with generic type', + '/** @phpstan-assert GenericType $this */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-assert', + new AssertTagValueNode( + new GenericTypeNode( + new IdentifierTypeNode('GenericType'), + [ + new IdentifierTypeNode('T'), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + ], + ), + '$this', + false, + '', + false, + ), ), ]), ]; @@ -4547,8 +5396,9 @@ public function provideAssertTagsData(): Iterator '$this', 'method', false, - '' - ) + '', + false, + ), ), ]), ]; @@ -4564,8 +5414,9 @@ public function provideAssertTagsData(): Iterator '$this', 'property', false, - '' - ) + '', + false, + ), ), ]), ]; @@ -4580,8 +5431,9 @@ public function provideAssertTagsData(): Iterator new IdentifierTypeNode('Type'), '$var', false, - '' - ) + '', + false, + ), ), ]), ]; @@ -4596,8 +5448,9 @@ public function provideAssertTagsData(): Iterator new IdentifierTypeNode('Type'), '$var', false, - '' - ) + '', + false, + ), ), ]), ]; @@ -4612,8 +5465,9 @@ public function provideAssertTagsData(): Iterator new IdentifierTypeNode('Type'), '$var', true, - '' - ) + '', + false, + ), ), ]), ]; @@ -4629,8 +5483,8 @@ public function provideAssertTagsData(): Iterator '$var', false, '', - true - ) + true, + ), ), ]), ]; @@ -4646,8 +5500,8 @@ public function provideAssertTagsData(): Iterator '$var', true, '', - true - ) + true, + ), ), ]), ]; @@ -4668,13 +5522,13 @@ public function providerDebug(): Iterator 'OK class line', $sample, new PhpDocNode([ - new PhpDocTextNode('Returns the schema for the field.'), - new PhpDocTextNode(''), - new PhpDocTextNode('This method is static because the field schema information is needed on + new PhpDocTextNode('Returns the schema for the field. + +This method is static because the field schema information is needed on creation of the field. FieldItemInterface objects instantiated at that -time are not reliable as field settings might be missing.'), - new PhpDocTextNode(''), - new PhpDocTextNode('Computed fields having no schema should return an empty array.'), +time are not reliable as field settings might be missing. + +Computed fields having no schema should return an empty array.'), ]), ]; } @@ -4722,13 +5576,13 @@ public function provideRealWorldExampleData(): Iterator 'OK FieldItemInterface::schema', $sample, new PhpDocNode([ - new PhpDocTextNode('Returns the schema for the field.'), - new PhpDocTextNode(''), - new PhpDocTextNode('This method is static because the field schema information is needed on + new PhpDocTextNode('Returns the schema for the field. + +This method is static because the field schema information is needed on creation of the field. FieldItemInterface objects instantiated at that -time are not reliable as field settings might be missing.'), - new PhpDocTextNode(''), - new PhpDocTextNode('Computed fields having no schema should return an empty array.'), +time are not reliable as field settings might be missing. + +Computed fields having no schema should return an empty array.'), new PhpDocTextNode(''), new PhpDocTagNode( '@param', @@ -4736,19 +5590,18 @@ public function provideRealWorldExampleData(): Iterator new IdentifierTypeNode('\Drupal\Core\Field\FieldStorageDefinitionInterface'), false, '$field_definition', - '' - ) + ' + The field definition.', + false, + ), ), - new PhpDocTextNode('The field definition.'), new PhpDocTextNode(''), new PhpDocTagNode( '@return', new ReturnTagValueNode( new IdentifierTypeNode('array'), - '' - ) - ), - new PhpDocTextNode("An empty array if there is no schema, or an associative array with the + " + An empty array if there is no schema, or an associative array with the following key/value pairs: - columns: An array of Schema API column specifications, keyed by column name. The columns need to be a subset of the properties defined in @@ -4770,7 +5623,9 @@ public function provideRealWorldExampleData(): Iterator definitions. Note, however, that the field data is not necessarily stored in SQL. Also, the possible usage is limited, as you cannot specify another field as related, only existing SQL tables, - such as {taxonomy_term_data}."), + such as {taxonomy_term_data}.", + ), + ), ]), ]; @@ -4795,9 +5650,9 @@ public function provideRealWorldExampleData(): Iterator 'OK AbstractChunkedController::parseChunkedRequest', $sample, new PhpDocNode([ - new PhpDocTextNode('Parses a chunked request and return relevant information.'), - new PhpDocTextNode(''), - new PhpDocTextNode('This function must return an array containing the following + new PhpDocTextNode('Parses a chunked request and return relevant information. + + This function must return an array containing the following keys and their corresponding values: - last: Wheter this is the last chunk of the uploaded file - uuid: A unique id which distinguishes two uploaded files @@ -4813,16 +5668,17 @@ public function provideRealWorldExampleData(): Iterator new IdentifierTypeNode('Request'), false, '$request', - '- The request object' - ) + '- The request object', + false, + ), ), new PhpDocTextNode(''), new PhpDocTagNode( '@return', new ReturnTagValueNode( new IdentifierTypeNode('array'), - '' - ) + '', + ), ), ]), ]; @@ -4840,9 +5696,9 @@ public function provideRealWorldExampleData(): Iterator * */", new PhpDocNode([ - new PhpDocTextNode('Finder allows searching through directory trees using iterator.'), - new PhpDocTextNode(''), - new PhpDocTextNode(" + new PhpDocTextNode("Finder allows searching through directory trees using iterator. + + Finder::findFiles('*.php') ->size('> 10kB') ->from('.') @@ -4859,11 +5715,11 @@ public function provideRealWorldExampleData(): Iterator '@return', new ReturnTagValueNode( new UnionTypeNode([ - new ConstTypeNode(new ConstExprStringNode('foo')), - new ConstTypeNode(new ConstExprStringNode('bar')), + new ConstTypeNode(new ConstExprStringNode('foo', ConstExprStringNode::SINGLE_QUOTED)), + new ConstTypeNode(new ConstExprStringNode('bar', ConstExprStringNode::SINGLE_QUOTED)), ]), - '' - ) + '', + ), ), ]), ]; @@ -4896,10 +5752,10 @@ public function provideRealWorldExampleData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), - '' - ) + '', + ), ), ]), ]; @@ -4925,10 +5781,10 @@ public function provideRealWorldExampleData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), - '' - ) + '', + ), ), ]), ]; @@ -4954,10 +5810,10 @@ public function provideRealWorldExampleData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), - '' - ) + '', + ), ), ]), ]; @@ -4979,12 +5835,14 @@ public function provideRealWorldExampleData(): Iterator new CallableTypeParameterNode(new IdentifierTypeNode('A'), false, false, '', false), new CallableTypeParameterNode(new IdentifierTypeNode('B'), false, false, '', false), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), false, '$foo', - '' - ) + '', + false, + ), ), ]), ]; @@ -5007,12 +5865,14 @@ public function provideRealWorldExampleData(): Iterator new CallableTypeParameterNode(new IdentifierTypeNode('A'), false, false, '', false), new CallableTypeParameterNode(new IdentifierTypeNode('B'), false, false, '', false), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), false, '$foo', - '' - ) + '', + false, + ), ), ]), ]; @@ -5035,12 +5895,14 @@ public function provideRealWorldExampleData(): Iterator new CallableTypeParameterNode(new IdentifierTypeNode('A'), false, false, '', false), new CallableTypeParameterNode(new IdentifierTypeNode('B'), false, false, '', false), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), false, '$foo', - '' - ) + '', + false, + ), ), ]), ]; @@ -5080,10 +5942,10 @@ public function provideRealWorldExampleData(): Iterator GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), - '' - ) + '', + ), ), new PhpDocTextNode(''), new PhpDocTagNode( @@ -5092,8 +5954,9 @@ public function provideRealWorldExampleData(): Iterator new IdentifierTypeNode('string'), false, '$pattern', - '' - ) + '', + false, + ), ), new PhpDocTagNode( '@param', @@ -5101,8 +5964,9 @@ public function provideRealWorldExampleData(): Iterator new IdentifierTypeNode('string'), false, '$subject', - '' - ) + '', + false, + ), ), new PhpDocTagNode( '@param', @@ -5110,8 +5974,9 @@ public function provideRealWorldExampleData(): Iterator new IdentifierTypeNode('mixed'), false, '$matches', - '' - ) + '', + false, + ), ), new PhpDocTagNode( '@param', @@ -5119,8 +5984,9 @@ public function provideRealWorldExampleData(): Iterator new IdentifierTypeNode('TFlags'), false, '$flags', - '' - ) + '', + false, + ), ), new PhpDocTagNode( '@param-out', @@ -5133,7 +5999,7 @@ public function provideRealWorldExampleData(): Iterator [ new IdentifierTypeNode('array-key'), new UnionTypeNode([ - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode(null, false, new IdentifierTypeNode('string')), new ArrayShapeItemNode( null, @@ -5141,11 +6007,11 @@ public function provideRealWorldExampleData(): Iterator new UnionTypeNode([ new ConstTypeNode(new ConstExprIntegerNode('0')), new IdentifierTypeNode('positive-int'), - ]) + ]), ), ]), - new ArrayShapeNode([ - new ArrayShapeItemNode(null, false, new ConstTypeNode(new ConstExprStringNode(''))), + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode(null, false, new ConstTypeNode(new ConstExprStringNode('', ConstExprStringNode::SINGLE_QUOTED))), new ArrayShapeItemNode(null, false, new ConstTypeNode(new ConstExprIntegerNode('-1'))), ]), ]), @@ -5153,7 +6019,7 @@ public function provideRealWorldExampleData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new ConditionalTypeNode( new IdentifierTypeNode('TFlags'), @@ -5170,7 +6036,7 @@ public function provideRealWorldExampleData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new ConditionalTypeNode( new IdentifierTypeNode('TFlags'), @@ -5180,7 +6046,7 @@ public function provideRealWorldExampleData(): Iterator [ new IdentifierTypeNode('array-key'), new UnionTypeNode([ - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode(null, false, new IdentifierTypeNode('string')), new ArrayShapeItemNode( null, @@ -5188,10 +6054,10 @@ public function provideRealWorldExampleData(): Iterator new UnionTypeNode([ new ConstTypeNode(new ConstExprIntegerNode('0')), new IdentifierTypeNode('positive-int'), - ]) + ]), ), ]), - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode(null, false, new IdentifierTypeNode('null')), new ArrayShapeItemNode(null, false, new ConstTypeNode(new ConstExprIntegerNode('-1'))), ]), @@ -5200,7 +6066,7 @@ public function provideRealWorldExampleData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new GenericTypeNode( new IdentifierTypeNode('array'), @@ -5211,17 +6077,17 @@ public function provideRealWorldExampleData(): Iterator [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), - false + false, ), - false + false, ), - false + false, ), '$matches', - '' - ) + '', + ), ), new PhpDocTagNode( '@return', @@ -5231,280 +6097,1267 @@ public function provideRealWorldExampleData(): Iterator new ConstTypeNode(new ConstExprIntegerNode('0')), new IdentifierTypeNode('false'), ]), - '' - ) + '', + ), ), new PhpDocTagNode('@psalm-ignore-falsable-return', new GenericTagValueNode('')), ]), ]; } - public function provideDescriptionWithOrWithoutHtml(): Iterator + public function provideDescriptionWithOrWithoutHtml(): Iterator + { + yield [ + 'Description with HTML tags in @return tag (close tags together)', + '/**' . PHP_EOL . + ' * @return Foo Important description' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@return', + new ReturnTagValueNode( + new IdentifierTypeNode('Foo'), + 'Important description', + ), + ), + ]), + ]; + + yield [ + 'Description with HTML tags in @throws tag (closed tags with text between)', + '/**' . PHP_EOL . + ' * @throws FooException Important description etc' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@throws', + new ThrowsTagValueNode( + new IdentifierTypeNode('FooException'), + 'Important description etc', + ), + ), + ]), + ]; + + yield [ + 'Description with HTML tags in @mixin tag', + '/**' . PHP_EOL . + ' * @mixin Mixin Important description' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@mixin', + new MixinTagValueNode( + new IdentifierTypeNode('Mixin'), + 'Important description', + ), + ), + ]), + ]; + + yield [ + 'Description with unclosed HTML tags in @return tag - unclosed HTML tag is parsed as generics', + '/**' . PHP_EOL . + ' * @return Foo Important description' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@return', + new InvalidTagValueNode( + 'Foo Important description', + new ParserException( + 'Important', + Lexer::TOKEN_IDENTIFIER, + PHP_EOL === "\n" ? 27 : 28, + Lexer::TOKEN_HORIZONTAL_WS, + null, + 2, + ), + ), + ), + ]), + ]; + } + + /** + * @return array + */ + public function dataParseTagValue(): array + { + return [ + [ + '@param', + 'DateTimeImmutable::ATOM $a', + new ParamTagValueNode( + new ConstTypeNode(new ConstFetchNode('DateTimeImmutable', 'ATOM')), + false, + '$a', + '', + false, + ), + ], + [ + '@var', + '$foo string[]', + new InvalidTagValueNode( + '$foo string[]', + new ParserException( + '$foo', + Lexer::TOKEN_VARIABLE, + 0, + Lexer::TOKEN_IDENTIFIER, + null, + 1, + ), + ), + ], + ]; + } + + public function provideTagsWithNumbers(): Iterator + { + yield [ + 'OK without description and tag with number in it', + '/** @special3 Foo */', + new PhpDocNode([ + new PhpDocTagNode( + '@special3', + new GenericTagValueNode('Foo'), + ), + ]), + ]; + } + + public function provideTagsWithBackslash(): Iterator + { + yield [ + 'OK without description and tag with backslashes in it', + '/** @ORM\Mapping\Entity User */', + new PhpDocNode([ + new PhpDocTagNode( + '@ORM\Mapping\Entity', + new GenericTagValueNode('User'), + ), + ]), + ]; + + yield [ + 'OK without description and tag with backslashes in it and parenthesis', + '/** @ORM\Mapping\JoinColumn(name="column_id", referencedColumnName="id") */', + new PhpDocNode([ + new PhpDocTagNode( + '@ORM\Mapping\JoinColumn', + new DoctrineTagValueNode(new DoctrineAnnotation('@ORM\Mapping\JoinColumn', [ + new DoctrineArgument(new IdentifierTypeNode('name'), new DoctrineConstExprStringNode('column_id')), + new DoctrineArgument(new IdentifierTypeNode('referencedColumnName'), new DoctrineConstExprStringNode('id')), + ]), ''), + ), + ]), + ]; + } + + public function provideSelfOutTagsData(): Iterator + { + yield [ + 'OK phpstan-self-out', + '/** @phpstan-self-out self */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-self-out', + new SelfOutTagValueNode( + new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), + '', + ), + ), + ]), + ]; + + yield [ + 'OK phpstan-this-out', + '/** @phpstan-this-out self */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-this-out', + new SelfOutTagValueNode( + new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), + '', + ), + ), + ]), + ]; + + yield [ + 'OK psalm-self-out', + '/** @psalm-self-out self */', + new PhpDocNode([ + new PhpDocTagNode( + '@psalm-self-out', + new SelfOutTagValueNode( + new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), + '', + ), + ), + ]), + ]; + + yield [ + 'OK psalm-this-out', + '/** @psalm-this-out self */', + new PhpDocNode([ + new PhpDocTagNode( + '@psalm-this-out', + new SelfOutTagValueNode( + new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), + '', + ), + ), + ]), + ]; + + yield [ + 'OK with description', + '/** @phpstan-self-out self description */', + new PhpDocNode([ + new PhpDocTagNode( + '@phpstan-self-out', + new SelfOutTagValueNode( + new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), + 'description', + ), + ), + ]), + ]; + } + + public function provideCommentLikeDescriptions(): Iterator + { + yield [ + 'Comment after @param', + '/** @param int $a // this is a description */', + new PhpDocNode([ + new PhpDocTagNode('@param', new ParamTagValueNode( + new IdentifierTypeNode('int'), + false, + '$a', + '// this is a description', + false, + )), + ]), + ]; + + yield [ + 'Comment after @param with https://', + '/** @param int $a https://phpstan.org/ */', + new PhpDocNode([ + new PhpDocTagNode('@param', new ParamTagValueNode( + new IdentifierTypeNode('int'), + false, + '$a', + 'https://phpstan.org/', + false, + )), + ]), + ]; + + yield [ + 'Comment after @param with https:// in // comment', + '/** @param int $a // comment https://phpstan.org/ */', + new PhpDocNode([ + new PhpDocTagNode('@param', new ParamTagValueNode( + new IdentifierTypeNode('int'), + false, + '$a', + '// comment https://phpstan.org/', + false, + )), + ]), + ]; + + yield [ + 'Comment in PHPDoc tag outside of type', + '/** @param // comment */', + new PhpDocNode([ + new PhpDocTagNode('@param', new InvalidTagValueNode('// comment', new ParserException( + '// comment ', + 37, + 11, + 24, + null, + 1, + ))), + ]), + ]; + + yield [ + 'Comment on a separate line', + '/**' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' * // this is a comment' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@param', new ParamTagValueNode( + new IdentifierTypeNode('int'), + false, + '$a', + PHP_EOL . '// this is a comment', + false, + )), + ]), + ]; + yield [ + 'Comment on a separate line 2', + '/**' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' *' . PHP_EOL . + ' * // this is a comment' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@param', new ParamTagValueNode( + new IdentifierTypeNode('int'), + false, + '$a', + PHP_EOL . PHP_EOL . '// this is a comment', + false, + )), + ]), + ]; + yield [ + 'Comment after Doctrine tag 1', + '/** @ORM\Doctrine // this is a description */', + new PhpDocNode([ + new PhpDocTagNode('@ORM\Doctrine', new GenericTagValueNode('// this is a description')), + ]), + ]; + yield [ + 'Comment after Doctrine tag 2', + '/** @\ORM\Doctrine // this is a description */', + new PhpDocNode([ + new PhpDocTagNode('@\ORM\Doctrine', new DoctrineTagValueNode( + new DoctrineAnnotation('@\ORM\Doctrine', []), + '// this is a description', + )), + ]), + ]; + yield [ + 'Comment after Doctrine tag 3', + '/** @\ORM\Doctrine() // this is a description */', + new PhpDocNode([ + new PhpDocTagNode('@\ORM\Doctrine', new DoctrineTagValueNode( + new DoctrineAnnotation('@\ORM\Doctrine', []), + '// this is a description', + )), + ]), + ]; + yield [ + 'Comment after Doctrine tag 4', + '/** @\ORM\Doctrine() @\ORM\Entity() // this is a description */', + new PhpDocNode([ + new PhpDocTagNode('@\ORM\Doctrine', new DoctrineTagValueNode( + new DoctrineAnnotation('@\ORM\Doctrine', []), + '', + )), + new PhpDocTagNode('@\ORM\Entity', new DoctrineTagValueNode( + new DoctrineAnnotation('@\ORM\Entity', []), + '// this is a description', + )), + ]), + ]; + } + + public function provideInlineTags(): Iterator + { + yield [ + 'Inline @link tag in @copyright', + '/**' . PHP_EOL . + ' * Unit tests for stored_progress_bar_cleanup' . PHP_EOL . + ' *' . PHP_EOL . + ' * @package core' . PHP_EOL . + ' * @copyright 2024 onwards Catalyst IT EU {@link https://catalyst-eu.net}' . PHP_EOL . + ' * @\ORM\Entity() 2024 onwards Catalyst IT EU {@link https://catalyst-eu.net}' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Unit tests for stored_progress_bar_cleanup'), + new PhpDocTextNode(''), + new PhpDocTagNode('@package', new GenericTagValueNode('core')), + new PhpDocTagNode('@copyright', new GenericTagValueNode('2024 onwards Catalyst IT EU {@link https://catalyst-eu.net}')), + new PhpDocTagNode('@\ORM\Entity', new DoctrineTagValueNode(new DoctrineAnnotation('@\ORM\Entity', []), '2024 onwards Catalyst IT EU {@link https://catalyst-eu.net}')), + ]), + ]; + } + + public function provideParamOutTagsData(): Iterator + { + yield [ + 'OK param-out', + '/** @param-out string $s */', + new PhpDocNode([ + new PhpDocTagNode( + '@param-out', + new ParamOutTagValueNode( + new IdentifierTypeNode('string'), + '$s', + '', + ), + ), + ]), + ]; + + yield [ + 'OK param-out description', + '/** @param-out string $s description */', + new PhpDocNode([ + new PhpDocTagNode( + '@param-out', + new ParamOutTagValueNode( + new IdentifierTypeNode('string'), + '$s', + 'description', + ), + ), + ]), + ]; + } + + public function provideDoctrineData(): Iterator + { + yield [ + 'single tag node with empty parameters', + '/**' . PHP_EOL . + ' * @X() Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@X', + new DoctrineTagValueNode( + new DoctrineAnnotation('@X', []), + 'Content', + ), + ), + ]), + [new Doctrine\X()], + ]; + + $xWithZ = new Doctrine\X(); + $xWithZ->a = new Doctrine\Z(); + yield [ + 'single tag node with nested PHPDoc tag', + '/**' . PHP_EOL . + ' * @X(@Z) Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@X', + new DoctrineTagValueNode( + new DoctrineAnnotation('@X', [ + new DoctrineArgument(null, new DoctrineAnnotation('@Z', [])), + ]), + 'Content', + ), + ), + ]), + [$xWithZ], + ]; + + yield [ + 'single tag node with nested PHPDoc tag with field name', + '/**' . PHP_EOL . + ' * @X(a=@Z) Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@X', + new DoctrineTagValueNode( + new DoctrineAnnotation('@X', [ + new DoctrineArgument(new IdentifierTypeNode('a'), new DoctrineAnnotation('@Z', [])), + ]), + 'Content', + ), + ), + ]), + [$xWithZ], + ]; + + yield [ + 'single tag node with nested Doctrine tag', + '/**' . PHP_EOL . + ' * @X(@\PHPStan\PhpDocParser\Parser\Doctrine\Z) Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@X', + new DoctrineTagValueNode( + new DoctrineAnnotation('@X', [ + new DoctrineArgument(null, new DoctrineAnnotation('@\PHPStan\PhpDocParser\Parser\Doctrine\Z', [])), + ]), + 'Content', + ), + ), + ]), + [$xWithZ], + ]; + + yield [ + 'single tag node with nested Doctrine tag with field name', + '/**' . PHP_EOL . + ' * @X( a = @\PHPStan\PhpDocParser\Parser\Doctrine\Z) Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@X', + new DoctrineTagValueNode( + new DoctrineAnnotation('@X', [ + new DoctrineArgument(new IdentifierTypeNode('a'), new DoctrineAnnotation('@\PHPStan\PhpDocParser\Parser\Doctrine\Z', [])), + ]), + 'Content', + ), + ), + ]), + [$xWithZ], + ]; + + yield [ + 'single tag node with empty parameters with crazy whitespace', + '/**' . PHP_EOL . + ' * @X ( ' . PHP_EOL . + ' * ) Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@X', + new DoctrineTagValueNode( + new DoctrineAnnotation('@X', []), + 'Content', + ), + ), + ]), + [new Doctrine\X()], + ]; + + yield [ + 'single tag node with empty parameters with crazy whitespace with extra text node', + '/**' . PHP_EOL . + ' * @X ()' . PHP_EOL . + ' * Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@X', + new DoctrineTagValueNode( + new DoctrineAnnotation('@X', []), + PHP_EOL . + 'Content', + ), + ), + ]), + [new Doctrine\X()], + ]; + + yield [ + 'single FQN tag node without parentheses', + '/**' . PHP_EOL . + ' * @\PHPStan\PhpDocParser\Parser\Doctrine\X Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@\PHPStan\PhpDocParser\Parser\Doctrine\X', + new DoctrineTagValueNode( + new DoctrineAnnotation('@\PHPStan\PhpDocParser\Parser\Doctrine\X', []), + 'Content', + ), + ), + ]), + [new Doctrine\X()], + ]; + + yield [ + 'single FQN tag node with empty parameters', + '/**' . PHP_EOL . + ' * @\PHPStan\PhpDocParser\Parser\Doctrine\X() Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@\PHPStan\PhpDocParser\Parser\Doctrine\X', + new DoctrineTagValueNode( + new DoctrineAnnotation('@\PHPStan\PhpDocParser\Parser\Doctrine\X', []), + 'Content', + ), + ), + ]), + [new Doctrine\X()], + ]; + + $x = new Doctrine\X(); + $x->a = Doctrine\Y::SOME; + + $z = new Doctrine\Z(); + $z->code = 123; + $x->b = [$z]; + yield [ + 'single tag node with other tags in parameters', + '/**' . PHP_EOL . + ' * @X(' . PHP_EOL . + ' * a=Y::SOME,' . PHP_EOL . + ' * b={' . PHP_EOL . + ' * @Z(' . PHP_EOL . + ' * code=123' . PHP_EOL . + ' * )' . PHP_EOL . + ' * }' . PHP_EOL . + ' * ) Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@X', + new DoctrineTagValueNode( + new DoctrineAnnotation( + '@X', + [ + new DoctrineArgument(new IdentifierTypeNode('a'), new ConstFetchNode('Y', 'SOME')), + new DoctrineArgument(new IdentifierTypeNode('b'), new DoctrineArray([ + new DoctrineArrayItem(null, new DoctrineAnnotation('@Z', [ + new DoctrineArgument(new IdentifierTypeNode('code'), new ConstExprIntegerNode('123')), + ])), + ])), + ], + ), + 'Content', + ), + ), + ]), + [$x], + ]; + + yield [ + 'single tag node with other tags in parameters with crazy whitespace inbetween', + '/**' . PHP_EOL . + ' * @X (' . PHP_EOL . + ' * a' . PHP_EOL . + ' * = Y::SOME,' . PHP_EOL . + ' * b = ' . PHP_EOL . + ' * {' . PHP_EOL . + ' * @Z (' . PHP_EOL . + ' * code=123,' . PHP_EOL . + ' * ),' . PHP_EOL . + ' * },' . PHP_EOL . + ' * ) Content' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode( + '@X', + new DoctrineTagValueNode( + new DoctrineAnnotation( + '@X', + [ + new DoctrineArgument(new IdentifierTypeNode('a'), new ConstFetchNode('Y', 'SOME')), + new DoctrineArgument(new IdentifierTypeNode('b'), new DoctrineArray([ + new DoctrineArrayItem(null, new DoctrineAnnotation('@Z', [ + new DoctrineArgument(new IdentifierTypeNode('code'), new ConstExprIntegerNode('123')), + ])), + ])), + ], + ), + 'Content', + ), + ), + ]), + [$x], + ]; + + yield [ + 'Multiline tag behaviour 1', + '/**' . PHP_EOL . + ' * @X() test' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode(new DoctrineAnnotation( + '@X', + [], + ), 'test')), + ]), + [new Doctrine\X()], + ]; + + yield [ + 'Multiline tag behaviour 2', + '/**' . PHP_EOL . + ' * @X() test' . PHP_EOL . + ' * test2' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode(new DoctrineAnnotation( + '@X', + [], + ), 'test' . PHP_EOL . 'test2')), + ]), + [new Doctrine\X()], + ]; + yield [ + 'Multiline tag behaviour 3', + '/**' . PHP_EOL . + ' * @X() test' . PHP_EOL . + ' *' . PHP_EOL . + ' * test2' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode(new DoctrineAnnotation( + '@X', + [], + ), 'test' . PHP_EOL . + PHP_EOL . 'test2')), + ]), + [new Doctrine\X()], + ]; + yield [ + 'Multiline tag behaviour 4', + '/**' . PHP_EOL . + ' * @X() test' . PHP_EOL . + ' *' . PHP_EOL . + ' * test2' . PHP_EOL . + ' * @Z()' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode(new DoctrineAnnotation( + '@X', + [], + ), 'test' . PHP_EOL . + PHP_EOL . 'test2')), + new PhpDocTagNode('@Z', new DoctrineTagValueNode(new DoctrineAnnotation( + '@Z', + [], + ), '')), + ]), + [new Doctrine\X(), new Doctrine\Z()], + ]; + + yield [ + 'Multiline generic tag behaviour 1', + '/**' . PHP_EOL . + ' * @X test' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@X', new GenericTagValueNode('test')), + ]), + [new Doctrine\X()], + ]; + + yield [ + 'Multiline generic tag behaviour 2', + '/**' . PHP_EOL . + ' * @X test' . PHP_EOL . + ' * test2' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@X', new GenericTagValueNode('test' . PHP_EOL . 'test2')), + ]), + [new Doctrine\X()], + ]; + yield [ + 'Multiline generic tag behaviour 3', + '/**' . PHP_EOL . + ' * @X test' . PHP_EOL . + ' *' . PHP_EOL . + ' * test2' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@X', new GenericTagValueNode('test' . PHP_EOL . + PHP_EOL . + 'test2')), + ]), + [new Doctrine\X()], + ]; + yield [ + 'Multiline generic tag behaviour 4', + '/**' . PHP_EOL . + ' * @X test' . PHP_EOL . + ' *' . PHP_EOL . + ' * test2' . PHP_EOL . + ' * @Z' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@X', new GenericTagValueNode('test' . PHP_EOL . + PHP_EOL . + 'test2')), + new PhpDocTagNode('@Z', new GenericTagValueNode('')), + ]), + [new Doctrine\X(), new Doctrine\Z()], + ]; + + yield [ + 'More tags on the same line', + '/** @X() @Z() */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode(new DoctrineAnnotation('@X', []), '')), + new PhpDocTagNode('@Z', new DoctrineTagValueNode(new DoctrineAnnotation('@Z', []), '')), + ]), + [new Doctrine\X(), new Doctrine\Z()], + ]; + + yield [ + 'More tags on the same line with description inbetween', + '/** @X() test @Z() */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode(new DoctrineAnnotation('@X', []), 'test')), + new PhpDocTagNode('@Z', new DoctrineTagValueNode(new DoctrineAnnotation('@Z', []), '')), + ]), + [new Doctrine\X(), new Doctrine\Z()], + ]; + + yield [ + 'More tags on the same line with description inbetween, first one generic', + '/** @X test @Z() */', + new PhpDocNode([ + new PhpDocTagNode('@X', new GenericTagValueNode('test')), + new PhpDocTagNode('@Z', new DoctrineTagValueNode(new DoctrineAnnotation('@Z', []), '')), + ]), + [new Doctrine\X(), new Doctrine\Z()], + ]; + + yield [ + 'More generic tags on the same line with description inbetween, 2nd one @param which should become description', + '/** @X @phpstan-param int $z */', + new PhpDocNode([ + new PhpDocTagNode('@X', new GenericTagValueNode('@phpstan-param int $z')), + ]), + [new Doctrine\X()], + ]; + + yield [ + 'More generic tags on the same line with description inbetween, 2nd one @param which should become description can have a parse error', + '/** @X @phpstan-param |int $z */', + new PhpDocNode([ + new PhpDocTagNode('@X', new GenericTagValueNode('@phpstan-param |int $z')), + ]), + [new Doctrine\X()], + ]; + + yield [ + 'More tags on the same line with description inbetween, 2nd one @param which should become description', + '/** @X() @phpstan-param int $z */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode(new DoctrineAnnotation('@X', []), '@phpstan-param int $z')), + ]), + [new Doctrine\X()], + ]; + + yield [ + 'More tags on the same line with description inbetween, 2nd one @param which should become description can have a parse error', + '/** @X() @phpstan-param |int $z */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode(new DoctrineAnnotation('@X', []), '@phpstan-param |int $z')), + ]), + [new Doctrine\X()], + ]; + + $apiResource = new Doctrine\ApiResource(); + $apiResource->itemOperations = [ + 'get' => [ + 'security' => 'is_granted(' . PHP_EOL . + "constant('REDACTED')," . PHP_EOL . + 'object' . PHP_EOL . ')', + 'normalization_context' => [ + 'groups' => ['Redacted:read'], + ], + ], + ]; + yield [ + 'Regression test for issue #207', + '/**' . PHP_EOL . + ' * @ApiResource(' . PHP_EOL . + ' * itemOperations={' . PHP_EOL . + ' * "get"={' . PHP_EOL . + ' * "security"="is_granted(' . PHP_EOL . + "constant('REDACTED')," . PHP_EOL . + 'object' . PHP_EOL . + ')",' . PHP_EOL . + ' * "normalization_context"={"groups"={"Redacted:read"}}' . PHP_EOL . + ' * }' . PHP_EOL . + ' * }' . PHP_EOL . + ' * )' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTagNode('@ApiResource', new DoctrineTagValueNode( + new DoctrineAnnotation('@ApiResource', [ + new DoctrineArgument(new IdentifierTypeNode('itemOperations'), new DoctrineArray([ + new DoctrineArrayItem( + new DoctrineConstExprStringNode('get'), + new DoctrineArray([ + new DoctrineArrayItem( + new DoctrineConstExprStringNode('security'), + new DoctrineConstExprStringNode('is_granted(' . PHP_EOL . + "constant('REDACTED')," . PHP_EOL . + 'object' . PHP_EOL . + ')'), + ), + new DoctrineArrayItem( + new DoctrineConstExprStringNode('normalization_context'), + new DoctrineArray([ + new DoctrineArrayItem( + new DoctrineConstExprStringNode('groups'), + new DoctrineArray([ + new DoctrineArrayItem(null, new DoctrineConstExprStringNode('Redacted:read')), + ]), + ), + ]), + ), + ]), + ), + ])), + ]), + '', + )), + ]), + [$apiResource], + ]; + + $xWithString = new Doctrine\X(); + $xWithString->a = '"bar"'; + yield [ + 'Escaped strings', + '/** @X(a="""bar""") */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode( + new DoctrineAnnotation('@X', [ + new DoctrineArgument(new IdentifierTypeNode('a'), new DoctrineConstExprStringNode($xWithString->a)), + ]), + '', + )), + ]), + [$xWithString], + ]; + + $xWithString2 = new Doctrine\X(); + $xWithString2->a = 'Allowed choices are "bar" or "baz".'; + yield [ + 'Escaped strings 2', + '/** @X(a="Allowed choices are ""bar"" or ""baz"".") */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode( + new DoctrineAnnotation('@X', [ + new DoctrineArgument(new IdentifierTypeNode('a'), new DoctrineConstExprStringNode($xWithString2->a)), + ]), + '', + )), + ]), + [$xWithString2], + ]; + + $xWithString3 = new Doctrine\X(); + $xWithString3->a = 'In PHP, "" is an empty string'; + yield [ + 'Escaped strings 3', + '/** @X(a="In PHP, """" is an empty string") */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode( + new DoctrineAnnotation('@X', [ + new DoctrineArgument(new IdentifierTypeNode('a'), new DoctrineConstExprStringNode($xWithString3->a)), + ]), + '', + )), + ]), + [$xWithString3], + ]; + + $xWithString4 = new Doctrine\X(); + $xWithString4->a = '"May the Force be with you," he said.'; + yield [ + 'Escaped strings 4', + '/** @X(a="""May the Force be with you,"" he said.") */', + new PhpDocNode([ + new PhpDocTagNode('@X', new DoctrineTagValueNode( + new DoctrineAnnotation('@X', [ + new DoctrineArgument(new IdentifierTypeNode('a'), new DoctrineConstExprStringNode($xWithString4->a)), + ]), + '', + )), + ]), + [$xWithString4], + ]; + } + + public function provideDoctrineWithoutDoctrineCheckData(): Iterator { yield [ - 'Description with HTML tags in @return tag (close tags together)', - '/**' . PHP_EOL . - ' * @return Foo Important description' . PHP_EOL . - ' */', + 'Dummy 1', + '/** @DummyAnnotation(dummyValue="hello") */', new PhpDocNode([ - new PhpDocTagNode( - '@return', - new ReturnTagValueNode( - new IdentifierTypeNode('Foo'), - 'Important description' - ) - ), + new PhpDocTagNode('@DummyAnnotation', new DoctrineTagValueNode( + new DoctrineAnnotation('@DummyAnnotation', [ + new DoctrineArgument(new IdentifierTypeNode('dummyValue'), new DoctrineConstExprStringNode('hello')), + ]), + '', + )), + ]), + ]; + yield [ + 'Dummy 2', + '/** + * @DummyJoinTable(name="join_table", + * joinColumns={@DummyJoinColumn(name="col1", referencedColumnName="col2")}, + * inverseJoinColumns={ + * @DummyJoinColumn(name="col3", referencedColumnName="col4") + * }) + */', + new PhpDocNode([ + new PhpDocTagNode('@DummyJoinTable', new DoctrineTagValueNode( + new DoctrineAnnotation('@DummyJoinTable', [ + new DoctrineArgument(new IdentifierTypeNode('name'), new DoctrineConstExprStringNode('join_table')), + new DoctrineArgument(new IdentifierTypeNode('joinColumns'), new DoctrineArray([ + new DoctrineArrayItem(null, new DoctrineAnnotation('@DummyJoinColumn', [ + new DoctrineArgument(new IdentifierTypeNode('name'), new DoctrineConstExprStringNode('col1')), + new DoctrineArgument(new IdentifierTypeNode('referencedColumnName'), new DoctrineConstExprStringNode('col2')), + ])), + ])), + new DoctrineArgument(new IdentifierTypeNode('inverseJoinColumns'), new DoctrineArray([ + new DoctrineArrayItem(null, new DoctrineAnnotation('@DummyJoinColumn', [ + new DoctrineArgument(new IdentifierTypeNode('name'), new DoctrineConstExprStringNode('col3')), + new DoctrineArgument(new IdentifierTypeNode('referencedColumnName'), new DoctrineConstExprStringNode('col4')), + ])), + ])), + ]), + '', + )), ]), ]; yield [ - 'Description with HTML tags in @throws tag (closed tags with text between)', - '/**' . PHP_EOL . - ' * @throws FooException Important description etc' . PHP_EOL . - ' */', + 'Annotation in annotation', + '/** @AnnotationTargetAll(@AnnotationTargetAnnotation) */', new PhpDocNode([ - new PhpDocTagNode( - '@throws', - new ThrowsTagValueNode( - new IdentifierTypeNode('FooException'), - 'Important description etc' - ) - ), + new PhpDocTagNode('@AnnotationTargetAll', new DoctrineTagValueNode( + new DoctrineAnnotation('@AnnotationTargetAll', [ + new DoctrineArgument(null, new DoctrineAnnotation('@AnnotationTargetAnnotation', [])), + ]), + '', + )), ]), ]; yield [ - 'Description with HTML tags in @mixin tag', - '/**' . PHP_EOL . - ' * @mixin Mixin Important description' . PHP_EOL . - ' */', + 'Dangling comma annotation', + '/** @DummyAnnotation(dummyValue = "bar",) */', new PhpDocNode([ - new PhpDocTagNode( - '@mixin', - new MixinTagValueNode( - new IdentifierTypeNode('Mixin'), - 'Important description' - ) - ), + new PhpDocTagNode('@DummyAnnotation', new DoctrineTagValueNode( + new DoctrineAnnotation('@DummyAnnotation', [ + new DoctrineArgument(new IdentifierTypeNode('dummyValue'), new DoctrineConstExprStringNode('bar')), + ]), + '', + )), ]), ]; yield [ - 'Description with unclosed HTML tags in @return tag - unclosed HTML tag is parsed as generics', - '/**' . PHP_EOL . - ' * @return Foo Important description' . PHP_EOL . - ' */', + 'Multiple on one line', + '/** + * @DummyId @DummyColumn(type="integer") @DummyGeneratedValue + * @var int + */', new PhpDocNode([ - new PhpDocTagNode( - '@return', - new ReturnTagValueNode( - new GenericTypeNode( - new IdentifierTypeNode('Foo'), - [ - new IdentifierTypeNode('strong'), - ], - [ - GenericTypeNode::VARIANCE_INVARIANT, - ] - ), - 'Important description' - ) - ), + new PhpDocTagNode('@DummyId', new GenericTagValueNode('')), + new PhpDocTagNode('@DummyColumn', new DoctrineTagValueNode( + new DoctrineAnnotation('@DummyColumn', [ + new DoctrineArgument(new IdentifierTypeNode('type'), new DoctrineConstExprStringNode('integer')), + ]), + '', + )), + new PhpDocTagNode('@DummyGeneratedValue', new GenericTagValueNode('')), + new PhpDocTagNode('@var', new VarTagValueNode(new IdentifierTypeNode('int'), '', '')), ]), + ]; + + yield [ + 'Parse error with dashes', + '/** @AlsoDoNot\Parse-me */', new PhpDocNode([ - new PhpDocTagNode( - '@return', - new InvalidTagValueNode( - 'Foo Important description', - new ParserException( - 'Important', - Lexer::TOKEN_IDENTIFIER, - 27, - Lexer::TOKEN_HORIZONTAL_WS, - null, - 2 - ) - ) - ), + new PhpDocTagNode('@AlsoDoNot\Parse-me', new GenericTagValueNode('')), ]), ]; - } - /** - * @return array - */ - public function dataParseTagValue(): array - { - return [ - [ - '@param', - 'DateTimeImmutable::ATOM $a', - new ParamTagValueNode( - new ConstTypeNode(new ConstFetchNode('DateTimeImmutable', 'ATOM')), - false, - '$a', - '' - ), - ], - [ - '@var', - '$foo string[]', - new InvalidTagValueNode( - '$foo string[]', - new ParserException( - '$foo', - Lexer::TOKEN_VARIABLE, - 0, - Lexer::TOKEN_IDENTIFIER, - null, - 1 - ) - ), - ], + yield [ + 'Annotation with constant', + '/** @AnnotationWithConstants(PHP_EOL) */', + new PhpDocNode([ + new PhpDocTagNode('@AnnotationWithConstants', new DoctrineTagValueNode( + new DoctrineAnnotation('@AnnotationWithConstants', [ + new DoctrineArgument(null, new IdentifierTypeNode('PHP_EOL')), + ]), + '', + )), + ]), ]; - } - public function provideTagsWithNumbers(): Iterator - { yield [ - 'OK without description and tag with number in it', - '/** @special3 Foo */', + 'Nested arrays with nested annotations', + '/** @Name(foo={1,2, {"key"=@Name}}) */', new PhpDocNode([ - new PhpDocTagNode( - '@special3', - new GenericTagValueNode('Foo') - ), + new PhpDocTagNode('@Name', new DoctrineTagValueNode( + new DoctrineAnnotation('@Name', [ + new DoctrineArgument(new IdentifierTypeNode('foo'), new DoctrineArray([ + new DoctrineArrayItem(null, new ConstExprIntegerNode('1')), + new DoctrineArrayItem(null, new ConstExprIntegerNode('2')), + new DoctrineArrayItem(null, new DoctrineArray([ + new DoctrineArrayItem(new DoctrineConstExprStringNode('key'), new DoctrineAnnotation( + '@Name', + [], + )), + ])), + ])), + ]), + '', + )), ]), ]; - } - public function provideTagsWithBackslash(): Iterator - { yield [ - 'OK without description and tag with backslashes in it', - '/** @ORM\Mapping\Entity User */', + 'Namespaced constant', + '/** @AnnotationWithConstants(Doctrine\Tests\Common\Annotations\Fixtures\AnnotationWithConstants::FLOAT) */', new PhpDocNode([ - new PhpDocTagNode( - '@ORM\Mapping\Entity', - new GenericTagValueNode('User') - ), + new PhpDocTagNode('@AnnotationWithConstants', new DoctrineTagValueNode( + new DoctrineAnnotation('@AnnotationWithConstants', [ + new DoctrineArgument(null, new ConstFetchNode('Doctrine\Tests\Common\Annotations\Fixtures\AnnotationWithConstants', 'FLOAT')), + ]), + '', + )), ]), ]; yield [ - 'OK without description and tag with backslashes in it and parenthesis', - '/** @ORM\Mapping\JoinColumn(name="column_id", referencedColumnName="id") */', + 'Another namespaced constant', + '/** @AnnotationWithConstants(\Doctrine\Tests\Common\Annotations\Fixtures\AnnotationWithConstants::FLOAT) */', new PhpDocNode([ - new PhpDocTagNode( - '@ORM\Mapping\JoinColumn', - new GenericTagValueNode('(name="column_id", referencedColumnName="id")') - ), + new PhpDocTagNode('@AnnotationWithConstants', new DoctrineTagValueNode( + new DoctrineAnnotation('@AnnotationWithConstants', [ + new DoctrineArgument(null, new ConstFetchNode('\Doctrine\Tests\Common\Annotations\Fixtures\AnnotationWithConstants', 'FLOAT')), + ]), + '', + )), ]), ]; - } - public function provideSelfOutTagsData(): Iterator - { yield [ - 'OK phpstan-self-out', - '/** @phpstan-self-out self */', + 'Array with namespaced constants', + '/** @AnnotationWithConstants({ + Doctrine\Tests\Common\Annotations\Fixtures\InterfaceWithConstants::SOME_KEY = AnnotationWithConstants::INTEGER +}) */', new PhpDocNode([ - new PhpDocTagNode( - '@phpstan-self-out', - new SelfOutTagValueNode( - new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), - '' - ) - ), + new PhpDocTagNode('@AnnotationWithConstants', new DoctrineTagValueNode( + new DoctrineAnnotation('@AnnotationWithConstants', [ + new DoctrineArgument(null, new DoctrineArray([ + new DoctrineArrayItem( + new ConstFetchNode('Doctrine\Tests\Common\Annotations\Fixtures\InterfaceWithConstants', 'SOME_KEY'), + new ConstFetchNode('AnnotationWithConstants', 'INTEGER'), + ), + ])), + ]), + '', + )), ]), ]; yield [ - 'OK phpstan-this-out', - '/** @phpstan-this-out self */', + 'Array with colon', + '/** @Name({"foo": "bar"}) */', new PhpDocNode([ - new PhpDocTagNode( - '@phpstan-this-out', - new SelfOutTagValueNode( - new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), - '' - ) - ), + new PhpDocTagNode('@Name', new DoctrineTagValueNode( + new DoctrineAnnotation('@Name', [ + new DoctrineArgument(null, new DoctrineArray([ + new DoctrineArrayItem(new DoctrineConstExprStringNode('foo'), new DoctrineConstExprStringNode('bar')), + ])), + ]), + '', + )), ]), ]; yield [ - 'OK psalm-self-out', - '/** @psalm-self-out self */', + 'More tags on the same line with description inbetween, second Doctrine one cannot have parse error', + '/** @X test @Z(test= */', new PhpDocNode([ - new PhpDocTagNode( - '@psalm-self-out', - new SelfOutTagValueNode( - new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), - '' - ) - ), + new PhpDocTagNode('@X', new GenericTagValueNode('test')), + new PhpDocTagNode('@Z', new InvalidTagValueNode('(test=', new ParserException( + '=', + 14, + 19, + 5, + null, + 1, + ))), ]), + [new Doctrine\X()], ]; yield [ - 'OK psalm-this-out', - '/** @psalm-this-out self */', + 'More tags on the same line with description inbetween, second Doctrine one cannot have parse error 2', + '/** @X() test @Z(test= */', new PhpDocNode([ - new PhpDocTagNode( - '@psalm-this-out', - new SelfOutTagValueNode( - new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), - '' - ) - ), + new PhpDocTagNode('@X', new DoctrineTagValueNode(new DoctrineAnnotation('@X', []), 'test')), + new PhpDocTagNode('@Z', new InvalidTagValueNode('(test=', new ParserException( + '=', + 14, + 21, + 5, + null, + 1, + ))), ]), + [new Doctrine\X()], ]; yield [ - 'OK with description', - '/** @phpstan-self-out self description */', + 'Doctrine tag after common tag is just a description', + '/** @phpstan-param int $z @X() */', new PhpDocNode([ - new PhpDocTagNode( - '@phpstan-self-out', - new SelfOutTagValueNode( - new GenericTypeNode(new IdentifierTypeNode('self'), [new IdentifierTypeNode('T')], [GenericTypeNode::VARIANCE_INVARIANT]), - 'description' - ) - ), + new PhpDocTagNode('@phpstan-param', new ParamTagValueNode( + new IdentifierTypeNode('int'), + false, + '$z', + '@X()', + false, + )), ]), ]; - } - public function provideParamOutTagsData(): Iterator - { yield [ - 'OK param-out', - '/** @param-out string $s */', + 'Doctrine tag after common tag is just a description 2', + '/** @phpstan-param int $z @\X\Y() */', new PhpDocNode([ - new PhpDocTagNode( - '@param-out', - new ParamOutTagValueNode( - new IdentifierTypeNode('string'), - '$s', - '' - ) - ), + new PhpDocTagNode('@phpstan-param', new ParamTagValueNode( + new IdentifierTypeNode('int'), + false, + '$z', + '@\X\Y()', + false, + )), ]), ]; yield [ - 'OK param-out description', - '/** @param-out string $s description */', + 'Generic tag after common tag is just a description', + '/** @phpstan-param int $z @X */', new PhpDocNode([ - new PhpDocTagNode( - '@param-out', - new ParamOutTagValueNode( - new IdentifierTypeNode('string'), - '$s', - 'description' - ) - ), + new PhpDocTagNode('@phpstan-param', new ParamTagValueNode( + new IdentifierTypeNode('int'), + false, + '$z', + '@X', + false, + )), + ]), + ]; + + yield [ + 'Slevomat CS issue #1608', + '/**' . PHP_EOL . + ' * `"= "`' . PHP_EOL . + ' * a' . PHP_EOL . + ' * "' . PHP_EOL . + ' *' . PHP_EOL . + ' * @package foo' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('`"= "`' . PHP_EOL . + ' * a' . PHP_EOL . + ' * "'), + new PhpDocTextNode(''), + new PhpDocTagNode('@package', new GenericTagValueNode('foo')), ]), ]; } @@ -5518,8 +7371,8 @@ public function provideSpecializedTags(): Iterator new PhpDocTagNode( '@special:param', new GenericTagValueNode( - 'this is special' - ) + 'this is special', + ), ), ]), ]; @@ -5532,7 +7385,6 @@ public function provideSpecializedTags(): Iterator public function testParseTagValue(string $tag, string $phpDoc, Node $expectedPhpDocNode): void { $this->executeTestParseTagValue($this->phpDocParser, $tag, $phpDoc, $expectedPhpDocNode); - $this->executeTestParseTagValue($this->phpDocParserWithRequiredWhitespaceBeforeDescription, $tag, $phpDoc, $expectedPhpDocNode); } private function executeTestParseTagValue(PhpDocParser $phpDocParser, string $tag, string $phpDoc, Node $expectedPhpDocNode): void @@ -5647,47 +7499,140 @@ public function dataLinesAndIndexes(): iterable yield [ '/** @param Foo::** $a*/', [ - [1, 1, 1, 8], + [1, 1, 1, 8], + ], + ]; + + yield [ + '/** @return Foo */', + [ + [1, 1, 1, 3], + ], + ]; + + yield [ + '/** @return Foo*/', + [ + [1, 1, 1, 3], + ], + ]; + + yield [ + '/** @api */', + [ + [1, 1, 1, 1], + ], + ]; + } + + /** + * @dataProvider dataLinesAndIndexes + * @param list $childrenLines + */ + public function testLinesAndIndexes(string $phpDoc, array $childrenLines): void + { + $tokens = new TokenIterator($this->lexer->tokenize($phpDoc)); + $config = new ParserConfig([ + 'lines' => true, + 'indexes' => true, + ]); + $constExprParser = new ConstExprParser($config); + $typeParser = new TypeParser($config, $constExprParser); + $phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); + $phpDocNode = $phpDocParser->parse($tokens); + $children = $phpDocNode->children; + $this->assertCount(count($childrenLines), $children); + foreach ($children as $i => $child) { + $this->assertSame($childrenLines[$i][0], $child->getAttribute(Attribute::START_LINE)); + $this->assertSame($childrenLines[$i][1], $child->getAttribute(Attribute::END_LINE)); + $this->assertSame($childrenLines[$i][2], $child->getAttribute(Attribute::START_INDEX)); + $this->assertSame($childrenLines[$i][3], $child->getAttribute(Attribute::END_INDEX)); + } + } + + + /** + * @return iterable}> + */ + public function dataDeepNodesLinesAndIndexes(): iterable + { + yield [ + '/**' . PHP_EOL . + ' * @X({' . PHP_EOL . + ' * 1,' . PHP_EOL . + ' * 2' . PHP_EOL . + ' * , ' . PHP_EOL . + ' * 3,' . PHP_EOL . + ' * }' . PHP_EOL . + ' * )' . PHP_EOL . + ' */', + [ + [1, 9, 0, 25], // PhpDocNode + [2, 8, 2, 23], // PhpDocTagNode + [2, 8, 3, 23], // DoctrineTagValueNode + [2, 8, 3, 23], // DoctrineAnnotation + [2, 8, 4, 21], // DoctrineArgument + [2, 8, 4, 21], // DoctrineArray + [3, 3, 7, 7], // DoctrineArrayItem + [3, 3, 7, 7], // ConstExprIntegerNode + [4, 5, 11, 11], // DoctrineArrayItem + [4, 5, 11, 11], // ConstExprIntegerNode + [6, 6, 18, 18], // DoctrineArrayItem + [6, 6, 18, 18], // ConstExprIntegerNode ], ]; yield [ - '/** @return Foo */', + '/**' . PHP_EOL . + ' * @\Foo\Bar({' . PHP_EOL . + ' * }' . PHP_EOL . + ' * )' . PHP_EOL . + ' */', [ - [1, 1, 1, 3], + [1, 5, 0, 10], // PhpDocNode + [2, 4, 2, 8], // PhpDocTagNode + [2, 4, 3, 8], // DoctrineTagValueNode + [2, 4, 3, 8], // DoctrineAnnotation + [2, 4, 4, 6], // DoctrineArgument + [2, 4, 4, 6], // DoctrineArray ], ]; yield [ - '/** @return Foo*/', + '/** @api */', [ - [1, 1, 1, 3], + [1, 1, 0, 3], + [1, 1, 1, 1], + [1, 1, 3, 1], // GenericTagValueNode is empty so start index is higher than end index ], ]; } + /** - * @dataProvider dataLinesAndIndexes - * @param list $childrenLines + * @dataProvider dataDeepNodesLinesAndIndexes + * @param list $nodeAttributes */ - public function testLinesAndIndexes(string $phpDoc, array $childrenLines): void + public function testDeepNodesLinesAndIndexes(string $phpDoc, array $nodeAttributes): void { $tokens = new TokenIterator($this->lexer->tokenize($phpDoc)); - $usedAttributes = [ + $config = new ParserConfig([ 'lines' => true, 'indexes' => true, - ]; - $constExprParser = new ConstExprParser(true, true, $usedAttributes); - $typeParser = new TypeParser($constExprParser, true, $usedAttributes); - $phpDocParser = new PhpDocParser($typeParser, $constExprParser, true, true, $usedAttributes); - $phpDocNode = $phpDocParser->parse($tokens); - $children = $phpDocNode->children; - $this->assertCount(count($childrenLines), $children); - foreach ($children as $i => $child) { - $this->assertSame($childrenLines[$i][0], $child->getAttribute(Attribute::START_LINE)); - $this->assertSame($childrenLines[$i][1], $child->getAttribute(Attribute::END_LINE)); - $this->assertSame($childrenLines[$i][2], $child->getAttribute(Attribute::START_INDEX)); - $this->assertSame($childrenLines[$i][3], $child->getAttribute(Attribute::END_INDEX)); + ]); + $constExprParser = new ConstExprParser($config); + $typeParser = new TypeParser($config, $constExprParser); + $phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); + $visitor = new NodeCollectingVisitor(); + $traverser = new NodeTraverser([$visitor]); + $traverser->traverse([$phpDocParser->parse($tokens)]); + $nodes = $visitor->nodes; + $this->assertCount(count($nodeAttributes), $nodes); + foreach ($nodes as $i => $node) { + $this->assertSame($nodeAttributes[$i][0], $node->getAttribute(Attribute::START_LINE), sprintf('Start line of %d. node', $i + 1)); + $this->assertSame($nodeAttributes[$i][1], $node->getAttribute(Attribute::END_LINE), sprintf('End line of %d. node', $i + 1)); + $this->assertSame($nodeAttributes[$i][2], $node->getAttribute(Attribute::START_INDEX), sprintf('Start index of %d. node', $i + 1)); + $this->assertSame($nodeAttributes[$i][3], $node->getAttribute(Attribute::END_INDEX), sprintf('End index of %d. node', $i + 1)); } } @@ -5744,13 +7689,13 @@ public function dataReturnTypeLinesAndIndexes(): iterable public function testReturnTypeLinesAndIndexes(string $phpDoc, array $lines): void { $tokens = new TokenIterator($this->lexer->tokenize($phpDoc)); - $usedAttributes = [ + $config = new ParserConfig([ 'lines' => true, 'indexes' => true, - ]; - $constExprParser = new ConstExprParser(true, true, $usedAttributes); - $typeParser = new TypeParser($constExprParser, true, $usedAttributes); - $phpDocParser = new PhpDocParser($typeParser, $constExprParser, true, true, $usedAttributes); + ]); + $constExprParser = new ConstExprParser($config); + $typeParser = new TypeParser($config, $constExprParser); + $phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); $phpDocNode = $phpDocParser->parse($tokens); $returnTag = $phpDocNode->getReturnTagValues()[0]; $type = $returnTag->type; @@ -5767,6 +7712,9 @@ public function testReturnTypeLinesAndIndexes(string $phpDoc, array $lines): voi * @dataProvider provideSpecializedTags * @dataProvider provideParamTagsData * @dataProvider provideTypelessParamTagsData + * @dataProvider provideParamImmediatelyInvokedCallableTagsData + * @dataProvider provideParamLaterInvokedCallableTagsData + * @dataProvider provideParamClosureThisTagsData * @dataProvider provideVarTagsData * @dataProvider provideReturnTagsData * @dataProvider provideThrowsTagsData @@ -5786,13 +7734,18 @@ public function testReturnTypeLinesAndIndexes(string $phpDoc, array $lines): voi * @dataProvider provideTagsWithBackslash * @dataProvider provideSelfOutTagsData * @dataProvider provideParamOutTagsData + * @dataProvider provideDoctrineData + * @dataProvider provideDoctrineWithoutDoctrineCheckData */ public function testVerifyAttributes(string $label, string $input): void { - $usedAttributes = ['lines' => true, 'indexes' => true]; - $constExprParser = new ConstExprParser(true, true, $usedAttributes); - $typeParser = new TypeParser($constExprParser, true, $usedAttributes); - $phpDocParser = new PhpDocParser($typeParser, $constExprParser, true, true, $usedAttributes); + $config = new ParserConfig([ + 'lines' => true, + 'indexes' => true, + ]); + $constExprParser = new ConstExprParser($config); + $typeParser = new TypeParser($config, $constExprParser); + $phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); $tokens = new TokenIterator($this->lexer->tokenize($input)); $visitor = new NodeCollectingVisitor(); @@ -5807,4 +7760,368 @@ public function testVerifyAttributes(string $label, string $input): void } } + /** + * @dataProvider provideDoctrineData + * @param list $expectedAnnotations + */ + public function testDoctrine( + string $label, + string $input, + PhpDocNode $expectedPhpDocNode, + array $expectedAnnotations = [] + ): void + { + $parser = new DocParser(); + $parser->addNamespace('PHPStan\PhpDocParser\Parser\Doctrine'); + $this->assertEquals($expectedAnnotations, $parser->parse($input, $label), $label); + } + + /** + * @return iterable + */ + public function dataTextBetweenTagsBelongsToDescription(): iterable + { + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' * paramA description' . PHP_EOL . + ' * @param int $b' . PHP_EOL . + ' * paramB description' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', PHP_EOL . ' paramA description', false)), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$b', PHP_EOL . ' paramB description', false)), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' *' . PHP_EOL . + ' * @param int $b' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', '', false)), + new PhpDocTextNode(''), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$b', '', false)), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a aaaa' . PHP_EOL . + ' * bbbb' . PHP_EOL . + ' *' . PHP_EOL . + ' * ccc' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', 'aaaa' . PHP_EOL . ' bbbb' . PHP_EOL . PHP_EOL . 'ccc', false)), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @ORM\Column()' . PHP_EOL . + ' * bbbb' . PHP_EOL . + ' *' . PHP_EOL . + ' * ccc' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@ORM\Column', new DoctrineTagValueNode(new DoctrineAnnotation('@ORM\Column', []), PHP_EOL . ' bbbb' . PHP_EOL . PHP_EOL . 'ccc')), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @ORM\Column() aaaa' . PHP_EOL . + ' * bbbb' . PHP_EOL . + ' *' . PHP_EOL . + ' * ccc' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@ORM\Column', new DoctrineTagValueNode(new DoctrineAnnotation('@ORM\Column', []), 'aaaa' . PHP_EOL . ' bbbb' . PHP_EOL . PHP_EOL . 'ccc')), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @ORM\Column() aaaa' . PHP_EOL . + ' * bbbb' . PHP_EOL . + ' *' . PHP_EOL . + ' * ccc' . PHP_EOL . + ' * @param int $b' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@ORM\Column', new DoctrineTagValueNode(new DoctrineAnnotation('@ORM\Column', []), 'aaaa' . PHP_EOL . ' bbbb' . PHP_EOL . PHP_EOL . 'ccc')), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$b', '', false)), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', '', false)), + new PhpDocTextNode(''), + new PhpDocTextNode(''), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' * test' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', PHP_EOL . PHP_EOL . PHP_EOL . 'test', false)), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a test' . PHP_EOL . + ' *' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', 'test', false)), + new PhpDocTextNode(''), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a test' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', 'test', false)), + new PhpDocTextNode(''), + new PhpDocTextNode(''), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' * test' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', PHP_EOL . ' test', false)), + new PhpDocTextNode(''), + new PhpDocTextNode(''), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' * test' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', PHP_EOL . ' test', false)), + new PhpDocTextNode(''), + new PhpDocTextNode(''), + new PhpDocTextNode(''), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' * test' . PHP_EOL . + ' *' . PHP_EOL . + ' * test 2' . PHP_EOL . + ' *' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', PHP_EOL . ' test' . PHP_EOL . PHP_EOL . 'test 2', false)), + new PhpDocTextNode(''), + ]), + ]; + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @param int $a' . PHP_EOL . + ' * test' . PHP_EOL . + ' *' . PHP_EOL . + ' * test 2' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('int'), false, '$a', PHP_EOL . ' test' . PHP_EOL . PHP_EOL . 'test 2', false)), + new PhpDocTextNode(''), + new PhpDocTextNode(''), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * Real description' . PHP_EOL . + ' * @ORM\Column()' . PHP_EOL . + ' * test' . PHP_EOL . + ' *' . PHP_EOL . + ' * test 2' . PHP_EOL . + ' *' . PHP_EOL . + ' *' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('Real description'), + new PhpDocTagNode('@ORM\Column', new DoctrineTagValueNode(new DoctrineAnnotation('@ORM\Column', []), PHP_EOL . ' test' . PHP_EOL . PHP_EOL . 'test 2')), + new PhpDocTextNode(''), + new PhpDocTextNode(''), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' * `"= "`' . PHP_EOL . + ' * a' . PHP_EOL . + ' * "' . PHP_EOL . + ' *' . PHP_EOL . + ' * @package foo' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode('`"= "`' . PHP_EOL . + ' * a' . PHP_EOL . + ' * "'), + new PhpDocTextNode(''), + new PhpDocTagNode('@package', new GenericTagValueNode('foo')), + ]), + ]; + + yield [ + '/** @deprecated in Drupal 8.6.0 and will be removed before Drupal 9.0.0. In + * Drupal 9 there will be no way to set the status and in Drupal 8 this + * ability has been removed because mb_*() functions are supplied using + * Symfony\'s polyfill. */', + new PhpDocNode([ + new PhpDocTagNode( + '@deprecated', + new DeprecatedTagValueNode('in Drupal 8.6.0 and will be removed before Drupal 9.0.0. In + Drupal 9 there will be no way to set the status and in Drupal 8 this + ability has been removed because mb_*() functions are supplied using + Symfony\'s polyfill.'), + ), + ]), + ]; + + yield [ + '/** @\ORM\Column() in Drupal 8.6.0 and will be removed before Drupal 9.0.0. In + * Drupal 9 there will be no way to set the status and in Drupal 8 this + * ability has been removed because mb_*() functions are supplied using + * Symfony\'s polyfill. */', + new PhpDocNode([ + new PhpDocTagNode( + '@\ORM\Column', + new DoctrineTagValueNode( + new DoctrineAnnotation('@\ORM\Column', []), + 'in Drupal 8.6.0 and will be removed before Drupal 9.0.0. In + Drupal 9 there will be no way to set the status and in Drupal 8 this + ability has been removed because mb_*() functions are supplied using + Symfony\'s polyfill.', + ), + ), + ]), + ]; + + yield [ + '/**' . PHP_EOL . + ' *' . PHP_EOL . + ' * MultiLine' . PHP_EOL . + ' * description' . PHP_EOL . + ' * @param bool $a' . PHP_EOL . + ' *' . PHP_EOL . + ' * @return void' . PHP_EOL . + ' *' . PHP_EOL . + ' * @throws \Exception' . PHP_EOL . + ' *' . PHP_EOL . + ' */', + new PhpDocNode([ + new PhpDocTextNode( + PHP_EOL . + 'MultiLine' . PHP_EOL . + 'description', + ), + new PhpDocTagNode('@param', new ParamTagValueNode( + new IdentifierTypeNode('bool'), + false, + '$a', + '', + false, + )), + new PhpDocTextNode(''), + new PhpDocTagNode('@return', new ReturnTagValueNode( + new IdentifierTypeNode('void'), + '', + )), + new PhpDocTextNode(''), + new PhpDocTagNode('@throws', new ThrowsTagValueNode( + new IdentifierTypeNode('\Exception'), + '', + )), + new PhpDocTextNode(''), + ]), + ]; + } + + /** + * @dataProvider dataTextBetweenTagsBelongsToDescription + */ + public function testTextBetweenTagsBelongsToDescription( + string $input, + PhpDocNode $expectedPhpDocNode + ): void + { + $config = new ParserConfig([]); + $constExprParser = new ConstExprParser($config); + $typeParser = new TypeParser($config, $constExprParser); + $phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); + + $tokens = new TokenIterator($this->lexer->tokenize($input)); + $actualPhpDocNode = $phpDocParser->parse($tokens); + + $this->assertEquals($expectedPhpDocNode, $actualPhpDocNode); + $this->assertSame((string) $expectedPhpDocNode, (string) $actualPhpDocNode); + $this->assertSame(Lexer::TOKEN_END, $tokens->currentTokenType()); + } + } diff --git a/tests/PHPStan/Parser/TokenIteratorTest.php b/tests/PHPStan/Parser/TokenIteratorTest.php new file mode 100644 index 00000000..3cc51fc1 --- /dev/null +++ b/tests/PHPStan/Parser/TokenIteratorTest.php @@ -0,0 +1,60 @@ + + */ + public function dataGetDetectedNewline(): iterable + { + yield [ + '/** @param Foo $a */', + null, + ]; + + yield [ + '/**' . "\n" . + ' * @param Foo $a' . "\n" . + ' */', + "\n", + ]; + + yield [ + '/**' . "\r\n" . + ' * @param Foo $a' . "\r\n" . + ' */', + "\r\n", + ]; + + yield [ + '/**' . PHP_EOL . + ' * @param Foo $a' . PHP_EOL . + ' */', + PHP_EOL, + ]; + } + + /** + * @dataProvider dataGetDetectedNewline + */ + public function testGetDetectedNewline(string $phpDoc, ?string $expectedNewline): void + { + $config = new ParserConfig([]); + $lexer = new Lexer($config); + $tokens = new TokenIterator($lexer->tokenize($phpDoc)); + $constExprParser = new ConstExprParser($config); + $typeParser = new TypeParser($config, $constExprParser); + $phpDocParser = new PhpDocParser($config, $typeParser, $constExprParser); + $phpDocParser->parse($tokens); + $this->assertSame($expectedNewline, $tokens->getDetectedNewline()); + } + +} diff --git a/tests/PHPStan/Parser/TypeParserTest.php b/tests/PHPStan/Parser/TypeParserTest.php index 42a3c40a..8fe96f5f 100644 --- a/tests/PHPStan/Parser/TypeParserTest.php +++ b/tests/PHPStan/Parser/TypeParserTest.php @@ -3,15 +3,21 @@ namespace PHPStan\PhpDocParser\Parser; use Exception; +use PHPStan\PhpDocParser\Ast\AbstractNodeVisitor; use PHPStan\PhpDocParser\Ast\Attribute; +use PHPStan\PhpDocParser\Ast\Comment; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprFloatNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprIntegerNode; +use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprStringNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstFetchNode; -use PHPStan\PhpDocParser\Ast\ConstExpr\QuoteAwareConstExprStringNode; use PHPStan\PhpDocParser\Ast\Node; use PHPStan\PhpDocParser\Ast\NodeTraverser; +use PHPStan\PhpDocParser\Ast\NodeVisitor\CloningVisitor; +use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\TemplateTagValueNode; use PHPStan\PhpDocParser\Ast\Type\ArrayShapeItemNode; use PHPStan\PhpDocParser\Ast\Type\ArrayShapeNode; +use PHPStan\PhpDocParser\Ast\Type\ArrayShapeUnsealedTypeNode; use PHPStan\PhpDocParser\Ast\Type\ArrayTypeNode; use PHPStan\PhpDocParser\Ast\Type\CallableTypeNode; use PHPStan\PhpDocParser\Ast\Type\CallableTypeParameterNode; @@ -29,6 +35,7 @@ use PHPStan\PhpDocParser\Ast\Type\TypeNode; use PHPStan\PhpDocParser\Ast\Type\UnionTypeNode; use PHPStan\PhpDocParser\Lexer\Lexer; +use PHPStan\PhpDocParser\ParserConfig; use PHPStan\PhpDocParser\Printer\Printer; use PHPUnit\Framework\TestCase; use function get_class; @@ -38,17 +45,16 @@ class TypeParserTest extends TestCase { - /** @var Lexer */ - private $lexer; + private Lexer $lexer; - /** @var TypeParser */ - private $typeParser; + private TypeParser $typeParser; protected function setUp(): void { parent::setUp(); - $this->lexer = new Lexer(); - $this->typeParser = new TypeParser(new ConstExprParser(true, true), true); + $config = new ParserConfig([]); + $this->lexer = new Lexer($config); + $this->typeParser = new TypeParser($config, new ConstExprParser($config)); } @@ -65,10 +71,11 @@ public function testParse(string $input, $expectedResult, int $nextTokenType = L $tokens = new TokenIterator($this->lexer->tokenize($input)); $typeNode = $this->typeParser->parse($tokens); + $this->assertInstanceOf(TypeNode::class, $expectedResult); $this->assertSame((string) $expectedResult, (string) $typeNode); $this->assertInstanceOf(get_class($expectedResult), $typeNode); - $this->assertEquals($expectedResult, $typeNode); + $this->assertEquals($this->unsetAllAttributes($expectedResult), $this->unsetAllAttributes($typeNode)); $this->assertSame($nextTokenType, $tokens->currentTokenType(), Lexer::TOKEN_LABELS[$nextTokenType]); if (strpos((string) $expectedResult, '$ref') !== false) { @@ -115,13 +122,16 @@ public function testVerifyAttributes(string $input, $expectedResult): void $this->expectExceptionMessage($expectedResult->getMessage()); } - $usedAttributes = ['lines' => true, 'indexes' => true]; - $typeParser = new TypeParser(new ConstExprParser(true, true, $usedAttributes), true, $usedAttributes); + $config = new ParserConfig(['lines' => true, 'indexes' => true, 'comments' => true]); + $typeParser = new TypeParser($config, new ConstExprParser($config)); $tokens = new TokenIterator($this->lexer->tokenize($input)); + $typeNode = $typeParser->parse($tokens); + $this->assertInstanceOf(TypeNode::class, $expectedResult); + $visitor = new NodeCollectingVisitor(); $traverser = new NodeTraverser([$visitor]); - $traverser->traverse([$typeParser->parse($tokens)]); + $traverser->traverse([$typeNode]); foreach ($visitor->nodes as $node) { $this->assertNotNull($node->getAttribute(Attribute::START_LINE), (string) $node); @@ -129,6 +139,84 @@ public function testVerifyAttributes(string $input, $expectedResult): void $this->assertNotNull($node->getAttribute(Attribute::START_INDEX), (string) $node); $this->assertNotNull($node->getAttribute(Attribute::END_INDEX), (string) $node); } + + $this->assertEquals( + $this->unsetAllAttributesButComments($expectedResult), + $this->unsetAllAttributesButComments($typeNode), + ); + } + + + private function unsetAllAttributes(Node $node): Node + { + $visitor = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + $node->setAttribute(Attribute::START_LINE, null); + $node->setAttribute(Attribute::END_LINE, null); + $node->setAttribute(Attribute::START_INDEX, null); + $node->setAttribute(Attribute::END_INDEX, null); + $node->setAttribute(Attribute::ORIGINAL_NODE, null); + $node->setAttribute(Attribute::COMMENTS, null); + + return $node; + } + + }; + + $cloningTraverser = new NodeTraverser([new CloningVisitor()]); + $newNodes = $cloningTraverser->traverse([$node]); + + $traverser = new NodeTraverser([$visitor]); + + /** @var PhpDocNode */ + return $traverser->traverse($newNodes)[0]; + } + + + private function unsetAllAttributesButComments(Node $node): Node + { + $visitor = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + $node->setAttribute(Attribute::START_LINE, null); + $node->setAttribute(Attribute::END_LINE, null); + $node->setAttribute(Attribute::START_INDEX, null); + $node->setAttribute(Attribute::END_INDEX, null); + $node->setAttribute(Attribute::ORIGINAL_NODE, null); + + if ($node->getAttribute(Attribute::COMMENTS) === []) { + $node->setAttribute(Attribute::COMMENTS, null); + } + + return $node; + } + + }; + + $cloningTraverser = new NodeTraverser([new CloningVisitor()]); + $newNodes = $cloningTraverser->traverse([$node]); + + $traverser = new NodeTraverser([$visitor]); + + /** @var PhpDocNode */ + return $traverser->traverse($newNodes)[0]; + } + + + /** + * @template TNode of Node + * @param TNode $node + * @return TNode + */ + public static function withComment(Node $node, string $comment, int $startLine, int $startIndex): Node + { + $comments = $node->getAttribute(Attribute::COMMENTS) ?? []; + $comments[] = new Comment($comment, $startLine, $startIndex); + $node->setAttribute(Attribute::COMMENTS, $comments); + return $node; } @@ -138,6 +226,100 @@ public function testVerifyAttributes(string $input, $expectedResult): void public function provideParseData(): array { return [ + [ + 'array{ + // a is for apple + a: int, + }', + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode( + self::withComment(new IdentifierTypeNode('a'), '// a is for apple', 2, 3), + false, + new IdentifierTypeNode('int'), + ), + ]), + ], + [ + 'array{ + // a is for // apple + a: int, + }', + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode( + self::withComment(new IdentifierTypeNode('a'), '// a is for // apple', 2, 3), + false, + new IdentifierTypeNode('int'), + ), + ]), + ], + [ + 'array{ + // a is for * apple + a: int, + }', + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode( + self::withComment(new IdentifierTypeNode('a'), '// a is for * apple', 2, 3), + false, + new IdentifierTypeNode('int'), + ), + ]), + ], + [ + 'array{ + // a is for http://www.apple.com/ + a: int, + }', + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode( + self::withComment(new IdentifierTypeNode('a'), '// a is for http://www.apple.com/', 2, 3), + false, + new IdentifierTypeNode('int'), + ), + ]), + ], + [ + 'array{ + // a is for apple + // a is also for awesome + a: int, + }', + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode( + self::withComment(self::withComment(new IdentifierTypeNode('a'), '// a is for apple', 2, 3), '// a is also for awesome', 3, 5), + false, + new IdentifierTypeNode('int'), + ), + ]), + ], + [ + 'string', + new IdentifierTypeNode('string'), + ], + [ + ' string ', + new IdentifierTypeNode('string'), + ], + [ + ' ( string ) ', + new IdentifierTypeNode('string'), + ], + [ + '( ( string ) )', + new IdentifierTypeNode('string'), + ], + [ + '\\Foo\Bar\\Baz', + new IdentifierTypeNode('\\Foo\Bar\\Baz'), + ], + [ + ' \\Foo\Bar\\Baz ', + new IdentifierTypeNode('\\Foo\Bar\\Baz'), + ], + [ + ' ( \\Foo\Bar\\Baz ) ', + new IdentifierTypeNode('\\Foo\Bar\\Baz'), + ], [ 'string', new IdentifierTypeNode('string'), @@ -278,13 +460,13 @@ public function provideParseData(): array [ 'string[]', new ArrayTypeNode( - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ], [ 'string [ ] ', new ArrayTypeNode( - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ], [ @@ -294,7 +476,7 @@ public function provideParseData(): array new IdentifierTypeNode('string'), new IdentifierTypeNode('int'), new IdentifierTypeNode('float'), - ]) + ]), ), ], [ @@ -302,9 +484,9 @@ public function provideParseData(): array new ArrayTypeNode( new ArrayTypeNode( new ArrayTypeNode( - new IdentifierTypeNode('string') - ) - ) + new IdentifierTypeNode('string'), + ), + ), ), ], [ @@ -312,9 +494,9 @@ public function provideParseData(): array new ArrayTypeNode( new ArrayTypeNode( new ArrayTypeNode( - new IdentifierTypeNode('string') - ) - ) + new IdentifierTypeNode('string'), + ), + ), ), ], [ @@ -326,9 +508,9 @@ public function provideParseData(): array new IdentifierTypeNode('string'), new IdentifierTypeNode('int'), new IdentifierTypeNode('float'), - ]) - ) - ) + ]), + ), + ), ), ], [ @@ -338,7 +520,7 @@ public function provideParseData(): array [ '?int', new NullableTypeNode( - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ], [ @@ -351,8 +533,8 @@ public function provideParseData(): array ], [ GenericTypeNode::VARIANCE_INVARIANT, - ] - ) + ], + ), ), ], [ @@ -366,7 +548,25 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], + ), + ], + [ + 'array< + // index with an int + int, + Foo\\Bar + >', + new GenericTypeNode( + new IdentifierTypeNode('array'), + [ + new IdentifierTypeNode('int'), + new IdentifierTypeNode('Foo\\Bar'), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + GenericTypeNode::VARIANCE_INVARIANT, + ], ), ], [ @@ -377,153 +577,154 @@ public function provideParseData(): array [ 'array{a: int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], [ 'array{a: ?int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, new NullableTypeNode( - new IdentifierTypeNode('int') - ) + new IdentifierTypeNode('int'), + ), ), ]), ], [ 'array{a?: ?int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), true, new NullableTypeNode( - new IdentifierTypeNode('int') - ) + new IdentifierTypeNode('int'), + ), ), ]), ], [ 'array{0: int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new ConstExprIntegerNode('0'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], [ 'array{0?: int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new ConstExprIntegerNode('0'), true, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], [ 'array{int, int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( null, false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( null, false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], [ 'array{a: int, b: string}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], [ 'array{a?: int, b: string, 0: int, 1?: DateTime, hello: string}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), true, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), new ArrayShapeItemNode( new ConstExprIntegerNode('0'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( new ConstExprIntegerNode('1'), true, - new IdentifierTypeNode('DateTime') + new IdentifierTypeNode('DateTime'), ), new ArrayShapeItemNode( new IdentifierTypeNode('hello'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], [ 'array{a: int, b: array{c: callable(): int}}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( new IdentifierTypeNode('b'), false, - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('c'), false, new CallableTypeNode( new IdentifierTypeNode('callable'), [], - new IdentifierTypeNode('int') - ) + new IdentifierTypeNode('int'), + [], + ), ), - ]) + ]), ), ]), ], [ '?array{a: int}', new NullableTypeNode( - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ]) + ]), ), ], [ @@ -532,7 +733,9 @@ public function provideParseData(): array '', Lexer::TOKEN_END, 6, - Lexer::TOKEN_IDENTIFIER + Lexer::TOKEN_IDENTIFIER, + null, + null, ), ], [ @@ -541,46 +744,48 @@ public function provideParseData(): array '=>', Lexer::TOKEN_OTHER, 8, - Lexer::TOKEN_CLOSE_CURLY_BRACKET + Lexer::TOKEN_CLOSE_CURLY_BRACKET, + null, + null, ), ], [ 'array{"a": int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( - new QuoteAwareConstExprStringNode('a', QuoteAwareConstExprStringNode::DOUBLE_QUOTED), + new ConstExprStringNode('a', ConstExprStringNode::DOUBLE_QUOTED), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], [ 'array{\'a\': int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( - new QuoteAwareConstExprStringNode('a', QuoteAwareConstExprStringNode::SINGLE_QUOTED), + new ConstExprStringNode('a', ConstExprStringNode::SINGLE_QUOTED), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], [ 'array{\'$ref\': int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( - new QuoteAwareConstExprStringNode('$ref', QuoteAwareConstExprStringNode::SINGLE_QUOTED), + new ConstExprStringNode('$ref', ConstExprStringNode::SINGLE_QUOTED), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], [ 'array{"$ref": int}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( - new QuoteAwareConstExprStringNode('$ref', QuoteAwareConstExprStringNode::DOUBLE_QUOTED), + new ConstExprStringNode('$ref', ConstExprStringNode::DOUBLE_QUOTED), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], @@ -588,11 +793,11 @@ public function provideParseData(): array 'array{ * a: int *}', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], @@ -600,11 +805,11 @@ public function provideParseData(): array 'array{ a: int, }', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], @@ -613,16 +818,16 @@ public function provideParseData(): array a: int, b: string, }', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], @@ -632,21 +837,21 @@ public function provideParseData(): array , b: string , c: string }', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), new ArrayShapeItemNode( new IdentifierTypeNode('c'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], @@ -655,78 +860,78 @@ public function provideParseData(): array a: int, b: string }', - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], [ 'array{a: int, b: int, ...}', - new ArrayShapeNode([ + ArrayShapeNode::createUnsealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ], false), + ], null), ], [ 'array{int, string, ...}', - new ArrayShapeNode([ + ArrayShapeNode::createUnsealed([ new ArrayShapeItemNode( null, false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( null, false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), - ], false), + ], null), ], [ 'array{...}', - new ArrayShapeNode([], false), + ArrayShapeNode::createUnsealed([], null), ], [ 'array{ * a: int, * ... *}', - new ArrayShapeNode([ + ArrayShapeNode::createUnsealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ], false), + ], null), ], [ 'array{ a: int, ..., }', - new ArrayShapeNode([ + ArrayShapeNode::createUnsealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ], false), + ], null), ], [ 'array{int, ..., string}', @@ -734,7 +939,9 @@ public function provideParseData(): array 'string', Lexer::TOKEN_IDENTIFIER, 16, - Lexer::TOKEN_CLOSE_CURLY_BRACKET + Lexer::TOKEN_CLOSE_CURLY_BRACKET, + null, + null, ), ], [ @@ -742,71 +949,561 @@ public function provideParseData(): array int, string }', - new ArrayShapeNode( + ArrayShapeNode::createSealed( [ new ArrayShapeItemNode( null, false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ArrayShapeItemNode( null, false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ], - true, - ArrayShapeNode::KIND_LIST + ArrayShapeNode::KIND_LIST, ), ], [ - 'callable(): Foo', - new CallableTypeNode( - new IdentifierTypeNode('callable'), - [], - new IdentifierTypeNode('Foo') + 'non-empty-array{ + int, + string + }', + ArrayShapeNode::createSealed( + [ + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('string'), + ), + ], + ArrayShapeNode::KIND_NON_EMPTY_ARRAY, ), ], [ - 'callable(): ?Foo', - new CallableTypeNode( - new IdentifierTypeNode('callable'), - [], - new NullableTypeNode( - new IdentifierTypeNode('Foo') - ) - ), + 'callable(): non-empty-array{int, string}', + new CallableTypeNode(new IdentifierTypeNode('callable'), [], ArrayShapeNode::createSealed( + [ + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('string'), + ), + ], + ArrayShapeNode::KIND_NON_EMPTY_ARRAY, + ), []), ], [ - 'callable(): Foo', - new CallableTypeNode( - new IdentifierTypeNode('callable'), - [], - new GenericTypeNode( - new IdentifierTypeNode('Foo'), - [ - new IdentifierTypeNode('Bar'), - ], - [ - GenericTypeNode::VARIANCE_INVARIANT, - ] - ) - ), + 'callable(): non-empty-list{int, string}', + new CallableTypeNode(new IdentifierTypeNode('callable'), [], ArrayShapeNode::createSealed( + [ + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('string'), + ), + ], + ArrayShapeNode::KIND_NON_EMPTY_LIST, + ), []), ], [ - 'callable(): Foo[]', - new CallableTypeNode( - new IdentifierTypeNode('callable'), - [], - new ArrayTypeNode(new GenericTypeNode( - new IdentifierTypeNode('Foo'), - [ - new IdentifierTypeNode('Bar'), - ], - [ - GenericTypeNode::VARIANCE_INVARIANT, - ] - )) + 'non-empty-list{ + int, + string + }', + ArrayShapeNode::createSealed( + [ + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('string'), + ), + ], + ArrayShapeNode::KIND_NON_EMPTY_LIST, + ), + ], + [ + 'array{...}', + ArrayShapeNode::createUnsealed( + [], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ArrayShapeNode::KIND_ARRAY, + ), + ], + [ + 'array{a: int, b?: int, ...}', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + new IdentifierTypeNode('a'), + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + new IdentifierTypeNode('b'), + true, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ), + ], + [ + 'array{a:int,b?:int,...}', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + new IdentifierTypeNode('a'), + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + new IdentifierTypeNode('b'), + true, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ArrayShapeNode::KIND_ARRAY, + ), + ], + [ + 'array{a: int, b?: int, ... ' . PHP_EOL + . ' < ' . PHP_EOL + . ' string ' . PHP_EOL + . ' > ' . PHP_EOL + . ' , ' . PHP_EOL + . ' }', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + new IdentifierTypeNode('a'), + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + new IdentifierTypeNode('b'), + true, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ), + ], + [ + 'array{...}', + ArrayShapeNode::createUnsealed( + [], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + new IdentifierTypeNode('int'), + ), + ArrayShapeNode::KIND_ARRAY, + ), + ], + [ + 'array{a: int, b?: int, ...}', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + new IdentifierTypeNode('a'), + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + new IdentifierTypeNode('b'), + true, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + new IdentifierTypeNode('int'), + ), + ), + ], + [ + 'array{a:int,b?:int,...}', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + new IdentifierTypeNode('a'), + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + new IdentifierTypeNode('b'), + true, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + new IdentifierTypeNode('int'), + ), + ), + ], + [ + 'array{a: int, b?: int, ... ' . PHP_EOL + . ' < ' . PHP_EOL + . ' int ' . PHP_EOL + . ' , ' . PHP_EOL + . ' string ' . PHP_EOL + . ' > ' . PHP_EOL + . ' , ' . PHP_EOL + . ' }', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + new IdentifierTypeNode('a'), + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + new IdentifierTypeNode('b'), + true, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + new IdentifierTypeNode('int'), + ), + ), + ], + [ + 'list{...}', + ArrayShapeNode::createUnsealed( + [], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ArrayShapeNode::KIND_LIST, + ), + ], + [ + 'list{int, int, ...}', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ArrayShapeNode::KIND_LIST, + ), + ], + [ + 'list{int,int,...}', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ArrayShapeNode::KIND_LIST, + ), + ], + [ + 'list{int, int, ... ' . PHP_EOL + . ' < ' . PHP_EOL + . ' string ' . PHP_EOL + . ' > ' . PHP_EOL + . ' , ' . PHP_EOL + . ' }', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ArrayShapeNode::KIND_LIST, + ), + ], + [ + 'list{0: int, 1?: int, ...}', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + new ConstExprIntegerNode('0'), + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + new ConstExprIntegerNode('1'), + true, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ArrayShapeNode::KIND_LIST, + ), + ], + [ + 'list{0:int,1?:int,...}', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + new ConstExprIntegerNode('0'), + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + new ConstExprIntegerNode('1'), + true, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ArrayShapeNode::KIND_LIST, + ), + ], + [ + 'list{0: int, 1?: int, ... ' . PHP_EOL + . ' < ' . PHP_EOL + . ' string ' . PHP_EOL + . ' > ' . PHP_EOL + . ' , ' . PHP_EOL + . ' }', + ArrayShapeNode::createUnsealed( + [ + new ArrayShapeItemNode( + new ConstExprIntegerNode('0'), + false, + new IdentifierTypeNode('int'), + ), + new ArrayShapeItemNode( + new ConstExprIntegerNode('1'), + true, + new IdentifierTypeNode('int'), + ), + ], + new ArrayShapeUnsealedTypeNode( + new IdentifierTypeNode('string'), + null, + ), + ArrayShapeNode::KIND_LIST, + ), + ], + [ + 'array{...<>}', + new ParserException( + '>', + Lexer::TOKEN_CLOSE_ANGLE_BRACKET, + 10, + Lexer::TOKEN_IDENTIFIER, + null, + null, + ), + ], + [ + 'array{...}', + new ParserException( + '>', + Lexer::TOKEN_CLOSE_ANGLE_BRACKET, + 14, + Lexer::TOKEN_IDENTIFIER, + null, + null, + ), + ], + [ + 'array{...}', + new ParserException( + ',', + Lexer::TOKEN_COMMA, + 21, + Lexer::TOKEN_CLOSE_ANGLE_BRACKET, + null, + null, + ), + ], + [ + 'array{...}', + new ParserException( + ',', + Lexer::TOKEN_COMMA, + 21, + Lexer::TOKEN_CLOSE_ANGLE_BRACKET, + null, + null, + ), + ], + [ + 'list{...<>}', + new ParserException( + '>', + Lexer::TOKEN_CLOSE_ANGLE_BRACKET, + 9, + Lexer::TOKEN_IDENTIFIER, + null, + null, + ), + ], + [ + 'list{...}', + new ParserException( + ',', + Lexer::TOKEN_COMMA, + 12, + Lexer::TOKEN_CLOSE_ANGLE_BRACKET, + null, + null, + ), + ], + [ + 'list{...}', + new ParserException( + ',', + Lexer::TOKEN_COMMA, + 12, + Lexer::TOKEN_CLOSE_ANGLE_BRACKET, + null, + null, + ), + ], + [ + 'callable(): Foo', + new CallableTypeNode( + new IdentifierTypeNode('callable'), + [], + new IdentifierTypeNode('Foo'), + [], + ), + ], + [ + 'pure-callable(): Foo', + new CallableTypeNode( + new IdentifierTypeNode('pure-callable'), + [], + new IdentifierTypeNode('Foo'), + [], + ), + ], + [ + 'pure-Closure(): Foo', + new CallableTypeNode( + new IdentifierTypeNode('pure-Closure'), + [], + new IdentifierTypeNode('Foo'), + [], + ), + ], + [ + 'callable(): ?Foo', + new CallableTypeNode( + new IdentifierTypeNode('callable'), + [], + new NullableTypeNode( + new IdentifierTypeNode('Foo'), + ), + [], + ), + ], + [ + 'callable(): Foo', + new CallableTypeNode( + new IdentifierTypeNode('callable'), + [], + new GenericTypeNode( + new IdentifierTypeNode('Foo'), + [ + new IdentifierTypeNode('Bar'), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + ], + ), + [], + ), + ], + [ + 'callable(): Foo[]', + new CallableTypeNode( + new IdentifierTypeNode('callable'), + [], + new ArrayTypeNode(new GenericTypeNode( + new IdentifierTypeNode('Foo'), + [ + new IdentifierTypeNode('Bar'), + ], + [ + GenericTypeNode::VARIANCE_INVARIANT, + ], + )), + [], ), ], [ @@ -815,7 +1512,8 @@ public function provideParseData(): array new CallableTypeNode( new IdentifierTypeNode('callable'), [], - new IdentifierTypeNode('Foo') + new IdentifierTypeNode('Foo'), + [], ), new IdentifierTypeNode('Bar'), ]), @@ -826,7 +1524,8 @@ public function provideParseData(): array new CallableTypeNode( new IdentifierTypeNode('callable'), [], - new IdentifierTypeNode('Foo') + new IdentifierTypeNode('Foo'), + [], ), new IdentifierTypeNode('Bar'), ]), @@ -839,7 +1538,8 @@ public function provideParseData(): array new UnionTypeNode([ new IdentifierTypeNode('Foo'), new IdentifierTypeNode('Bar'), - ]) + ]), + [], ), ], [ @@ -850,7 +1550,8 @@ public function provideParseData(): array new IntersectionTypeNode([ new IdentifierTypeNode('Foo'), new IdentifierTypeNode('Bar'), - ]) + ]), + [], ), ], [ @@ -858,13 +1559,14 @@ public function provideParseData(): array new CallableTypeNode( new IdentifierTypeNode('callable'), [], - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ]) + ]), + [], ), ], [ @@ -877,24 +1579,125 @@ public function provideParseData(): array true, true, '$a', - true + true, ), new CallableTypeParameterNode( new IdentifierTypeNode('B'), true, true, '', - true + true, ), new CallableTypeParameterNode( new IdentifierTypeNode('C'), false, false, '', - false + false, + ), + ], + new IdentifierTypeNode('Foo'), + [], + ), + ], + [ + 'callable(B): C', + new CallableTypeNode( + new IdentifierTypeNode('callable'), + [ + new CallableTypeParameterNode( + new IdentifierTypeNode('B'), + false, + false, + '', + false, + ), + ], + new IdentifierTypeNode('C'), + [ + new TemplateTagValueNode('A', null, ''), + ], + ), + ], + [ + 'callable<>(): void', + new ParserException( + '>', + Lexer::TOKEN_END, + 9, + Lexer::TOKEN_IDENTIFIER, + null, + null, + ), + ], + [ + 'Closure(T, int): (T|false)', + new CallableTypeNode( + new IdentifierTypeNode('Closure'), + [ + new CallableTypeParameterNode( + new IdentifierTypeNode('T'), + false, + false, + '', + false, + ), + new CallableTypeParameterNode( + new IdentifierTypeNode('int'), + false, + false, + '', + false, + ), + ], + new UnionTypeNode([ + new IdentifierTypeNode('T'), + new IdentifierTypeNode('false'), + ]), + [ + new TemplateTagValueNode('T', new IdentifierTypeNode('Model'), ''), + ], + ), + ], + [ + '\Closure(Tx, Ty): array{ Ty, Tx }', + new CallableTypeNode( + new IdentifierTypeNode('\Closure'), + [ + new CallableTypeParameterNode( + new IdentifierTypeNode('Tx'), + false, + false, + '', + false, + ), + new CallableTypeParameterNode( + new IdentifierTypeNode('Ty'), + false, + false, + '', + false, + ), + ], + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('Ty'), + ), + new ArrayShapeItemNode( + null, + false, + new IdentifierTypeNode('Tx'), ), + ]), + [ + new TemplateTagValueNode('Tx', new UnionTypeNode([ + new IdentifierTypeNode('X'), + new IdentifierTypeNode('Z'), + ]), ''), + new TemplateTagValueNode('Ty', new IdentifierTypeNode('Y'), ''), ], - new IdentifierTypeNode('Foo') ), ], [ @@ -912,7 +1715,7 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new UnionTypeNode([ new IdentifierTypeNode('int'), @@ -925,17 +1728,17 @@ public function provideParseData(): array ], [ GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new IdentifierTypeNode('bar'), - ]) + ]), ), ]), ], [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new IdentifierTypeNode('Lorem'), ]), @@ -949,13 +1752,20 @@ public function provideParseData(): array 'array[ int ]', new OffsetAccessTypeNode( new IdentifierTypeNode('array'), - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), + ), + ], + [ + 'self::TYPES[ int ]', + new OffsetAccessTypeNode( + new ConstTypeNode(new ConstFetchNode('self', 'TYPES')), + new IdentifierTypeNode('int'), ), ], [ "?\t\xA009", // edge-case with \h new NullableTypeNode( - new IdentifierTypeNode("\xA009") + new IdentifierTypeNode("\xA009"), ), ], [ @@ -970,8 +1780,8 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] - ) + ], + ), ), ], [ @@ -988,21 +1798,21 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] - ) + ], + ), ), ]), ], [ 'array{foo: int}[]', new ArrayTypeNode( - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('foo'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ]) + ]), ), ], [ @@ -1010,20 +1820,20 @@ public function provideParseData(): array new UnionTypeNode([ new IdentifierTypeNode('int'), new ArrayTypeNode( - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('foo'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ]) + ]), ), ]), ], [ '$this[]', new ArrayTypeNode( - new ThisTypeNode() + new ThisTypeNode(), ), ], [ @@ -1031,7 +1841,7 @@ public function provideParseData(): array new UnionTypeNode([ new IdentifierTypeNode('int'), new ArrayTypeNode( - new ThisTypeNode() + new ThisTypeNode(), ), ]), ], @@ -1041,29 +1851,30 @@ public function provideParseData(): array new IdentifierTypeNode('callable'), [], new ArrayTypeNode( - new IdentifierTypeNode('int') - ) + new IdentifierTypeNode('int'), + ), + [], ), ], [ '?int[]', new NullableTypeNode( new ArrayTypeNode( - new IdentifierTypeNode('int') - ) + new IdentifierTypeNode('int'), + ), ), ], [ 'callable(mixed...): TReturn', new CallableTypeNode(new IdentifierTypeNode('callable'), [ new CallableTypeParameterNode(new IdentifierTypeNode('mixed'), false, true, '', false), - ], new IdentifierTypeNode('TReturn')), + ], new IdentifierTypeNode('TReturn'), []), ], [ "'foo'|'bar'", new UnionTypeNode([ - new ConstTypeNode(new QuoteAwareConstExprStringNode('foo', QuoteAwareConstExprStringNode::SINGLE_QUOTED)), - new ConstTypeNode(new QuoteAwareConstExprStringNode('bar', QuoteAwareConstExprStringNode::SINGLE_QUOTED)), + new ConstTypeNode(new ConstExprStringNode('foo', ConstExprStringNode::SINGLE_QUOTED)), + new ConstTypeNode(new ConstExprStringNode('bar', ConstExprStringNode::SINGLE_QUOTED)), ]), ], [ @@ -1095,9 +1906,24 @@ public function provideParseData(): array '123_456.789_012', new ConstTypeNode(new ConstExprFloatNode('123456.789012')), ], + [ + '+0x10_20|+8e+2 | -0b11', + new UnionTypeNode([ + new ConstTypeNode(new ConstExprIntegerNode('+0x1020')), + new ConstTypeNode(new ConstExprFloatNode('+8e+2')), + new ConstTypeNode(new ConstExprIntegerNode('-0b11')), + ]), + ], + [ + '18_446_744_073_709_551_616|8.2023437675747321e-18_446_744_073_709_551_617', + new UnionTypeNode([ + new ConstTypeNode(new ConstExprIntegerNode('18446744073709551616')), + new ConstTypeNode(new ConstExprFloatNode('8.2023437675747321e-18446744073709551617')), + ]), + ], [ '"bar"', - new ConstTypeNode(new QuoteAwareConstExprStringNode('bar', QuoteAwareConstExprStringNode::DOUBLE_QUOTED)), + new ConstTypeNode(new ConstExprStringNode('bar', ConstExprStringNode::DOUBLE_QUOTED)), ], [ 'Foo::FOO_*', @@ -1135,7 +1961,7 @@ public function provideParseData(): array [ '( "foo" | Foo::FOO_* )', new UnionTypeNode([ - new ConstTypeNode(new QuoteAwareConstExprStringNode('foo', QuoteAwareConstExprStringNode::DOUBLE_QUOTED)), + new ConstTypeNode(new ConstExprStringNode('foo', ConstExprStringNode::DOUBLE_QUOTED)), new ConstTypeNode(new ConstFetchNode('Foo', 'FOO_*')), ]), ], @@ -1172,7 +1998,7 @@ public function provideParseData(): array ], [ GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ], [ @@ -1189,7 +2015,7 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ], [ @@ -1205,7 +2031,7 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ], [ @@ -1226,13 +2052,13 @@ public function provideParseData(): array ], [ GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ], [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ], [ @@ -1253,26 +2079,26 @@ public function provideParseData(): array ], [ GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ], [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ], [ 'array{}', - new ArrayShapeNode([]), + ArrayShapeNode::createSealed([]), ], [ 'array{}|int', - new UnionTypeNode([new ArrayShapeNode([]), new IdentifierTypeNode('int')]), + new UnionTypeNode([ArrayShapeNode::createSealed([]), new IdentifierTypeNode('int')]), ], [ 'int|array{}', - new UnionTypeNode([new IdentifierTypeNode('int'), new ArrayShapeNode([])]), + new UnionTypeNode([new IdentifierTypeNode('int'), ArrayShapeNode::createSealed([])]), ], [ 'callable(' . PHP_EOL . @@ -1283,7 +2109,8 @@ public function provideParseData(): array [ new CallableTypeParameterNode(new IdentifierTypeNode('Foo'), false, false, '', false), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), ], [ @@ -1297,7 +2124,8 @@ public function provideParseData(): array new CallableTypeParameterNode(new IdentifierTypeNode('Foo'), false, false, '', false), new CallableTypeParameterNode(new IdentifierTypeNode('Bar'), false, false, '', false), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), ], [ @@ -1310,7 +2138,8 @@ public function provideParseData(): array new CallableTypeParameterNode(new IdentifierTypeNode('Foo'), false, false, '', false), new CallableTypeParameterNode(new IdentifierTypeNode('Bar'), false, false, '', false), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), ], [ @@ -1330,15 +2159,17 @@ public function provideParseData(): array [ new CallableTypeParameterNode(new IdentifierTypeNode('Bar'), false, false, '', false), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), false, false, '', - false + false, ), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), ], [ @@ -1358,15 +2189,17 @@ public function provideParseData(): array [ new CallableTypeParameterNode(new IdentifierTypeNode('Bar'), false, false, '', false), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), false, false, '', - false + false, ), ], - new IdentifierTypeNode('void') + new IdentifierTypeNode('void'), + [], ), ], [ @@ -1376,7 +2209,23 @@ public function provideParseData(): array new IdentifierTypeNode('Bar'), new IdentifierTypeNode('never'), new IdentifierTypeNode('int'), - false + false, + ), + ], + [ + '( + Foo is Bar + ? + // never, I say + never + : + int)', + new ConditionalTypeNode( + new IdentifierTypeNode('Foo'), + new IdentifierTypeNode('Bar'), + new IdentifierTypeNode('never'), + new IdentifierTypeNode('int'), + false, ), ], [ @@ -1386,7 +2235,7 @@ public function provideParseData(): array new IdentifierTypeNode('Bar'), new IdentifierTypeNode('never'), new IdentifierTypeNode('int'), - true + true, ), ], [ @@ -1400,9 +2249,9 @@ public function provideParseData(): array new ConstTypeNode(new ConstFetchNode('self', 'TYPE_INT')), new IdentifierTypeNode('int'), new IdentifierTypeNode('bool'), - false + false, ), - false + false, ), ], [ @@ -1418,7 +2267,7 @@ public function provideParseData(): array new IdentifierTypeNode('int'), new IdentifierTypeNode('string'), ]), - false + false, ), ], [ @@ -1439,7 +2288,7 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new ConditionalTypeNode( new IdentifierTypeNode('TRandList'), @@ -1453,7 +2302,7 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new UnionTypeNode([ new GenericTypeNode( @@ -1465,7 +2314,7 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), new GenericTypeNode( new IdentifierTypeNode('LimitIterator'), @@ -1476,12 +2325,12 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ]), - false + false, ), - false + false, ), ], [ @@ -1497,7 +2346,7 @@ public function provideParseData(): array new IdentifierTypeNode('int'), new IdentifierTypeNode('string'), ]), - false + false, ), ], [ @@ -1517,7 +2366,7 @@ public function provideParseData(): array new IdentifierTypeNode('int'), new IdentifierTypeNode('string'), ]), - false + false, ), ], [ @@ -1526,9 +2375,9 @@ public function provideParseData(): array new ConstTypeNode( new ConstFetchNode( 'Currency', - 'CURRENCY_*' - ) - ) + 'CURRENCY_*', + ), + ), ), ], [ @@ -1542,9 +2391,9 @@ public function provideParseData(): array new IdentifierTypeNode('Bar'), new IdentifierTypeNode('false'), new IdentifierTypeNode('null'), - false + false, ), - false + false, ), ], [ @@ -1553,7 +2402,9 @@ public function provideParseData(): array 'is', Lexer::TOKEN_IDENTIFIER, 14, - Lexer::TOKEN_COLON + Lexer::TOKEN_COLON, + null, + null, ), ], [ @@ -1567,9 +2418,9 @@ public function provideParseData(): array new IdentifierTypeNode('Bar'), new IdentifierTypeNode('false'), new IdentifierTypeNode('null'), - false + false, ), - false + false, ), ], [ @@ -1578,7 +2429,9 @@ public function provideParseData(): array '$foo', Lexer::TOKEN_VARIABLE, 15, - Lexer::TOKEN_IDENTIFIER + Lexer::TOKEN_IDENTIFIER, + null, + null, ), ], [ @@ -1592,7 +2445,7 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_COVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), ], [ @@ -1606,7 +2459,7 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_CONTRAVARIANT, - ] + ], ), ], [ @@ -1615,7 +2468,9 @@ public function provideParseData(): array '>', Lexer::TOKEN_CLOSE_ANGLE_BRACKET, 13, - Lexer::TOKEN_IDENTIFIER + Lexer::TOKEN_IDENTIFIER, + null, + null, ), ], [ @@ -1624,7 +2479,9 @@ public function provideParseData(): array 'Bar', Lexer::TOKEN_IDENTIFIER, 16, - Lexer::TOKEN_CLOSE_ANGLE_BRACKET + Lexer::TOKEN_CLOSE_ANGLE_BRACKET, + null, + null, ), ], [ @@ -1638,7 +2495,7 @@ public function provideParseData(): array [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_BIVARIANT, - ] + ], ), ], [ @@ -1647,7 +2504,7 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], @@ -1658,8 +2515,8 @@ public function provideParseData(): array new IdentifierTypeNode('a'), false, new NullableTypeNode( - new IdentifierTypeNode('int') - ) + new IdentifierTypeNode('int'), + ), ), ]), ], @@ -1670,8 +2527,8 @@ public function provideParseData(): array new IdentifierTypeNode('a'), true, new NullableTypeNode( - new IdentifierTypeNode('int') - ) + new IdentifierTypeNode('int'), + ), ), ]), ], @@ -1681,12 +2538,12 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ObjectShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], @@ -1696,22 +2553,23 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ObjectShapeItemNode( new IdentifierTypeNode('b'), false, - new ArrayShapeNode([ + ArrayShapeNode::createSealed([ new ArrayShapeItemNode( new IdentifierTypeNode('c'), false, new CallableTypeNode( new IdentifierTypeNode('callable'), [], - new IdentifierTypeNode('int') - ) + new IdentifierTypeNode('int'), + [], + ), ), - ]) + ]), ), ]), ], @@ -1721,7 +2579,7 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ObjectShapeItemNode( new IdentifierTypeNode('b'), @@ -1733,10 +2591,11 @@ public function provideParseData(): array new CallableTypeNode( new IdentifierTypeNode('callable'), [], - new IdentifierTypeNode('int') - ) + new IdentifierTypeNode('int'), + [], + ), ), - ]) + ]), ), ]), ], @@ -1747,9 +2606,9 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ]) + ]), ), ], [ @@ -1758,7 +2617,9 @@ public function provideParseData(): array '', Lexer::TOKEN_END, 7, - Lexer::TOKEN_IDENTIFIER + Lexer::TOKEN_IDENTIFIER, + null, + null, ), ], [ @@ -1767,7 +2628,9 @@ public function provideParseData(): array '=>', Lexer::TOKEN_OTHER, 9, - Lexer::TOKEN_COLON + Lexer::TOKEN_COLON, + null, + null, ), ], [ @@ -1776,7 +2639,9 @@ public function provideParseData(): array '}', Lexer::TOKEN_CLOSE_CURLY_BRACKET, 10, - Lexer::TOKEN_COLON + Lexer::TOKEN_COLON, + null, + null, ), ], [ @@ -1785,7 +2650,9 @@ public function provideParseData(): array '0', Lexer::TOKEN_END, 7, - Lexer::TOKEN_IDENTIFIER + Lexer::TOKEN_IDENTIFIER, + null, + null, ), ], [ @@ -1794,16 +2661,18 @@ public function provideParseData(): array '0', Lexer::TOKEN_END, 7, - Lexer::TOKEN_IDENTIFIER + Lexer::TOKEN_IDENTIFIER, + null, + null, ), ], [ 'object{"a": int}', new ObjectShapeNode([ new ObjectShapeItemNode( - new QuoteAwareConstExprStringNode('a', QuoteAwareConstExprStringNode::DOUBLE_QUOTED), + new ConstExprStringNode('a', ConstExprStringNode::DOUBLE_QUOTED), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], @@ -1811,9 +2680,9 @@ public function provideParseData(): array 'object{\'a\': int}', new ObjectShapeNode([ new ObjectShapeItemNode( - new QuoteAwareConstExprStringNode('a', QuoteAwareConstExprStringNode::SINGLE_QUOTED), + new ConstExprStringNode('a', ConstExprStringNode::SINGLE_QUOTED), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], @@ -1821,9 +2690,9 @@ public function provideParseData(): array 'object{\'$ref\': int}', new ObjectShapeNode([ new ObjectShapeItemNode( - new QuoteAwareConstExprStringNode('$ref', QuoteAwareConstExprStringNode::SINGLE_QUOTED), + new ConstExprStringNode('$ref', ConstExprStringNode::SINGLE_QUOTED), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], @@ -1831,9 +2700,9 @@ public function provideParseData(): array 'object{"$ref": int}', new ObjectShapeNode([ new ObjectShapeItemNode( - new QuoteAwareConstExprStringNode('$ref', QuoteAwareConstExprStringNode::DOUBLE_QUOTED), + new ConstExprStringNode('$ref', ConstExprStringNode::DOUBLE_QUOTED), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], @@ -1845,7 +2714,7 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), ]), ], @@ -1857,7 +2726,20 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), + ), + ]), + ], + [ + 'object{ + // a is for apple + a: int, + }', + new ObjectShapeNode([ + new ObjectShapeItemNode( + self::withComment(new IdentifierTypeNode('a'), '// a is for apple', 2, 3), + false, + new IdentifierTypeNode('int'), ), ]), ], @@ -1870,12 +2752,12 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ObjectShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], @@ -1889,17 +2771,17 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ObjectShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), new ObjectShapeItemNode( new IdentifierTypeNode('c'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], @@ -1912,12 +2794,12 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('a'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), new ObjectShapeItemNode( new IdentifierTypeNode('b'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], @@ -1928,9 +2810,9 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('foo'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ]) + ]), ), ], [ @@ -1942,9 +2824,9 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('foo'), false, - new IdentifierTypeNode('int') + new IdentifierTypeNode('int'), ), - ]) + ]), ), ]), ], @@ -1966,12 +2848,12 @@ public function provideParseData(): array new ObjectShapeItemNode( new IdentifierTypeNode('attribute'), false, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), new ObjectShapeItemNode( new IdentifierTypeNode('value'), true, - new IdentifierTypeNode('string') + new IdentifierTypeNode('string'), ), ]), ], @@ -1987,19 +2869,21 @@ public function provideParseData(): array [ new CallableTypeParameterNode(new IdentifierTypeNode('Foo'), false, false, '', false), ], - new IdentifierTypeNode('Bar') - ) + new IdentifierTypeNode('Bar'), + [], + ), + [], ), ], [ 'callable(): ?int', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new NullableTypeNode(new IdentifierTypeNode('int'))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new NullableTypeNode(new IdentifierTypeNode('int')), []), ], [ 'callable(): object{foo: int}', new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ObjectShapeNode([ new ObjectShapeItemNode(new IdentifierTypeNode('foo'), false, new IdentifierTypeNode('int')), - ])), + ]), []), ], [ 'callable(): object{foo: int}[]', @@ -2009,33 +2893,34 @@ public function provideParseData(): array new ArrayTypeNode( new ObjectShapeNode([ new ObjectShapeItemNode(new IdentifierTypeNode('foo'), false, new IdentifierTypeNode('int')), - ]) - ) + ]), + ), + [], ), ], [ 'callable(): int[][][]', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new ArrayTypeNode(new ArrayTypeNode(new IdentifierTypeNode('int'))))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new ArrayTypeNode(new ArrayTypeNode(new IdentifierTypeNode('int')))), []), ], [ 'callable(): (int[][][])', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new ArrayTypeNode(new ArrayTypeNode(new IdentifierTypeNode('int'))))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new ArrayTypeNode(new ArrayTypeNode(new IdentifierTypeNode('int')))), []), ], [ '(callable(): int[])[][]', new ArrayTypeNode( new ArrayTypeNode( - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new IdentifierTypeNode('int'))) - ) + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new IdentifierTypeNode('int')), []), + ), ), ], [ 'callable(): $this', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ThisTypeNode()), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ThisTypeNode(), []), ], [ 'callable(): $this[]', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new ThisTypeNode())), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new ThisTypeNode()), []), ], [ '2.5|3', @@ -2046,29 +2931,29 @@ public function provideParseData(): array ], [ 'callable(): 3.5', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ConstTypeNode(new ConstExprFloatNode('3.5'))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ConstTypeNode(new ConstExprFloatNode('3.5')), []), ], [ 'callable(): 3.5[]', new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode( - new ConstTypeNode(new ConstExprFloatNode('3.5')) - )), + new ConstTypeNode(new ConstExprFloatNode('3.5')), + ), []), ], [ 'callable(): Foo', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new IdentifierTypeNode('Foo')), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new IdentifierTypeNode('Foo'), []), ], [ 'callable(): (Foo)[]', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new IdentifierTypeNode('Foo'))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new IdentifierTypeNode('Foo')), []), ], [ 'callable(): Foo::BAR', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ConstTypeNode(new ConstFetchNode('Foo', 'BAR'))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ConstTypeNode(new ConstFetchNode('Foo', 'BAR')), []), ], [ 'callable(): Foo::*', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ConstTypeNode(new ConstFetchNode('Foo', '*'))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ConstTypeNode(new ConstFetchNode('Foo', '*')), []), ], [ '?Foo[]', @@ -2076,11 +2961,11 @@ public function provideParseData(): array ], [ 'callable(): ?Foo', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new NullableTypeNode(new IdentifierTypeNode('Foo'))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new NullableTypeNode(new IdentifierTypeNode('Foo')), []), ], [ 'callable(): ?Foo[]', - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new NullableTypeNode(new ArrayTypeNode(new IdentifierTypeNode('Foo')))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new NullableTypeNode(new ArrayTypeNode(new IdentifierTypeNode('Foo'))), []), ], [ '?(Foo|Bar)', @@ -2139,6 +3024,23 @@ public function provideParseData(): array ]), ]), ], + [ + 'Closure(Container):($serviceId is class-string ? TService : mixed)', + new CallableTypeNode(new IdentifierTypeNode('Closure'), [ + new CallableTypeParameterNode(new IdentifierTypeNode('Container'), false, false, '', false), + ], new ConditionalTypeForParameterNode( + '$serviceId', + new GenericTypeNode(new IdentifierTypeNode('class-string'), [new IdentifierTypeNode('TService')], ['invariant']), + new IdentifierTypeNode('TService'), + new IdentifierTypeNode('mixed'), + false, + ), []), + ], + [ + 'MongoCollection

Returns a collection object representing the new collection.

', + new IdentifierTypeNode('MongoCollection'), + Lexer::TOKEN_OPEN_ANGLE_BRACKET, + ], ]; } @@ -2151,9 +3053,7 @@ public function dataLinesAndIndexes(): iterable 'int | object{foo: int}[]', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'int | object{foo: int}[]', 1, 1, @@ -2161,9 +3061,7 @@ static function (TypeNode $typeNode): TypeNode { 12, ], [ - static function (UnionTypeNode $typeNode): TypeNode { - return $typeNode->types[0]; - }, + static fn (UnionTypeNode $typeNode): TypeNode => $typeNode->types[0], 'int', 1, 1, @@ -2171,9 +3069,7 @@ static function (UnionTypeNode $typeNode): TypeNode { 0, ], [ - static function (UnionTypeNode $typeNode): TypeNode { - return $typeNode->types[1]; - }, + static fn (UnionTypeNode $typeNode): TypeNode => $typeNode->types[1], 'object{foo: int}[]', 1, 1, @@ -2187,9 +3083,7 @@ static function (UnionTypeNode $typeNode): TypeNode { 'int | object{foo: int}[] ', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'int | object{foo: int}[]', 1, 1, @@ -2197,9 +3091,7 @@ static function (TypeNode $typeNode): TypeNode { 12, ], [ - static function (UnionTypeNode $typeNode): TypeNode { - return $typeNode->types[0]; - }, + static fn (UnionTypeNode $typeNode): TypeNode => $typeNode->types[0], 'int', 1, 1, @@ -2207,9 +3099,7 @@ static function (UnionTypeNode $typeNode): TypeNode { 0, ], [ - static function (UnionTypeNode $typeNode): TypeNode { - return $typeNode->types[1]; - }, + static fn (UnionTypeNode $typeNode): TypeNode => $typeNode->types[1], 'object{foo: int}[]', 1, 1, @@ -2226,9 +3116,7 @@ static function (UnionTypeNode $typeNode): TypeNode { }', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'array{ a: int, b: string @@ -2245,9 +3133,7 @@ static function (TypeNode $typeNode): TypeNode { 'callable(Foo, Bar): void', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'callable(Foo, Bar): void', 1, 1, @@ -2255,9 +3141,7 @@ static function (TypeNode $typeNode): TypeNode { 9, ], [ - static function (CallableTypeNode $typeNode): TypeNode { - return $typeNode->identifier; - }, + static fn (CallableTypeNode $typeNode): TypeNode => $typeNode->identifier, 'callable', 1, 1, @@ -2265,9 +3149,7 @@ static function (CallableTypeNode $typeNode): TypeNode { 0, ], [ - static function (CallableTypeNode $typeNode): Node { - return $typeNode->parameters[0]; - }, + static fn (CallableTypeNode $typeNode): Node => $typeNode->parameters[0], 'Foo', 1, 1, @@ -2275,9 +3157,7 @@ static function (CallableTypeNode $typeNode): Node { 2, ], [ - static function (CallableTypeNode $typeNode): TypeNode { - return $typeNode->returnType; - }, + static fn (CallableTypeNode $typeNode): TypeNode => $typeNode->returnType, 'void', 1, 1, @@ -2291,9 +3171,7 @@ static function (CallableTypeNode $typeNode): TypeNode { '$this', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, '$this', 1, 1, @@ -2307,9 +3185,7 @@ static function (TypeNode $typeNode): TypeNode { 'array{foo: int}', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'array{foo: int}', 1, 1, @@ -2317,9 +3193,7 @@ static function (TypeNode $typeNode): TypeNode { 6, ], [ - static function (ArrayShapeNode $typeNode): TypeNode { - return $typeNode->items[0]; - }, + static fn (ArrayShapeNode $typeNode): Node => $typeNode->items[0], 'foo: int', 1, 1, @@ -2333,9 +3207,7 @@ static function (ArrayShapeNode $typeNode): TypeNode { 'array{}', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'array{}', 1, 1, @@ -2349,9 +3221,7 @@ static function (TypeNode $typeNode): TypeNode { 'object{foo: int}', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'object{foo: int}', 1, 1, @@ -2359,9 +3229,7 @@ static function (TypeNode $typeNode): TypeNode { 6, ], [ - static function (ObjectShapeNode $typeNode): TypeNode { - return $typeNode->items[0]; - }, + static fn (ObjectShapeNode $typeNode): Node => $typeNode->items[0], 'foo: int', 1, 1, @@ -2375,9 +3243,7 @@ static function (ObjectShapeNode $typeNode): TypeNode { 'object{}', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'object{}', 1, 1, @@ -2391,9 +3257,7 @@ static function (TypeNode $typeNode): TypeNode { 'object{}[]', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'object{}[]', 1, 1, @@ -2407,9 +3271,7 @@ static function (TypeNode $typeNode): TypeNode { 'int[][][]', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'int[][][]', 1, 1, @@ -2417,9 +3279,7 @@ static function (TypeNode $typeNode): TypeNode { 6, ], [ - static function (ArrayTypeNode $typeNode): TypeNode { - return $typeNode->type; - }, + static fn (ArrayTypeNode $typeNode): TypeNode => $typeNode->type, 'int[][]', 1, 1, @@ -2464,9 +3324,7 @@ static function (ArrayTypeNode $typeNode): TypeNode { 'int[foo][bar][baz]', [ [ - static function (TypeNode $typeNode): TypeNode { - return $typeNode; - }, + static fn (TypeNode $typeNode): TypeNode => $typeNode, 'int[foo][bar][baz]', 1, 1, @@ -2474,9 +3332,7 @@ static function (TypeNode $typeNode): TypeNode { 9, ], [ - static function (OffsetAccessTypeNode $typeNode): TypeNode { - return $typeNode->type; - }, + static fn (OffsetAccessTypeNode $typeNode): TypeNode => $typeNode->type, 'int[foo][bar]', 1, 1, @@ -2484,9 +3340,7 @@ static function (OffsetAccessTypeNode $typeNode): TypeNode { 6, ], [ - static function (OffsetAccessTypeNode $typeNode): TypeNode { - return $typeNode->offset; - }, + static fn (OffsetAccessTypeNode $typeNode): TypeNode => $typeNode->offset, 'baz', 1, 1, @@ -2567,11 +3421,11 @@ public function testLinesAndIndexes(string $input, array $assertions): void { $tokensArray = $this->lexer->tokenize($input); $tokens = new TokenIterator($tokensArray); - $usedAttributes = [ + $config = new ParserConfig([ 'lines' => true, 'indexes' => true, - ]; - $typeParser = new TypeParser(new ConstExprParser(true, true), true, $usedAttributes); + ]); + $typeParser = new TypeParser($config, new ConstExprParser($config)); $typeNode = $typeParser->parse($tokens); foreach ($assertions as [$callable, $expectedContent, $startLine, $endLine, $startIndex, $endIndex]) { diff --git a/tests/PHPStan/Printer/DifferTest.php b/tests/PHPStan/Printer/DifferTest.php index b06f1692..cca1f61c 100644 --- a/tests/PHPStan/Printer/DifferTest.php +++ b/tests/PHPStan/Printer/DifferTest.php @@ -44,9 +44,7 @@ private function formatDiffString(array $diff): string */ public function testDiff(string $oldStr, string $newStr, string $expectedDiffStr): void { - $differ = new Differ(static function ($a, $b) { - return $a === $b; - }); + $differ = new Differ(static fn ($a, $b) => $a === $b); $diff = $differ->diff(str_split($oldStr), str_split($newStr)); $this->assertSame($expectedDiffStr, $this->formatDiffString($diff)); } @@ -72,9 +70,7 @@ public function provideTestDiff(): array */ public function testDiffWithReplacements(string $oldStr, string $newStr, string $expectedDiffStr): void { - $differ = new Differ(static function ($a, $b) { - return $a === $b; - }); + $differ = new Differ(static fn ($a, $b) => $a === $b); $diff = $differ->diffWithReplacements(str_split($oldStr), str_split($newStr)); $this->assertSame($expectedDiffStr, $this->formatDiffString($diff)); } diff --git a/tests/PHPStan/Printer/IntegrationPrinterWithPhpParserTest.php b/tests/PHPStan/Printer/IntegrationPrinterWithPhpParserTest.php index 74d05072..045156e8 100644 --- a/tests/PHPStan/Printer/IntegrationPrinterWithPhpParserTest.php +++ b/tests/PHPStan/Printer/IntegrationPrinterWithPhpParserTest.php @@ -2,13 +2,15 @@ namespace PHPStan\PhpDocParser\Printer; +use LogicException; use PhpParser\Comment\Doc; -use PhpParser\Lexer\Emulative; +use PhpParser\Internal\TokenStream; use PhpParser\Node as PhpNode; use PhpParser\NodeTraverser as PhpParserNodeTraverser; use PhpParser\NodeVisitor\CloningVisitor as PhpParserCloningVisitor; use PhpParser\NodeVisitorAbstract; -use PhpParser\Parser\Php7; +use PhpParser\ParserFactory; +use PhpParser\PrettyPrinter\Standard; use PHPStan\PhpDocParser\Ast\AbstractNodeVisitor; use PHPStan\PhpDocParser\Ast\Node; use PHPStan\PhpDocParser\Ast\NodeTraverser; @@ -22,12 +24,16 @@ use PHPStan\PhpDocParser\Parser\PhpDocParser; use PHPStan\PhpDocParser\Parser\TokenIterator; use PHPStan\PhpDocParser\Parser\TypeParser; +use PHPStan\PhpDocParser\ParserConfig; use PHPUnit\Framework\TestCase; use function file_get_contents; +use function str_repeat; class IntegrationPrinterWithPhpParserTest extends TestCase { + private const TAB_WIDTH = 4; + /** * @return iterable */ @@ -42,7 +48,8 @@ public function enterNode(Node $node) new IdentifierTypeNode('Bar'), false, '$b', - '' + '', + false, )); } return $node; @@ -66,32 +73,31 @@ public function enterNode(Node $node) */ public function testPrint(string $file, string $expectedFile, NodeVisitor $visitor): void { - $lexer = new Emulative([ - 'usedAttributes' => [ - 'comments', - 'startLine', 'endLine', - 'startTokenPos', 'endTokenPos', - ], - ]); - $phpParser = new Php7($lexer); + $phpParserFactory = new ParserFactory(); + $phpParser = $phpParserFactory->createForNewestSupportedVersion(); $phpTraverser = new PhpParserNodeTraverser(); $phpTraverser->addVisitor(new PhpParserCloningVisitor()); - $printer = new PhpPrinter(); $fileContents = file_get_contents($file); if ($fileContents === false) { $this->fail('Could not read ' . $file); } - /** @var PhpNode[] $oldStmts */ $oldStmts = $phpParser->parse($fileContents); - $oldTokens = $lexer->getTokens(); + if ($oldStmts === null) { + throw new LogicException(); + } + $oldTokens = $phpParser->getTokens(); + + $phpTraverserIndent = new PhpParserNodeTraverser(); + $indentDetector = new PhpPrinterIndentationDetectorVisitor(new TokenStream($oldTokens, self::TAB_WIDTH)); + $phpTraverserIndent->addVisitor($indentDetector); + $phpTraverserIndent->traverse($oldStmts); $phpTraverser2 = new PhpParserNodeTraverser(); $phpTraverser2->addVisitor(new class ($visitor) extends NodeVisitorAbstract { - /** @var NodeVisitor */ - private $visitor; + private NodeVisitor $visitor; public function __construct(NodeVisitor $visitor) { @@ -106,16 +112,14 @@ public function enterNode(PhpNode $phpNode) $phpDoc = $phpNode->getDocComment()->getText(); - $usedAttributes = ['lines' => true, 'indexes' => true]; - $constExprParser = new ConstExprParser(true, true, $usedAttributes); + $config = new ParserConfig(['lines' => true, 'indexes' => true]); + $constExprParser = new ConstExprParser($config); $phpDocParser = new PhpDocParser( - new TypeParser($constExprParser, true, $usedAttributes), + $config, + new TypeParser($config, $constExprParser), $constExprParser, - true, - true, - $usedAttributes ); - $lexer = new Lexer(); + $lexer = new Lexer($config); $tokens = new TokenIterator($lexer->tokenize($phpDoc)); $phpDocNode = $phpDocParser->parse($tokens); $cloningTraverser = new NodeTraverser([new NodeVisitor\CloningVisitor()]); @@ -139,6 +143,7 @@ public function enterNode(PhpNode $phpNode) $newStmts = $phpTraverser->traverse($oldStmts); $newStmts = $phpTraverser2->traverse($newStmts); + $printer = new Standard(['indent' => str_repeat($indentDetector->indentCharacter, $indentDetector->indentSize)]); $newCode = $printer->printFormatPreserving($newStmts, $oldStmts, $oldTokens); $this->assertStringEqualsFile($expectedFile, $newCode); } diff --git a/tests/PHPStan/Printer/PhpPrinter.php b/tests/PHPStan/Printer/PhpPrinter.php deleted file mode 100644 index d691a981..00000000 --- a/tests/PHPStan/Printer/PhpPrinter.php +++ /dev/null @@ -1,60 +0,0 @@ -indentCharacter = ' '; - $this->indentSize = 4; - } - - protected function preprocessNodes(array $nodes): void - { - parent::preprocessNodes($nodes); - if ($this->origTokens === null) { - return; - } - - $traverser = new NodeTraverser(); - - $visitor = new PhpPrinterIndentationDetectorVisitor($this->origTokens); - $traverser->addVisitor($visitor); - $traverser->traverse($nodes); - - $this->indentCharacter = $visitor->indentCharacter; - $this->indentSize = $visitor->indentSize; - } - - protected function setIndentLevel(int $level): void - { - $this->indentLevel = $level; - $this->nl = "\n" . str_repeat($this->indentCharacter, $level); - } - - protected function indent(): void - { - $this->indentLevel += $this->indentSize; - $this->nl = "\n" . str_repeat($this->indentCharacter, $this->indentLevel); - } - - protected function outdent(): void - { - $this->indentLevel -= $this->indentSize; - $this->nl = "\n" . str_repeat($this->indentCharacter, $this->indentLevel); - } - -} diff --git a/tests/PHPStan/Printer/PhpPrinterIndentationDetectorVisitor.php b/tests/PHPStan/Printer/PhpPrinterIndentationDetectorVisitor.php index 9e2b9248..49987eb3 100644 --- a/tests/PHPStan/Printer/PhpPrinterIndentationDetectorVisitor.php +++ b/tests/PHPStan/Printer/PhpPrinterIndentationDetectorVisitor.php @@ -4,8 +4,8 @@ use PhpParser\Internal\TokenStream; use PhpParser\Node; +use PhpParser\NodeVisitor; use PhpParser\NodeVisitorAbstract; -use PHPStan\PhpDocParser\Ast\NodeTraverser; use function count; use function preg_match; use function preg_match_all; @@ -16,14 +16,11 @@ class PhpPrinterIndentationDetectorVisitor extends NodeVisitorAbstract { - /** @var string */ - public $indentCharacter = ' '; + public string $indentCharacter = ' '; - /** @var int */ - public $indentSize = 4; + public int $indentSize = 4; - /** @var TokenStream */ - private $origTokens; + private TokenStream $origTokens; public function __construct(TokenStream $origTokens) { @@ -74,7 +71,7 @@ public function enterNode(Node $node) $this->indentCharacter = $char; $this->indentSize = $size; - return NodeTraverser::STOP_TRAVERSAL; + return NodeVisitor::STOP_TRAVERSAL; } return null; diff --git a/tests/PHPStan/Printer/PrinterTest.php b/tests/PHPStan/Printer/PrinterTest.php index d0a9759d..464b7234 100644 --- a/tests/PHPStan/Printer/PrinterTest.php +++ b/tests/PHPStan/Printer/PrinterTest.php @@ -4,23 +4,33 @@ use PHPStan\PhpDocParser\Ast\AbstractNodeVisitor; use PHPStan\PhpDocParser\Ast\Attribute; +use PHPStan\PhpDocParser\Ast\Comment; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprArrayItemNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprArrayNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprIntegerNode; +use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprStringNode; use PHPStan\PhpDocParser\Ast\ConstExpr\ConstFetchNode; -use PHPStan\PhpDocParser\Ast\ConstExpr\QuoteAwareConstExprStringNode; use PHPStan\PhpDocParser\Ast\Node; use PHPStan\PhpDocParser\Ast\NodeTraverser; use PHPStan\PhpDocParser\Ast\NodeVisitor; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineAnnotation; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArgument; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArray; +use PHPStan\PhpDocParser\Ast\PhpDoc\Doctrine\DoctrineArrayItem; use PHPStan\PhpDocParser\Ast\PhpDoc\MethodTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\ParamClosureThisTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\ParamImmediatelyInvokedCallableTagValueNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\ParamLaterInvokedCallableTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ParamTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocNode; use PHPStan\PhpDocParser\Ast\PhpDoc\PhpDocTagNode; +use PHPStan\PhpDocParser\Ast\PhpDoc\PureUnlessCallableIsImpureTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\ReturnTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\TemplateTagValueNode; use PHPStan\PhpDocParser\Ast\PhpDoc\TypeAliasImportTagValueNode; use PHPStan\PhpDocParser\Ast\Type\ArrayShapeItemNode; use PHPStan\PhpDocParser\Ast\Type\ArrayShapeNode; +use PHPStan\PhpDocParser\Ast\Type\ArrayShapeUnsealedTypeNode; use PHPStan\PhpDocParser\Ast\Type\ArrayTypeNode; use PHPStan\PhpDocParser\Ast\Type\CallableTypeNode; use PHPStan\PhpDocParser\Ast\Type\CallableTypeParameterNode; @@ -39,33 +49,41 @@ use PHPStan\PhpDocParser\Parser\PhpDocParser; use PHPStan\PhpDocParser\Parser\TokenIterator; use PHPStan\PhpDocParser\Parser\TypeParser; +use PHPStan\PhpDocParser\ParserConfig; use PHPUnit\Framework\TestCase; +use function array_map; use function array_pop; +use function array_slice; use function array_splice; use function array_unshift; use function array_values; +use function assert; use function count; +use function implode; +use function preg_match; +use function preg_replace_callback; +use function preg_split; +use function str_repeat; +use function str_replace; +use function strlen; +use const PHP_EOL; class PrinterTest extends TestCase { - /** @var TypeParser */ - private $typeParser; + private TypeParser $typeParser; - /** @var PhpDocParser */ - private $phpDocParser; + private PhpDocParser $phpDocParser; protected function setUp(): void { - $usedAttributes = ['lines' => true, 'indexes' => true]; - $constExprParser = new ConstExprParser(true, true, $usedAttributes); - $this->typeParser = new TypeParser($constExprParser, true, $usedAttributes); + $config = new ParserConfig(['lines' => true, 'indexes' => true, 'comments' => true]); + $constExprParser = new ConstExprParser($config); + $this->typeParser = new TypeParser($config, $constExprParser); $this->phpDocParser = new PhpDocParser( + $config, $this->typeParser, $constExprParser, - true, - true, - $usedAttributes ); } @@ -78,6 +96,7 @@ public function dataPrintFormatPreserving(): iterable }; yield ['/** */', '/** */', $noopVisitor]; + yield ['/** @api */', '/** @api */', $noopVisitor]; yield ['/** */', '/** */', $noopVisitor]; @@ -87,12 +106,14 @@ public function dataPrintFormatPreserving(): iterable $noopVisitor, ]; yield [ - '/** - * @param Foo $foo - */', - '/** - * @param Foo $foo - */', + self::nowdoc(' + /** + * @param Foo $foo + */'), + self::nowdoc(' + /** + * @param Foo $foo + */'), $noopVisitor, ]; @@ -134,33 +155,39 @@ public function enterNode(Node $node) ]; yield [ - '/** - * @param Foo $foo - */', - '/** - */', + self::nowdoc(' + /** + * @param Foo $foo + */'), + self::nowdoc(' + /** + */'), $removeFirst, ]; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - */', - '/** - * @param Bar $bar - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @param Bar $bar + */'), $removeFirst, ]; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - */', - '/** - * @param Bar $bar - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @param Bar $bar + */'), $removeFirst, ]; @@ -180,13 +207,15 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - */', - '/** - * @param Foo $foo - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @param Foo $foo + */'), $removeLast, ]; @@ -207,39 +236,45 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - */', - '/** - * @param Foo $foo - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @param Foo $foo + */'), $removeSecond, ]; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - * @param Baz $baz - */', - '/** - * @param Foo $foo - * @param Baz $baz - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + * @param Baz $baz + */'), + self::nowdoc(' + /** + * @param Foo $foo + * @param Baz $baz + */'), $removeSecond, ]; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - * @param Baz $baz - */', - '/** - * @param Foo $foo - * @param Baz $baz - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + * @param Baz $baz + */'), + self::nowdoc(' + /** + * @param Foo $foo + * @param Baz $baz + */'), $removeSecond, ]; @@ -271,58 +306,66 @@ public function enterNode(Node $node) ]; yield [ - '/** -* @return Foo -* @param Foo $foo -* @param Bar $bar -*/', - '/** -* @return Bar -* @param Foo $foo -* @param Bar $bar -*/', + self::nowdoc(' + /** + * @return Foo + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @return Bar + * @param Foo $foo + * @param Bar $bar + */'), $changeReturnType, ]; yield [ - '/** -* @param Foo $foo -* @return Foo -* @param Bar $bar -*/', - '/** -* @param Foo $foo -* @return Bar -* @param Bar $bar -*/', + self::nowdoc(' + /** + * @param Foo $foo + * @return Foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @param Foo $foo + * @return Bar + * @param Bar $bar + */'), $changeReturnType, ]; yield [ - '/** -* @return Foo -* @param Foo $foo -* @param Bar $bar -*/', - '/** -* @return Bar -* @param Foo $foo -* @param Bar $bar -*/', + self::nowdoc(' + /** + * @return Foo + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @return Bar + * @param Foo $foo + * @param Bar $bar + */'), $changeReturnType, ]; yield [ - '/** -* @param Foo $foo Foo description -* @return Foo Foo return description -* @param Bar $bar Bar description -*/', - '/** -* @param Foo $foo Foo description -* @return Bar Foo return description -* @param Bar $bar Bar description -*/', + self::nowdoc(' + /** + * @param Foo $foo Foo description + * @return Foo Foo return description + * @param Bar $bar Bar description + */'), + self::nowdoc(' + /** + * @param Foo $foo Foo description + * @return Bar Foo return description + * @param Bar $bar Bar description + */'), $changeReturnType, ]; @@ -331,7 +374,7 @@ public function enterNode(Node $node) public function enterNode(Node $node) { if ($node instanceof PhpDocNode) { - $node->children[0] = new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('Baz'), false, '$a', '')); + $node->children[0] = new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('Baz'), false, '$a', '', false)); return $node; } @@ -347,22 +390,26 @@ public function enterNode(Node $node) ]; yield [ - '/** - * @param Foo $foo - */', - '/** - * @param Baz $a - */', + self::nowdoc(' + /** + * @param Foo $foo + */'), + self::nowdoc(' + /** + * @param Baz $a + */'), $replaceFirst, ]; yield [ - '/** - * @param Foo $foo - */', - '/** - * @param Baz $a - */', + self::nowdoc(' + /** + * @param Foo $foo + */'), + self::nowdoc(' + /** + * @param Baz $a + */'), $replaceFirst, ]; @@ -371,7 +418,7 @@ public function enterNode(Node $node) public function enterNode(Node $node) { if ($node instanceof PhpDocNode) { - array_unshift($node->children, new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('Baz'), false, '$a', ''))); + array_unshift($node->children, new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('Baz'), false, '$a', '', false))); return $node; } @@ -382,24 +429,28 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param Foo $foo - */', - '/** - * @param Baz $a - * @param Foo $foo - */', + self::nowdoc(' + /** + * @param Foo $foo + */'), + self::nowdoc(' + /** + * @param Baz $a + * @param Foo $foo + */'), $insertFirst, ]; yield [ - '/** - * @param Foo $foo - */', - '/** - * @param Baz $a - * @param Foo $foo - */', + self::nowdoc(' + /** + * @param Foo $foo + */'), + self::nowdoc(' + /** + * @param Baz $a + * @param Foo $foo + */'), $insertFirst, ]; @@ -409,7 +460,7 @@ public function enterNode(Node $node) { if ($node instanceof PhpDocNode) { array_splice($node->children, 1, 0, [ - new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('Baz'), false, '$a', '')), + new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('Baz'), false, '$a', '', false)), ]); return $node; @@ -421,52 +472,60 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param Foo $foo - */', - '/** - * @param Foo $foo - * @param Baz $a - */', + self::nowdoc(' + /** + * @param Foo $foo + */'), + self::nowdoc(' + /** + * @param Foo $foo + * @param Baz $a + */'), $insertSecond, ]; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - */', - '/** - * @param Foo $foo - * @param Baz $a - * @param Bar $bar - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @param Foo $foo + * @param Baz $a + * @param Bar $bar + */'), $insertSecond, ]; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - */', - '/** - * @param Foo $foo - * @param Baz $a - * @param Bar $bar - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @param Foo $foo + * @param Baz $a + * @param Bar $bar + */'), $insertSecond, ]; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - */', - '/** - * @param Foo $foo - * @param Baz $a - * @param Bar $bar - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @param Foo $foo + * @param Baz $a + * @param Bar $bar + */'), $insertSecond, ]; @@ -475,7 +534,7 @@ public function enterNode(Node $node) public function enterNode(Node $node) { if ($node instanceof PhpDocNode) { - $node->children[count($node->children) - 1] = new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('Baz'), false, '$a', '')); + $node->children[count($node->children) - 1] = new PhpDocTagNode('@param', new ParamTagValueNode(new IdentifierTypeNode('Baz'), false, '$a', '', false)); return $node; } @@ -485,24 +544,28 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param Foo $foo - */', - '/** - * @param Baz $a - */', + self::nowdoc(' + /** + * @param Foo $foo + */'), + self::nowdoc(' + /** + * @param Baz $a + */'), $replaceLast, ]; yield [ - '/** - * @param Foo $foo - * @param Bar $bar - */', - '/** - * @param Foo $foo - * @param Baz $a - */', + self::nowdoc(' + /** + * @param Foo $foo + * @param Bar $bar + */'), + self::nowdoc(' + /** + * @param Foo $foo + * @param Baz $a + */'), $replaceLast, ]; @@ -520,24 +583,28 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param Bar|Baz $foo - */', - '/** - * @param Foo|Bar|Baz $foo - */', + self::nowdoc(' + /** + * @param Bar|Baz $foo + */'), + self::nowdoc(' + /** + * @param Foo|Bar|Baz $foo + */'), $insertFirstTypeInUnionType, ]; yield [ - '/** - * @param Bar|Baz $foo - * @param Foo $bar - */', - '/** - * @param Foo|Bar|Baz $foo - * @param Foo $bar - */', + self::nowdoc(' + /** + * @param Bar|Baz $foo + * @param Foo $bar + */'), + self::nowdoc(' + /** + * @param Foo|Bar|Baz $foo + * @param Foo $bar + */'), $insertFirstTypeInUnionType, ]; @@ -558,12 +625,14 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param Foo|Bar $bar - */', - '/** - * @param Lorem|Ipsum $bar - */', + self::nowdoc(' + /** + * @param Foo|Bar $bar + */'), + self::nowdoc(' + /** + * @param Lorem|Ipsum $bar + */'), $replaceTypesInUnionType, ]; @@ -583,13 +652,44 @@ public function enterNode(Node $node) }; + $addCallableTemplateType = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof CallableTypeNode) { + $node->templateTypes[] = new TemplateTagValueNode( + 'T', + new IdentifierTypeNode('int'), + '', + ); + } + + return $node; + } + + }; + yield [ - '/** - * @param callable(): void $cb - */', - '/** - * @param callable(Foo $foo, Bar $bar): void $cb - */', + '/** @var Closure(): T */', + '/** @var Closure(): T */', + $addCallableTemplateType, + ]; + + yield [ + '/** @var \Closure(U): T */', + '/** @var \Closure(U): T */', + $addCallableTemplateType, + ]; + + yield [ + self::nowdoc(' + /** + * @param callable(): void $cb + */'), + self::nowdoc(' + /** + * @param callable(Foo $foo, Bar $bar): void $cb + */'), $replaceParametersInCallableType, ]; @@ -607,12 +707,14 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param callable(Foo $foo, Bar $bar): void $cb - */', - '/** - * @param callable(): void $cb - */', + self::nowdoc(' + /** + * @param callable(Foo $foo, Bar $bar): void $cb + */'), + self::nowdoc(' + /** + * @param callable(): void $cb + */'), $removeParametersInCallableType, ]; @@ -630,14 +732,16 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param callable(Foo $foo, Bar $bar): void $cb - * @param callable(): void $cb2 - */', - '/** - * @param Closure(Foo $foo, Bar $bar): void $cb - * @param Closure(): void $cb2 - */', + self::nowdoc(' + /** + * @param callable(Foo $foo, Bar $bar): void $cb + * @param callable(): void $cb2 + */'), + self::nowdoc(' + /** + * @param Closure(Foo $foo, Bar $bar): void $cb + * @param Closure(): void $cb2 + */'), $changeCallableTypeIdentifier, ]; @@ -668,120 +772,134 @@ public function enterNode(Node $node) ]; yield [ - '/** - * @return array{float, Foo} - */', - '/** - * @return array{float, int, Foo, string} - */', + self::nowdoc(' + /** + * @return array{float, Foo} + */'), + self::nowdoc(' + /** + * @return array{float, int, Foo, string} + */'), $addItemsToArrayShape, ]; yield [ - '/** - * @return array{ - * float, - * Foo, - * } - */', - '/** - * @return array{ - * float, - * int, - * Foo, - * string, - * } - */', + self::nowdoc(' + /** + * @return array{ + * float, + * Foo, + * } + */'), + self::nowdoc(' + /** + * @return array{ + * float, + * int, + * Foo, + * string, + * } + */'), $addItemsToArrayShape, ]; yield [ - '/** - * @return array{ - * float, - * Foo - * } - */', - '/** - * @return array{ - * float, - * int, - * Foo, - * string - * } - */', + self::nowdoc(' + /** + * @return array{ + * float, + * Foo + * } + */'), + self::nowdoc(' + /** + * @return array{ + * float, + * int, + * Foo, + * string + * } + */'), $addItemsToArrayShape, ]; yield [ - '/** - * @return array{ - * float, - * Foo - * } - */', - '/** - * @return array{ - * float, - * int, - * Foo, - * string - * } - */', + self::nowdoc(' + /** + * @return array{ + * float, + * Foo + * } + */'), + self::nowdoc(' + /** + * @return array{ + * float, + * int, + * Foo, + * string + * } + */'), $addItemsToArrayShape, ]; yield [ - '/** - * @return array{ - * float, - * Foo, - * } - */', - '/** - * @return array{ - * float, - * int, - * Foo, - * string, - * } - */', + self::nowdoc(' + /** + * @return array{ + * float, + * Foo, + * } + */'), + self::nowdoc(' + /** + * @return array{ + * float, + * int, + * Foo, + * string, + * } + */'), $addItemsToArrayShape, ]; yield [ - '/** - * @return array{ - * float, - * Foo - * } - */', - '/** - * @return array{ - * float, - * int, - * Foo, - * string - * } - */', + self::nowdoc(' + /** + * @return array{ + * float, + * Foo + * } + */'), + self::nowdoc(' + /** + * @return array{ + * float, + * int, + * Foo, + * string + * } + */'), $addItemsToArrayShape, ]; yield [ - '/** - * @return array{ - * float, - * Foo - * } - */', - '/** - * @return array{ - * float, - * int, - * Foo, - * string - * } - */', + self::nowdoc(' + /** + * @return array{ + * float, + * Foo + * } + */'), + self::nowdoc(' + /** + * @return array{ + * float, + * int, + * Foo, + * string + * } + */'), $addItemsToArrayShape, ]; @@ -799,25 +917,254 @@ public function enterNode(Node $node) }; yield [ - '/** - * @return object{} - */', - '/** - * @return object{foo: int} - */', + self::nowdoc(' + /** + * @return object{} + */'), + self::nowdoc(' + /** + * @return object{foo: int} + */'), $addItemsToObjectShape, ]; yield [ - '/** - * @return object{bar: string} - */', - '/** - * @return object{bar: string, foo: int} - */', + self::nowdoc(' + /** + * @return object{bar: string} + */'), + self::nowdoc(' + /** + * @return object{bar: string, foo: int} + */'), + $addItemsToObjectShape, + ]; + + yield [ + self::nowdoc(' + /** + * @return object{bar: string} + */'), + self::nowdoc(' + /** + * @return object{bar: string, foo: int} + */'), $addItemsToObjectShape, ]; + $addItemsWithCommentsToMultilineArrayShape = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + $commentedNode = new ArrayShapeItemNode(new IdentifierTypeNode('b'), false, new IdentifierTypeNode('int')); + $commentedNode->setAttribute(Attribute::COMMENTS, [new Comment('// bar')]); + array_splice($node->items, 1, 0, [ + $commentedNode, + ]); + $commentedNode = new ArrayShapeItemNode(new IdentifierTypeNode('d'), false, new IdentifierTypeNode('string')); + $commentedNode->setAttribute(Attribute::COMMENTS, [new Comment( + PrinterTest::nowdoc(' + // first comment'), + )]); + $node->items[] = $commentedNode; + + $commentedNode = new ArrayShapeItemNode(new IdentifierTypeNode('e'), false, new IdentifierTypeNode('string')); + $commentedNode->setAttribute(Attribute::COMMENTS, [new Comment( + PrinterTest::nowdoc(' + // second comment'), + )]); + $node->items[] = $commentedNode; + + $commentedNode = new ArrayShapeItemNode(new IdentifierTypeNode('f'), false, new IdentifierTypeNode('string')); + $commentedNode->setAttribute(Attribute::COMMENTS, [ + new Comment('// third comment'), + new Comment('// fourth comment'), + ]); + $node->items[] = $commentedNode; + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @return array{ + * // foo + * a: int, + * c: string + * } + */'), + self::nowdoc(' + /** + * @return array{ + * // foo + * a: int, + * // bar + * b: int, + * c: string, + * // first comment + * d: string, + * // second comment + * e: string, + * // third comment + * // fourth comment + * f: string + * } + */'), + $addItemsWithCommentsToMultilineArrayShape, + ]; + + $prependItemsWithCommentsToMultilineArrayShape = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + $commentedNode = new ArrayShapeItemNode(new IdentifierTypeNode('a'), false, new IdentifierTypeNode('int')); + $commentedNode->setAttribute(Attribute::COMMENTS, [new Comment('// first item')]); + array_splice($node->items, 0, 0, [ + $commentedNode, + ]); + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @return array{ + * b: int, + * } + */'), + self::nowdoc(' + /** + * @return array{ + * // first item + * a: int, + * b: int, + * } + */'), + $prependItemsWithCommentsToMultilineArrayShape, + ]; + + $changeCommentOnArrayShapeItem = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeItemNode) { + $node->setAttribute(Attribute::COMMENTS, [new Comment('// puppies')]); + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @return array{ + * a: int, + * } + */'), + self::nowdoc(' + /** + * @return array{ + * // puppies + * a: int, + * } + */'), + $changeCommentOnArrayShapeItem, + ]; + + $addItemsWithCommentsToObjectShape = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ObjectShapeNode) { + $item = new ObjectShapeItemNode(new IdentifierTypeNode('foo'), false, new IdentifierTypeNode('int')); + $item->setAttribute(Attribute::COMMENTS, [new Comment('// favorite foo')]); + $node->items[] = $item; + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @return object{ + * // your favorite bar + * bar: string + * } + */'), + self::nowdoc(' + /** + * @return object{ + * // your favorite bar + * bar: string, + * // favorite foo + * foo: int + * } + */'), + $addItemsWithCommentsToObjectShape, + ]; + + $removeItemWithComment = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if (!$node instanceof ArrayShapeNode) { + return null; + } + + foreach ($node->items as $i => $item) { + if ($item->keyName === null) { + continue; + } + + $comments = $item->keyName->getAttribute(Attribute::COMMENTS); + if ($comments === null) { + continue; + } + if ($comments === []) { + continue; + } + + unset($node->items[$i]); + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @return array{ + * a: string, + * // b comment + * b: int, + * } + */'), + self::nowdoc(' + /** + * @return array{ + * a: string, + * } + */'), + $removeItemWithComment, + ]; + $addItemsToConstExprArray = new class extends AbstractNodeVisitor { public function enterNode(Node $node) @@ -906,6 +1253,12 @@ public function enterNode(Node $node) $addTemplateTagBound, ]; + yield [ + '/** @template T super string */', + '/** @template T of int super string */', + $addTemplateTagBound, + ]; + $removeTemplateTagBound = new class extends AbstractNodeVisitor { public function enterNode(Node $node) @@ -925,12 +1278,12 @@ public function enterNode(Node $node) $removeTemplateTagBound, ]; - $addKeyNameToArrayShapeItemNode = new class extends AbstractNodeVisitor { + $addTemplateTagLowerBound = new class extends AbstractNodeVisitor { public function enterNode(Node $node) { - if ($node instanceof ArrayShapeItemNode) { - $node->keyName = new QuoteAwareConstExprStringNode('test', QuoteAwareConstExprStringNode::SINGLE_QUOTED); + if ($node instanceof TemplateTagValueNode) { + $node->lowerBound = new IdentifierTypeNode('int'); } return $node; @@ -939,17 +1292,29 @@ public function enterNode(Node $node) }; yield [ - '/** @return array{Foo} */', - "/** @return array{'test': Foo} */", - $addKeyNameToArrayShapeItemNode, + '/** @template T */', + '/** @template T super int */', + $addTemplateTagLowerBound, ]; - $removeKeyNameToArrayShapeItemNode = new class extends AbstractNodeVisitor { + yield [ + '/** @template T super string */', + '/** @template T super int */', + $addTemplateTagLowerBound, + ]; + + yield [ + '/** @template T of string */', + '/** @template T of string super int */', + $addTemplateTagLowerBound, + ]; + + $removeTemplateTagLowerBound = new class extends AbstractNodeVisitor { public function enterNode(Node $node) { - if ($node instanceof ArrayShapeItemNode) { - $node->keyName = null; + if ($node instanceof TemplateTagValueNode) { + $node->lowerBound = null; } return $node; @@ -958,17 +1323,55 @@ public function enterNode(Node $node) }; yield [ - "/** @return array{'test': Foo} */", - '/** @return array{Foo} */', - $removeKeyNameToArrayShapeItemNode, + '/** @template T super int */', + '/** @template T */', + $removeTemplateTagLowerBound, ]; - $changeArrayShapeKind = new class extends AbstractNodeVisitor { + $addKeyNameToArrayShapeItemNode = new class extends AbstractNodeVisitor { public function enterNode(Node $node) { - if ($node instanceof ArrayShapeNode) { - $node->kind = ArrayShapeNode::KIND_LIST; + if ($node instanceof ArrayShapeItemNode) { + $node->keyName = new ConstExprStringNode('test', ConstExprStringNode::SINGLE_QUOTED); + } + + return $node; + } + + }; + + yield [ + '/** @return array{Foo} */', + "/** @return array{'test': Foo} */", + $addKeyNameToArrayShapeItemNode, + ]; + + $removeKeyNameToArrayShapeItemNode = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeItemNode) { + $node->keyName = null; + } + + return $node; + } + + }; + + yield [ + "/** @return array{'test': Foo} */", + '/** @return array{Foo} */', + $removeKeyNameToArrayShapeItemNode, + ]; + + $changeArrayShapeKind = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + $node->kind = ArrayShapeNode::KIND_LIST; } return $node; @@ -1002,36 +1405,42 @@ public function enterNode(Node $node) ]; yield [ - '/** - * @param int $a - */', - '/** - * @param int $bz - */', + self::nowdoc(' + /** + * @param int $a + */'), + self::nowdoc(' + /** + * @param int $bz + */'), $changeParameterName, ]; yield [ - '/** - * @param int $a - * @return string - */', - '/** - * @param int $bz - * @return string - */', + self::nowdoc(' + /** + * @param int $a + * @return string + */'), + self::nowdoc(' + /** + * @param int $bz + * @return string + */'), $changeParameterName, ]; yield [ - '/** - * @param int $a haha description - * @return string - */', - '/** - * @param int $bz haha description - * @return string - */', + self::nowdoc(' + /** + * @param int $a haha description + * @return string + */'), + self::nowdoc(' + /** + * @param int $bz haha description + * @return string + */'), $changeParameterName, ]; @@ -1067,12 +1476,14 @@ public function enterNode(Node $node) ]; yield [ - '/** - * @param int $a haha - */', - '/** - * @param int $a hehe - */', + self::nowdoc(' + /** + * @param int $a haha + */'), + self::nowdoc(' + /** + * @param int $a hehe + */'), $changeParameterDescription, ]; @@ -1090,12 +1501,14 @@ public function enterNode(Node $node) }; yield [ - '/** - * @param Foo[awesome] $a haha - */', - '/** - * @param Foo[baz] $a haha - */', + self::nowdoc(' + /** + * @param Foo[awesome] $a haha + */'), + self::nowdoc(' + /** + * @param Foo[baz] $a haha + */'), $changeOffsetAccess, ]; @@ -1113,12 +1526,14 @@ public function enterNode(Node $node) }; yield [ - '/** - * @phpstan-import-type TypeAlias from AnotherClass as DifferentAlias - */', - '/** - * @phpstan-import-type TypeAlias from AnotherClass as Ciao - */', + self::nowdoc(' + /** + * @phpstan-import-type TypeAlias from AnotherClass as DifferentAlias + */'), + self::nowdoc(' + /** + * @phpstan-import-type TypeAlias from AnotherClass as Ciao + */'), $changeTypeAliasImportAs, ]; @@ -1136,12 +1551,14 @@ public function enterNode(Node $node) }; yield [ - '/** - * @phpstan-import-type TypeAlias from AnotherClass as DifferentAlias - */', - '/** - * @phpstan-import-type TypeAlias from AnotherClass - */', + self::nowdoc(' + /** + * @phpstan-import-type TypeAlias from AnotherClass as DifferentAlias + */'), + self::nowdoc(' + /** + * @phpstan-import-type TypeAlias from AnotherClass + */'), $removeImportAs, ]; @@ -1153,7 +1570,7 @@ public function enterNode(Node $node) $node->templateTypes[] = new TemplateTagValueNode( 'T', new IdentifierTypeNode('int'), - '' + '', ); } @@ -1174,27 +1591,24 @@ public function enterNode(Node $node) $addMethodTemplateType, ]; - $changeCallableReturnTypeFactory = function (TypeNode $type): NodeVisitor { - return new class ($type) extends AbstractNodeVisitor { - - /** @var TypeNode */ - private $type; + $changeCallableReturnTypeFactory = static fn (TypeNode $type): NodeVisitor => new class ($type) extends AbstractNodeVisitor { - public function __construct(TypeNode $type) - { - $this->type = $type; - } + private TypeNode $type; - public function enterNode(Node $node) - { - if ($node instanceof CallableTypeNode) { - $node->returnType = $this->type; - } + public function __construct(TypeNode $type) + { + $this->type = $type; + } - return $node; + public function enterNode(Node $node) + { + if ($node instanceof CallableTypeNode) { + $node->returnType = $this->type; } - }; + return $node; + } + }; yield [ @@ -1227,44 +1641,38 @@ public function enterNode(Node $node) ])), ]; - $changeCallableReturnTypeCallbackFactory = function (callable $callback): NodeVisitor { - return new class ($callback) extends AbstractNodeVisitor { + $changeCallableReturnTypeCallbackFactory = fn (callable $callback): NodeVisitor => new class ($callback) extends AbstractNodeVisitor { - /** @var callable(TypeNode): TypeNode */ - private $callback; + /** @var callable(TypeNode): TypeNode */ + private $callback; - public function __construct(callable $callback) - { - $this->callback = $callback; - } - - public function enterNode(Node $node) - { - if ($node instanceof CallableTypeNode) { - $cb = $this->callback; - $node->returnType = $cb($node->returnType); - } + public function __construct(callable $callback) + { + $this->callback = $callback; + } - return $node; + public function enterNode(Node $node) + { + if ($node instanceof CallableTypeNode) { + $cb = $this->callback; + $node->returnType = $cb($node->returnType); } - }; + return $node; + } + }; yield [ '/** @param callable(): int $a */', '/** @param callable(): string $a */', - $changeCallableReturnTypeCallbackFactory(static function (TypeNode $typeNode): TypeNode { - return new IdentifierTypeNode('string'); - }), + $changeCallableReturnTypeCallbackFactory(static fn (TypeNode $typeNode) => new IdentifierTypeNode('string')), ]; yield [ '/** @param callable(): (int) $a */', '/** @param callable(): string $a */', - $changeCallableReturnTypeCallbackFactory(static function (TypeNode $typeNode): TypeNode { - return new IdentifierTypeNode('string'); - }), + $changeCallableReturnTypeCallbackFactory(static fn (TypeNode $typeNode) => new IdentifierTypeNode('string')), ]; yield [ @@ -1479,6 +1887,743 @@ public function enterNode(Node $node) }, ]; + + yield [ + '/** @Foo({a: 1}) */', + '/** @Foo({1}) */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof DoctrineArrayItem) { + $node->key = null; + } + + return $node; + } + + }, + ]; + + yield [ + '/** @Foo({a: 1}) */', + '/** @Foo({b: 1}) */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof DoctrineArrayItem) { + $node->key = new IdentifierTypeNode('b'); + } + + return $node; + } + + }, + ]; + + yield [ + '/** @Foo({a = 1}) */', + '/** @Foo({b = 1}) */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof DoctrineArrayItem) { + $node->key = new IdentifierTypeNode('b'); + } + + return $node; + } + + }, + ]; + + yield [ + '/** @Foo() */', + '/** @Foo(1, 2, 3) */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof DoctrineAnnotation) { + $node->arguments = [ + new DoctrineArgument(null, new ConstExprIntegerNode('1')), + new DoctrineArgument(null, new ConstExprIntegerNode('2')), + new DoctrineArgument(null, new ConstExprIntegerNode('3')), + ]; + } + + return $node; + } + + }, + ]; + + yield [ + self::nowdoc(' + /** @Foo( + * 1, + * 2, + * ) + */'), + self::nowdoc(' + /** @Foo( + * 1, + * 2, + * 3, + * ) + */'), + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof DoctrineAnnotation) { + $node->arguments[] = new DoctrineArgument(null, new ConstExprIntegerNode('3')); + } + + return $node; + } + + }, + ]; + + yield [ + '/**' . PHP_EOL . + ' * @X({' . PHP_EOL . + ' * 1,' . PHP_EOL . + ' * 2' . PHP_EOL . + ' * , ' . PHP_EOL . + ' * 3,' . PHP_EOL . + ' * }' . PHP_EOL . + ' * )' . PHP_EOL . + ' */', + '/**' . PHP_EOL . + ' * @X({' . PHP_EOL . + ' * 1,' . PHP_EOL . + ' * 2' . PHP_EOL . + ' * , ' . PHP_EOL . + ' * 3,' . PHP_EOL . + ' * 4,' . PHP_EOL . + ' * }' . PHP_EOL . + ' * )' . PHP_EOL . + ' */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof DoctrineArray) { + $node->items[] = new DoctrineArrayItem(null, new ConstExprIntegerNode('4')); + } + + return $node; + } + + }, + ]; + + yield [ + '/** @Foo() */', + '/** @Bar() */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof PhpDocTagNode) { + $node->name = '@Bar'; + } + if ($node instanceof DoctrineAnnotation) { + $node->name = '@Bar'; + } + + return $node; + } + + }, + ]; + + yield [ + '/** @param-immediately-invoked-callable $foo test */', + '/** @param-immediately-invoked-callable $bar foo */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ParamImmediatelyInvokedCallableTagValueNode) { + $node->parameterName = '$bar'; + $node->description = 'foo'; + } + + return $node; + } + + }, + ]; + + yield [ + '/** @param-later-invoked-callable $foo test */', + '/** @param-later-invoked-callable $bar foo */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ParamLaterInvokedCallableTagValueNode) { + $node->parameterName = '$bar'; + $node->description = 'foo'; + } + + return $node; + } + + }, + ]; + + yield [ + '/** @param-closure-this Foo $test haha */', + '/** @param-closure-this Bar $taste hehe */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ParamClosureThisTagValueNode) { + $node->type = new IdentifierTypeNode('Bar'); + $node->parameterName = '$taste'; + $node->description = 'hehe'; + } + + return $node; + } + + }, + ]; + + yield [ + '/** @pure-unless-callable-is-impure $foo test */', + '/** @pure-unless-callable-is-impure $bar foo */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof PureUnlessCallableIsImpureTagValueNode) { + $node->parameterName = '$bar'; + $node->description = 'foo'; + } + + return $node; + } + + }, + ]; + + yield [ + '/** @return Foo[abc] */', + '/** @return self::FOO[abc] */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ReturnTagValueNode && $node->type instanceof OffsetAccessTypeNode) { + $node->type->type = new ConstTypeNode(new ConstFetchNode('self', 'FOO')); + } + + return $node; + } + + }, + ]; + + yield [ + '/** @return array{foo: int, ...} */', + '/** @return array{foo: int} */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + $node->sealed = true; + } + + return $node; + } + + }, + ]; + + yield [ + '/** @return array{foo: int, ...} */', + '/** @return array{foo: int, ...} */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + $node->unsealedType = new ArrayShapeUnsealedTypeNode(new IdentifierTypeNode('string'), null); + } + + return $node; + } + + }, + ]; + + yield [ + '/** @return array{foo: int, ...} */', + '/** @return array{foo: int, ...} */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + assert($node->unsealedType !== null); + $node->unsealedType->keyType = new IdentifierTypeNode('int'); + } + + return $node; + } + + }, + ]; + + yield [ + '/** @return array{foo: int, ...} */', + '/** @return array{foo: int} */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + $node->sealed = true; + $node->unsealedType = null; + } + + return $node; + } + + }, + ]; + + yield [ + '/** @return list{int, ...} */', + '/** @return list{int} */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + $node->sealed = true; + } + + return $node; + } + + }, + ]; + + yield [ + '/** @return list{int, ...} */', + '/** @return list{int, ...} */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + $node->unsealedType = new ArrayShapeUnsealedTypeNode(new IdentifierTypeNode('string'), null); + } + + return $node; + } + + }, + ]; + + yield [ + '/** @return list{int, ...} */', + '/** @return list{int} */', + new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + $node->sealed = true; + $node->unsealedType = null; + } + + return $node; + } + + }, + ]; + + $singleCommentLineAddFront = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ArrayShapeNode) { + array_unshift($node->items, PrinterTest::withComment( + new ArrayShapeItemNode(null, false, new IdentifierTypeNode('float')), + '// A fractional number', + )); + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @param array{} $foo + */'), + self::nowdoc(' + /** + * @param array{float} $foo + */'), + $singleCommentLineAddFront, + ]; + + yield [ + self::nowdoc(' + /** + * @param array{string} $foo + */'), + self::nowdoc(' + /** + * @param array{// A fractional number + * float, + * string} $foo + */'), + $singleCommentLineAddFront, + ]; + + yield [ + self::nowdoc(' + /** + * @param array{ + * string,int} $foo + */'), + self::nowdoc(' + /** + * @param array{ + * // A fractional number + * float, + * string,int} $foo + */'), + $singleCommentLineAddFront, + ]; + + yield [ + self::nowdoc(' + /** + * @param array{ + * string, + * int + * } $foo + */'), + self::nowdoc(' + /** + * @param array{ + * // A fractional number + * float, + * string, + * int + * } $foo + */'), + $singleCommentLineAddFront, + ]; + + $singleCommentLineAddMiddle = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + $newItem = PrinterTest::withComment( + new ArrayShapeItemNode(null, false, new IdentifierTypeNode('float')), + '// A fractional number', + ); + + if ($node instanceof ArrayShapeNode) { + if (count($node->items) === 0) { + $node->items[] = $newItem; + } else { + array_splice($node->items, 1, 0, [$newItem]); + } + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @param array{} $foo + */'), + self::nowdoc(' + /** + * @param array{float} $foo + */'), + $singleCommentLineAddMiddle, + ]; + + yield [ + self::nowdoc(' + /** + * @param array{string} $foo + */'), + self::nowdoc(' + /** + * @param array{string, + * // A fractional number + * float} $foo + */'), + $singleCommentLineAddMiddle, + ]; + + yield [ + self::nowdoc(' + /** + * @param array{ + * string,int} $foo + */'), + self::nowdoc(' + /** + * @param array{ + * string, + * // A fractional number + * float,int} $foo + */'), + $singleCommentLineAddMiddle, + ]; + + yield [ + self::nowdoc(' + /** + * @param array{ + * string, + * int + * } $foo + */'), + self::nowdoc(' + /** + * @param array{ + * string, + * // A fractional number + * float, + * int + * } $foo + */'), + $singleCommentLineAddMiddle, + ]; + + $addCommentToCallableParamsFront = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof CallableTypeNode) { + array_unshift($node->parameters, PrinterTest::withComment( + new CallableTypeParameterNode(new IdentifierTypeNode('Foo'), false, false, '$foo', false), + '// never pet a burning dog', + )); + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @param callable(Bar $bar): int $a + */'), + self::nowdoc(' + /** + * @param callable(// never pet a burning dog + * Foo $foo, + * Bar $bar): int $a + */'), + $addCommentToCallableParamsFront, + ]; + + $addCommentToCallableParamsMiddle = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof CallableTypeNode) { + $node->parameters[] = PrinterTest::withComment( + new CallableTypeParameterNode(new IdentifierTypeNode('Bar'), false, false, '$bar', false), + '// never pet a burning dog', + ); + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @param callable(Foo $foo): int $a + */'), + self::nowdoc(' + /** + * @param callable(Foo $foo, + * // never pet a burning dog + * Bar $bar): int $a + */'), + $addCommentToCallableParamsMiddle, + ]; + + $addCommentToObjectShapeItemFront = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ObjectShapeNode) { + array_unshift($node->items, PrinterTest::withComment( + new ObjectShapeItemNode(new IdentifierTypeNode('foo'), false, new IdentifierTypeNode('float')), + '// A fractional number', + )); + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @param object{bar: string} $foo + */'), + self::nowdoc(' + /** + * @param object{// A fractional number + * foo: float, + * bar: string} $foo + */'), + $addCommentToObjectShapeItemFront, + ]; + + yield [ + self::nowdoc(' + /** + * @param object{ + * bar:string,naz:int} $foo + */'), + self::nowdoc(' + /** + * @param object{ + * // A fractional number + * foo: float, + * bar:string,naz:int} $foo + */'), + $addCommentToObjectShapeItemFront, + ]; + + yield [ + self::nowdoc(' + /** + * @param object{ + * bar:string, + * naz:int + * } $foo + */'), + self::nowdoc(' + /** + * @param object{ + * // A fractional number + * foo: float, + * bar:string, + * naz:int + * } $foo + */'), + $addCommentToObjectShapeItemFront, + ]; + + $addCommentToObjectShapeItemMiddle = new class extends AbstractNodeVisitor { + + public function enterNode(Node $node) + { + if ($node instanceof ObjectShapeNode) { + $newItem = PrinterTest::withComment( + new ObjectShapeItemNode(new IdentifierTypeNode('bar'), false, new IdentifierTypeNode('float')), + '// A fractional number', + ); + if (count($node->items) === 0) { + $node->items[] = $newItem; + } else { + array_splice($node->items, 1, 0, [$newItem]); + } + } + + return $node; + } + + }; + + yield [ + self::nowdoc(' + /** + * @param object{} $foo + */'), + self::nowdoc(' + /** + * @param object{bar: float} $foo + */'), + $addCommentToObjectShapeItemMiddle, + ]; + + yield [ + self::nowdoc(' + /** + * @param object{foo:string} $foo + */'), + self::nowdoc(' + /** + * @param object{foo:string, + * // A fractional number + * bar: float} $foo + */'), + $addCommentToObjectShapeItemMiddle, + ]; + + yield [ + self::nowdoc(' + /** + * @param object{ + * foo:string,naz:int} $foo + */'), + self::nowdoc(' + /** + * @param object{ + * foo:string, + * // A fractional number + * bar: float,naz:int} $foo + */'), + $addCommentToObjectShapeItemMiddle, + ]; + + yield [ + self::nowdoc(' + /** + * @param object{ + * foo:string, + * naz:int + * } $foo + */'), + self::nowdoc(' + /** + * @param object{ + * foo:string, + * // A fractional number + * bar: float, + * naz:int + * } $foo + */'), + $addCommentToObjectShapeItemMiddle, + ]; } /** @@ -1486,7 +2631,8 @@ public function enterNode(Node $node) */ public function testPrintFormatPreserving(string $phpDoc, string $expectedResult, NodeVisitor $visitor): void { - $lexer = new Lexer(); + $config = new ParserConfig(['lines' => true, 'indexes' => true, 'comments' => true]); + $lexer = new Lexer($config); $tokens = new TokenIterator($lexer->tokenize($phpDoc)); $phpDocNode = $this->phpDocParser->parse($tokens); $cloningTraverser = new NodeTraverser([new NodeVisitor\CloningVisitor()]); @@ -1503,7 +2649,7 @@ public function testPrintFormatPreserving(string $phpDoc, string $expectedResult $this->assertEquals( $this->unsetAttributes($newNode), - $this->unsetAttributes($this->phpDocParser->parse(new TokenIterator($lexer->tokenize($newPhpDoc)))) + $this->unsetAttributes($this->phpDocParser->parse(new TokenIterator($lexer->tokenize($newPhpDoc)))), ); } @@ -1518,6 +2664,7 @@ public function enterNode(Node $node) $node->setAttribute(Attribute::START_INDEX, null); $node->setAttribute(Attribute::END_INDEX, null); $node->setAttribute(Attribute::ORIGINAL_NODE, null); + $node->setAttribute(Attribute::COMMENTS, null); return $node; } @@ -1531,7 +2678,7 @@ public function enterNode(Node $node) } /** - * @return iterable + * @return iterable */ public function dataPrintType(): iterable { @@ -1559,7 +2706,7 @@ public function dataPrintType(): iterable [ GenericTypeNode::VARIANCE_INVARIANT, GenericTypeNode::VARIANCE_INVARIANT, - ] + ], ), 'array', ]; @@ -1567,18 +2714,18 @@ public function dataPrintType(): iterable new CallableTypeNode(new IdentifierTypeNode('callable'), [], new UnionTypeNode([ new IdentifierTypeNode('int'), new IdentifierTypeNode('string'), - ])), + ]), []), 'callable(): (int|string)', ]; yield [ - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new ArrayTypeNode(new ArrayTypeNode(new IdentifierTypeNode('int'))))), + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new ArrayTypeNode(new ArrayTypeNode(new IdentifierTypeNode('int')))), []), 'callable(): int[][][]', ]; yield [ new ArrayTypeNode( new ArrayTypeNode( - new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new IdentifierTypeNode('int'))) - ) + new CallableTypeNode(new IdentifierTypeNode('callable'), [], new ArrayTypeNode(new IdentifierTypeNode('int')), []), + ), ), '(callable(): int[])[][]', ]; @@ -1618,6 +2765,43 @@ public function dataPrintType(): iterable ]), 'Foo|Bar|(Baz|Lorem)', ]; + yield [ + new OffsetAccessTypeNode( + new ConstTypeNode(new ConstFetchNode('self', 'TYPES')), + new IdentifierTypeNode('int'), + ), + 'self::TYPES[int]', + ]; + yield [ + ArrayShapeNode::createSealed([ + new ArrayShapeItemNode( + new IdentifierTypeNode('name'), + false, + new IdentifierTypeNode('string'), + ), + new ArrayShapeItemNode( + new ConstExprStringNode('Full Name', ConstExprStringNode::SINGLE_QUOTED), + false, + new IdentifierTypeNode('string'), + ), + ]), + "array{name: string, 'Full Name': string}", + ]; + yield [ + new ObjectShapeNode([ + new ObjectShapeItemNode( + new IdentifierTypeNode('name'), + false, + new IdentifierTypeNode('string'), + ), + new ObjectShapeItemNode( + new ConstExprStringNode('Full Name', ConstExprStringNode::SINGLE_QUOTED), + false, + new IdentifierTypeNode('string'), + ), + ]), + "object{name: string, 'Full Name': string}", + ]; } /** @@ -1629,15 +2813,16 @@ public function testPrintType(TypeNode $node, string $expectedResult): void $phpDoc = $printer->print($node); $this->assertSame($expectedResult, $phpDoc); - $lexer = new Lexer(); + $config = new ParserConfig([]); + $lexer = new Lexer($config); $this->assertEquals( $this->unsetAttributes($node), - $this->unsetAttributes($this->typeParser->parse(new TokenIterator($lexer->tokenize($phpDoc)))) + $this->unsetAttributes($this->typeParser->parse(new TokenIterator($lexer->tokenize($phpDoc)))), ); } /** - * @return iterable + * @return iterable */ public function dataPrintPhpDocNode(): iterable { @@ -1647,7 +2832,8 @@ public function dataPrintPhpDocNode(): iterable new IdentifierTypeNode('int'), false, '$a', - '' + '', + false, )), ]), '/** @@ -1665,11 +2851,59 @@ public function testPrintPhpDocNode(PhpDocNode $node, string $expectedResult): v $phpDoc = $printer->print($node); $this->assertSame($expectedResult, $phpDoc); - $lexer = new Lexer(); + $config = new ParserConfig([]); + $lexer = new Lexer($config); $this->assertEquals( $this->unsetAttributes($node), - $this->unsetAttributes($this->phpDocParser->parse(new TokenIterator($lexer->tokenize($phpDoc)))) + $this->unsetAttributes($this->phpDocParser->parse(new TokenIterator($lexer->tokenize($phpDoc)))), ); } + /** + * @template TNode of Node + * @param TNode $node + * @return TNode + */ + public static function withComment(Node $node, string $comment): Node + { + $node->setAttribute(Attribute::COMMENTS, [new Comment($comment)]); + return $node; + } + + public static function nowdoc(string $str): string + { + $lines = preg_split('/\\n/', $str); + + if ($lines === false) { + return ''; + } + + if (count($lines) < 2) { + return ''; + } + + // Toss out the first line + $lines = array_slice($lines, 1, count($lines) - 1); + + // normalize any tabs to spaces + $lines = array_map(static fn ($line) => preg_replace_callback('/(\t+)/m', static function ($matches) { + $fixed = str_repeat(' ', strlen($matches[1])); + return $fixed; + }, $line), $lines); + + // take the ws from the first line and subtract them from all lines + $matches = []; + + if (preg_match('/(^[ \t]+)/', $lines[0] ?? '', $matches) !== 1) { + return ''; + } + + $numLines = count($lines); + for ($i = 0; $i < $numLines; ++$i) { + $lines[$i] = str_replace($matches[0], '', $lines[$i] ?? ''); + } + + return implode("\n", $lines); + } + } diff --git a/tests/bootstrap.php b/tests/bootstrap.php index 558d6ec4..f5fb2f9b 100644 --- a/tests/bootstrap.php +++ b/tests/bootstrap.php @@ -1,3 +1,7 @@