Skip to content

Commit

Permalink
nikic/php-parser v5 + phpstan/phpdoc-parser v2 (#76)
Browse files Browse the repository at this point in the history
  • Loading branch information
janedbal authored Nov 26, 2024
1 parent 7671a65 commit c019f58
Show file tree
Hide file tree
Showing 7 changed files with 57 additions and 39 deletions.
6 changes: 3 additions & 3 deletions composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,8 @@
"require": {
"php": "^8.1",
"nette/utils": "^3.2 || ^4.0",
"nikic/php-parser": "^4.15 || ^5.0",
"phpstan/phpdoc-parser": "^1.18.1"
"nikic/php-parser": "^5.0",
"phpstan/phpdoc-parser": "^2.0.0"
},
"require-dev": {
"editorconfig-checker/editorconfig-checker": "^10.6.0",
Expand All @@ -20,7 +20,7 @@
"shipmonk/composer-dependency-analyser": "^1.8.1",
"shipmonk/name-collision-detector": "^2.1.1",
"shipmonk/phpstan-rules": "^4.0.0",
"slevomat/coding-standard": "^8.15.0"
"slevomat/coding-standard": "dev-master"
},
"autoload": {
"psr-4": {
Expand Down
7 changes: 4 additions & 3 deletions src/Compiler/Mapper/Array/MapArrayShape.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@
use Attribute;
use PhpParser\Node\Expr;
use PhpParser\Node\Stmt;
use PHPStan\PhpDocParser\Ast\ConstExpr\ConstExprStringNode;
use PHPStan\PhpDocParser\Ast\Type\ArrayShapeItemNode;
use PHPStan\PhpDocParser\Ast\Type\ArrayShapeNode;
use PHPStan\PhpDocParser\Ast\Type\IdentifierTypeNode;
Expand Down Expand Up @@ -102,13 +101,15 @@ public function getOutputType(): TypeNode

foreach ($this->items as $mapping) {
$items[] = new ArrayShapeItemNode(
new ConstExprStringNode($mapping->key),
new IdentifierTypeNode($mapping->key),
$mapping->optional,
$mapping->mapper->getOutputType(),
);
}

return new ArrayShapeNode($items, $this->sealed, ArrayShapeNode::KIND_ARRAY);
return $this->sealed
? ArrayShapeNode::createSealed($items)
: ArrayShapeNode::createUnsealed($items, null);
}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
use PHPStan\PhpDocParser\Parser\ConstExprParser;
use PHPStan\PhpDocParser\Parser\PhpDocParser;
use PHPStan\PhpDocParser\Parser\TypeParser;
use PHPStan\PhpDocParser\ParserConfig;

class DefaultMapperCompilerFactoryProvider implements MapperCompilerFactoryProvider
{
Expand All @@ -19,20 +20,26 @@ public function get(): MapperCompilerFactory

protected function create(): MapperCompilerFactory
{
return new DefaultMapperCompilerFactory($this->createPhpDocLexer(), $this->createPhpDocParser());
$config = $this->createParserConfig();
return new DefaultMapperCompilerFactory($this->createPhpDocLexer($config), $this->createPhpDocParser($config));
}

protected function createPhpDocLexer(): Lexer
protected function createPhpDocLexer(ParserConfig $config): Lexer
{
return new Lexer();
return new Lexer($config);
}

protected function createPhpDocParser(): PhpDocParser
protected function createParserConfig(): ParserConfig
{
$phpDocExprParser = new ConstExprParser(unescapeStrings: true);
$phpDocTypeParser = new TypeParser($phpDocExprParser);
return new ParserConfig([]);
}

protected function createPhpDocParser(ParserConfig $config): PhpDocParser
{
$phpDocExprParser = new ConstExprParser($config);
$phpDocTypeParser = new TypeParser($config, $phpDocExprParser);

return new PhpDocParser($phpDocTypeParser, $phpDocExprParser);
return new PhpDocParser($config, $phpDocTypeParser, $phpDocExprParser);
}

}
4 changes: 2 additions & 2 deletions src/Compiler/Php/PhpCodeBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,10 @@
use PhpParser\Builder\Method;
use PhpParser\BuilderFactory;
use PhpParser\Node\Arg;
use PhpParser\Node\ArrayItem;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\Array_;
use PhpParser\Node\Expr\ArrayDimFetch;
use PhpParser\Node\Expr\ArrayItem;
use PhpParser\Node\Expr\Assign;
use PhpParser\Node\Expr\BinaryOp\BooleanAnd;
use PhpParser\Node\Expr\BinaryOp\BooleanOr;
Expand Down Expand Up @@ -528,7 +528,7 @@ public function mapperClassConstructor(MapperCompiler $mapperCompiler): ClassMet
$innerMappersParameter->flags = ClassNode::MODIFIER_PRIVATE | ClassNode::MODIFIER_READONLY;
$mapperConstructorBuilder->addParam($innerMappersParameter);

$innerMappersType = new ArrayShapeNode(Arrays::map(
$innerMappersType = ArrayShapeNode::createSealed(Arrays::map(
$mapperCompiler->getGenericParameters(),
static function (GenericTypeParameter $genericParameter): ArrayShapeItemNode {
return new ArrayShapeItemNode(
Expand Down
18 changes: 11 additions & 7 deletions src/Compiler/Type/PhpDocTypeUtils.php
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@
use PHPStan\PhpDocParser\Parser\PhpDocParser;
use PHPStan\PhpDocParser\Parser\TokenIterator;
use PHPStan\PhpDocParser\Parser\TypeParser;
use PHPStan\PhpDocParser\ParserConfig;
use ReflectionClass;
use ReflectionIntersectionType;
use ReflectionNamedType;
Expand Down Expand Up @@ -179,7 +180,7 @@ public static function fromValue(mixed $value): TypeNode
);
}

return new ArrayShapeNode($items);
return ArrayShapeNode::createSealed($items);
}

if (is_object($value)) {
Expand Down Expand Up @@ -947,10 +948,11 @@ private static function getGenericTypeDefinitionFromPhpDoc(string $className): G

private static function parsePhpDoc(string $phpDoc): PhpDocNode
{
$phpDocLexer = new Lexer();
$phpDocTypeParser = new TypeParser();
$phpDocConstExprParser = new ConstExprParser(unescapeStrings: true);
$phpDocParser = new PhpDocParser($phpDocTypeParser, $phpDocConstExprParser);
$config = new ParserConfig([]);
$phpDocLexer = new Lexer($config);
$phpDocConstExprParser = new ConstExprParser($config);
$phpDocTypeParser = new TypeParser($config, $phpDocConstExprParser);
$phpDocParser = new PhpDocParser($config, $phpDocTypeParser, $phpDocConstExprParser);
$phpDocTokens = $phpDocLexer->tokenize($phpDoc);

return $phpDocParser->parse(new TokenIterator($phpDocTokens));
Expand Down Expand Up @@ -1065,7 +1067,7 @@ private static function normalizeType(TypeNode $type): TypeNode

} elseif ($item->keyName instanceof IdentifierTypeNode) {
$newItems[] = new ArrayShapeItemNode(
keyName: new ConstExprStringNode($item->keyName->name),
keyName: new ConstExprStringNode($item->keyName->name, ConstExprStringNode::SINGLE_QUOTED),
optional: $item->optional,
valueType: $item->valueType,
);
Expand All @@ -1075,7 +1077,9 @@ private static function normalizeType(TypeNode $type): TypeNode
}
}

return new ArrayShapeNode($newItems, $type->sealed, $type->kind);
return $type->sealed
? ArrayShapeNode::createSealed($newItems, $type->kind)
: ArrayShapeNode::createUnsealed($newItems, null, $type->kind);
}

if ($type instanceof GenericTypeNode) {
Expand Down
28 changes: 16 additions & 12 deletions tests/Compiler/MapperFactory/DefaultMapperCompilerFactoryTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@
use PHPStan\PhpDocParser\Parser\PhpDocParser;
use PHPStan\PhpDocParser\Parser\TokenIterator;
use PHPStan\PhpDocParser\Parser\TypeParser;
use PHPStan\PhpDocParser\ParserConfig;
use PHPUnit\Framework\Attributes\DataProvider;
use ShipMonk\InputMapper\Compiler\Exception\CannotCreateMapperCompilerException;
use ShipMonk\InputMapper\Compiler\Mapper\Array\ArrayShapeItemMapping;
Expand Down Expand Up @@ -71,10 +72,11 @@ class DefaultMapperCompilerFactoryTest extends InputMapperTestCase
#[DataProvider('provideCreateOkData')]
public function testCreateOk(string $type, array $options, MapperCompiler $expectedMapperCompiler): void
{
$phpDocLexer = new Lexer();
$phpDocExprParser = new ConstExprParser(unescapeStrings: true);
$phpDocTypeParser = new TypeParser($phpDocExprParser);
$phpDocParser = new PhpDocParser($phpDocTypeParser, $phpDocExprParser);
$config = new ParserConfig([]);
$phpDocLexer = new Lexer($config);
$phpDocConstExprParser = new ConstExprParser($config);
$phpDocTypeParser = new TypeParser($config, $phpDocConstExprParser);
$phpDocParser = new PhpDocParser($config, $phpDocTypeParser, $phpDocConstExprParser);
$phpDocType = $phpDocTypeParser->parse(new TokenIterator($phpDocLexer->tokenize($type)));

$mapperCompilerFactory = new DefaultMapperCompilerFactory($phpDocLexer, $phpDocParser);
Expand Down Expand Up @@ -435,10 +437,11 @@ className: InputWithRenamedSourceKey::class,
#[DataProvider('provideCreateErrorData')]
public function testCreateError(string $type, array $options, ?string $expectedMessage = null): void
{
$phpDocLexer = new Lexer();
$phpDocExprParser = new ConstExprParser(unescapeStrings: true);
$phpDocTypeParser = new TypeParser($phpDocExprParser);
$phpDocParser = new PhpDocParser($phpDocTypeParser, $phpDocExprParser);
$config = new ParserConfig([]);
$phpDocLexer = new Lexer($config);
$phpDocConstExprParser = new ConstExprParser($config);
$phpDocTypeParser = new TypeParser($config, $phpDocConstExprParser);
$phpDocParser = new PhpDocParser($config, $phpDocTypeParser, $phpDocConstExprParser);
$phpDocType = $phpDocTypeParser->parse(new TokenIterator($phpDocLexer->tokenize($type)));

$mapperCompilerFactory = new DefaultMapperCompilerFactory($phpDocLexer, $phpDocParser);
Expand Down Expand Up @@ -510,10 +513,11 @@ public static function provideCreateErrorData(): iterable

public function testCreateWithCustomFactory(): void
{
$phpDocLexer = new Lexer();
$phpDocExprParser = new ConstExprParser(unescapeStrings: true);
$phpDocTypeParser = new TypeParser($phpDocExprParser);
$phpDocParser = new PhpDocParser($phpDocTypeParser, $phpDocExprParser);
$config = new ParserConfig([]);
$phpDocLexer = new Lexer($config);
$phpDocConstExprParser = new ConstExprParser($config);
$phpDocTypeParser = new TypeParser($config, $phpDocConstExprParser);
$phpDocParser = new PhpDocParser($config, $phpDocTypeParser, $phpDocConstExprParser);

$carMapperCompiler = new MapObject(CarInput::class, [
'id' => new MapInt(),
Expand Down
12 changes: 7 additions & 5 deletions tests/Compiler/Type/PhpDocTypeUtilsTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
use PHPStan\PhpDocParser\Parser\ConstExprParser;
use PHPStan\PhpDocParser\Parser\TokenIterator;
use PHPStan\PhpDocParser\Parser\TypeParser;
use PHPStan\PhpDocParser\ParserConfig;
use PHPUnit\Framework\Attributes\DataProvider;
use PHPUnit\Framework\TestCase;
use ReflectionClass;
Expand Down Expand Up @@ -279,7 +280,7 @@ public static function provideToNativeTypeData(): iterable
];

yield 'array{int, string}' => [
new ArrayShapeNode([
ArrayShapeNode::createSealed([
new ArrayShapeItemNode(keyName: null, optional: false, valueType: new IdentifierTypeNode('int')),
new ArrayShapeItemNode(keyName: null, optional: false, valueType: new IdentifierTypeNode('string')),
]),
Expand Down Expand Up @@ -397,7 +398,7 @@ public static function provideIsNullableData(): iterable
];

yield 'array{int, string}' => [
new ArrayShapeNode([
ArrayShapeNode::createSealed([
new ArrayShapeItemNode(keyName: null, optional: false, valueType: new IdentifierTypeNode('int')),
new ArrayShapeItemNode(keyName: null, optional: false, valueType: new IdentifierTypeNode('string')),
]),
Expand Down Expand Up @@ -1609,9 +1610,10 @@ public static function provideInferGenericParameterData(): iterable

private function parseType(string $type): TypeNode
{
$lexer = new Lexer();
$constExprParser = new ConstExprParser(unescapeStrings: true);
$typeParser = new TypeParser($constExprParser);
$config = new ParserConfig([]);
$lexer = new Lexer($config);
$constExprParser = new ConstExprParser($config);
$typeParser = new TypeParser($config, $constExprParser);

$tokens = new TokenIterator($lexer->tokenize($type));
$typeNode = $typeParser->parse($tokens);
Expand Down

0 comments on commit c019f58

Please sign in to comment.