From 7485b1cc4462b74d9a3af7cda74cca9b7d58b51a Mon Sep 17 00:00:00 2001 From: spawnia Date: Mon, 13 Apr 2020 19:34:42 +0200 Subject: [PATCH 1/7] Support repeatable directives See https://github.com/graphql/graphql-js/pull/1965 --- phpstan-baseline.neon | 22 ++--- src/Language/AST/DirectiveDefinitionNode.php | 3 + src/Language/Parser.php | 55 +++++++----- src/Language/Printer.php | 1 + src/Type/Definition/Directive.php | 26 ++++-- src/Utils/ASTDefinitionBuilder.php | 1 + src/Utils/SchemaPrinter.php | 86 ++++++++++--------- .../Rules/UniqueDirectivesPerLocation.php | 35 +++++++- tests/Language/SchemaParserTest.php | 81 +++++++++++++++++ tests/Language/SchemaPrinterTest.php | 2 + tests/Language/schema-kitchen-sink.graphql | 4 + tests/Type/DirectiveTest.php | 15 ++++ tests/Utils/BuildSchemaTest.php | 2 + tests/Utils/SchemaExtenderTest.php | 4 +- tests/Utils/SchemaPrinterTest.php | 34 ++++++-- .../UniqueDirectivesPerLocationTest.php | 74 ++++++++++------ tests/Validator/ValidatorTestCase.php | 29 +++++-- 17 files changed, 344 insertions(+), 130 deletions(-) create mode 100644 tests/Type/DirectiveTest.php diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index 09f3df772..6ad8a8e67 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -440,11 +440,6 @@ parameters: count: 2 path: src/Server/OperationParams.php - - - message: "#^Variable property access on \\$this\\(GraphQL\\\\Type\\\\Definition\\\\Directive\\)\\.$#" - count: 1 - path: src/Type/Definition/Directive.php - - message: "#^Only booleans are allowed in a negated boolean, ArrayObject\\ given\\.$#" count: 1 @@ -891,13 +886,8 @@ parameters: path: src/Utils/SchemaExtender.php - - message: "#^Anonymous function should have native return typehint \"bool\"\\.$#" - count: 3 - path: src/Utils/SchemaPrinter.php - - - - message: "#^Anonymous function should have native return typehint \"string\"\\.$#" - count: 7 + message: "#^Anonymous function should return bool but returns string\\.$#" + count: 1 path: src/Utils/SchemaPrinter.php - @@ -1496,22 +1486,22 @@ parameters: path: src/Validator/Rules/ValuesOfCorrectType.php - - message: "#^Only booleans are allowed in \\|\\|, GraphQL\\\\Type\\\\Definition\\\\EnumType\\|GraphQL\\\\Type\\\\Definition\\\\InputObjectType\\|GraphQL\\\\Type\\\\Definition\\\\ListOfType\\|GraphQL\\\\Type\\\\Definition\\\\NonNull\\|GraphQL\\\\Type\\\\Definition\\\\ScalarType given on the left side\\.$#" + message: "#^Anonymous function should have native return typehint \"string\"\\.$#" count: 1 path: src/Validator/Rules/ValuesOfCorrectType.php - - message: "#^Only booleans are allowed in a negated boolean, GraphQL\\\\Type\\\\Definition\\\\EnumValueDefinition\\|null given\\.$#" + message: "#^Only booleans are allowed in \\|\\|, GraphQL\\\\Type\\\\Definition\\\\EnumType\\|GraphQL\\\\Type\\\\Definition\\\\InputObjectType\\|GraphQL\\\\Type\\\\Definition\\\\ListOfType\\|GraphQL\\\\Type\\\\Definition\\\\NonNull\\|GraphQL\\\\Type\\\\Definition\\\\ScalarType given on the left side\\.$#" count: 1 path: src/Validator/Rules/ValuesOfCorrectType.php - - message: "#^Only booleans are allowed in a negated boolean, GraphQL\\\\Type\\\\Definition\\\\EnumType\\|GraphQL\\\\Type\\\\Definition\\\\InputObjectType\\|GraphQL\\\\Type\\\\Definition\\\\ListOfType\\|GraphQL\\\\Type\\\\Definition\\\\NonNull\\|GraphQL\\\\Type\\\\Definition\\\\ScalarType given\\.$#" + message: "#^Only booleans are allowed in a negated boolean, GraphQL\\\\Type\\\\Definition\\\\EnumValueDefinition\\|null given\\.$#" count: 1 path: src/Validator/Rules/ValuesOfCorrectType.php - - message: "#^Anonymous function should have native return typehint \"string\"\\.$#" + message: "#^Only booleans are allowed in a negated boolean, GraphQL\\\\Type\\\\Definition\\\\EnumType\\|GraphQL\\\\Type\\\\Definition\\\\InputObjectType\\|GraphQL\\\\Type\\\\Definition\\\\ListOfType\\|GraphQL\\\\Type\\\\Definition\\\\NonNull\\|GraphQL\\\\Type\\\\Definition\\\\ScalarType given\\.$#" count: 1 path: src/Validator/Rules/ValuesOfCorrectType.php diff --git a/src/Language/AST/DirectiveDefinitionNode.php b/src/Language/AST/DirectiveDefinitionNode.php index 825c7f789..96b0c542c 100644 --- a/src/Language/AST/DirectiveDefinitionNode.php +++ b/src/Language/AST/DirectiveDefinitionNode.php @@ -15,6 +15,9 @@ class DirectiveDefinitionNode extends Node implements TypeSystemDefinitionNode /** @var ArgumentNode[] */ public $arguments; + /** @var bool */ + public $repeatable; + /** @var NameNode[] */ public $locations; diff --git a/src/Language/Parser.php b/src/Language/Parser.php index b0ccc7b7c..202d8ede5 100644 --- a/src/Language/Parser.php +++ b/src/Language/Parser.php @@ -327,26 +327,39 @@ private function expect(string $kind) : Token } /** - * If the next token is a keyword with the given value, return that token after - * advancing the parser. Otherwise, do not change the parser state and return - * false. + * If the next token is a keyword with the given value, advance the lexer. + * Otherwise, throw an error. * * @throws SyntaxError */ - private function expectKeyword(string $value) : Token + private function expectKeyword(string $value) : void { $token = $this->lexer->token; + if ($token->kind !== Token::NAME || $token->value !== $value) { + throw new SyntaxError( + $this->lexer->source, + $token->start, + 'Expected "' . $value . '", found ' . $token->getDescription() + ); + } + + $this->lexer->advance(); + } + /** + * If the next token is a given keyword, return "true" after advancing + * the lexer. Otherwise, do not change the parser state and return "false". + */ + private function expectOptionalKeyword(string $value) : bool + { + $token = $this->lexer->token; if ($token->kind === Token::NAME && $token->value === $value) { $this->lexer->advance(); - return $token; + return true; } - throw new SyntaxError( - $this->lexer->source, - $token->start, - 'Expected "' . $value . '", found ' . $token->getDescription() - ); + + return false; } private function unexpected(?Token $atToken = null) : SyntaxError @@ -716,7 +729,8 @@ private function parseFragment() : SelectionNode $start = $this->lexer->token; $this->expect(Token::SPREAD); - if ($this->peek(Token::NAME) && $this->lexer->token->value !== 'on') { + $hasTypeCondition = $this->expectOptionalKeyword('on'); + if (! $hasTypeCondition && $this->peek(Token::NAME)) { return new FragmentSpreadNode([ 'name' => $this->parseFragmentName(), 'directives' => $this->parseDirectives(false), @@ -724,14 +738,8 @@ private function parseFragment() : SelectionNode ]); } - $typeCondition = null; - if ($this->lexer->token->value === 'on') { - $this->lexer->advance(); - $typeCondition = $this->parseNamedType(); - } - return new InlineFragmentNode([ - 'typeCondition' => $typeCondition, + 'typeCondition' => $hasTypeCondition ? $this->parseNamedType() : null, 'directives' => $this->parseDirectives(false), 'selectionSet' => $this->parseSelectionSet(), 'loc' => $this->loc($start), @@ -1172,8 +1180,7 @@ private function parseObjectTypeDefinition() : ObjectTypeDefinitionNode private function parseImplementsInterfaces() : array { $types = []; - if ($this->lexer->token->value === 'implements') { - $this->lexer->advance(); + if ($this->expectOptionalKeyword('implements')) { // Optional leading ampersand $this->skip(Token::AMP); do { @@ -1668,7 +1675,7 @@ private function parseInputObjectTypeExtension() : InputObjectTypeExtensionNode /** * DirectiveDefinition : - * - directive @ Name ArgumentsDefinition? on DirectiveLocations + * - Description? directive @ Name ArgumentsDefinition? `repeatable`? on DirectiveLocations * * @throws SyntaxError */ @@ -1678,14 +1685,16 @@ private function parseDirectiveDefinition() : DirectiveDefinitionNode $description = $this->parseDescription(); $this->expectKeyword('directive'); $this->expect(Token::AT); - $name = $this->parseName(); - $args = $this->parseArgumentsDefinition(); + $name = $this->parseName(); + $args = $this->parseArgumentsDefinition(); + $repeatable = $this->expectOptionalKeyword('repeatable'); $this->expectKeyword('on'); $locations = $this->parseDirectiveLocations(); return new DirectiveDefinitionNode([ 'name' => $name, 'arguments' => $args, + 'repeatable' => $repeatable, 'locations' => $locations, 'loc' => $this->loc($start), 'description' => $description, diff --git a/src/Language/Printer.php b/src/Language/Printer.php index e54eee601..0b93df6dc 100644 --- a/src/Language/Printer.php +++ b/src/Language/Printer.php @@ -446,6 +446,7 @@ function (InterfaceTypeDefinitionNode $def) { . ($noIndent ? $this->wrap('(', $this->join($def->arguments, ', '), ')') : $this->wrap("(\n", $this->indent($this->join($def->arguments, "\n")), "\n")) + . ($def->repeatable ? ' repeatable' : '') . ' on ' . $this->join($def->locations, ' | '); }), ], diff --git a/src/Type/Definition/Directive.php b/src/Type/Definition/Directive.php index 75bc03407..727ca63e7 100644 --- a/src/Type/Definition/Directive.php +++ b/src/Type/Definition/Directive.php @@ -4,9 +4,9 @@ namespace GraphQL\Type\Definition; +use GraphQL\Error\InvariantViolation; use GraphQL\Language\AST\DirectiveDefinitionNode; use GraphQL\Language\DirectiveLocation; -use GraphQL\Utils\Utils; use function array_key_exists; use function is_array; @@ -37,6 +37,9 @@ class Directive /** @var FieldArgument[] */ public $args = []; + /** @var bool */ + public $isRepeatable; + /** @var DirectiveDefinitionNode|null */ public $astNode; @@ -48,6 +51,18 @@ class Directive */ public function __construct(array $config) { + if (! isset($config['name'])) { + throw new InvariantViolation('Directive must be named.'); + } + $this->name = $config['name']; + + $this->description = $config['description'] ?? null; + + if (! isset($config['locations']) || ! is_array($config['locations'])) { + throw new InvariantViolation('Must provide locations for directive.'); + } + $this->locations = $config['locations']; + if (isset($config['args'])) { $args = []; foreach ($config['args'] as $name => $arg) { @@ -58,14 +73,11 @@ public function __construct(array $config) } } $this->args = $args; - unset($config['args']); - } - foreach ($config as $key => $value) { - $this->{$key} = $value; } - Utils::invariant($this->name, 'Directive must be named.'); - Utils::invariant(is_array($this->locations), 'Must provide locations for directive.'); + $this->isRepeatable = $config['isRepeatable'] ?? false; + $this->astNode = $config['astNode'] ?? null; + $this->config = $config; } diff --git a/src/Utils/ASTDefinitionBuilder.php b/src/Utils/ASTDefinitionBuilder.php index 851f1f430..71e934961 100644 --- a/src/Utils/ASTDefinitionBuilder.php +++ b/src/Utils/ASTDefinitionBuilder.php @@ -86,6 +86,7 @@ static function ($node) { } ), 'args' => $directiveNode->arguments ? FieldArgument::createMap($this->makeInputValues($directiveNode->arguments)) : null, + 'isRepeatable' => $directiveNode->repeatable, 'astNode' => $directiveNode, ]); } diff --git a/src/Utils/SchemaPrinter.php b/src/Utils/SchemaPrinter.php index ccbb00734..acc612ae6 100644 --- a/src/Utils/SchemaPrinter.php +++ b/src/Utils/SchemaPrinter.php @@ -38,23 +38,21 @@ class SchemaPrinter { /** - * Accepts options as a second argument: - * + * @param array $options + * Available options: * - commentDescriptions: * Provide true to use preceding comments as the description. * - * @param bool[] $options - * * @api */ public static function doPrint(Schema $schema, array $options = []) : string { return self::printFilteredSchema( $schema, - static function ($type) { + static function ($type): bool { return ! Directive::isSpecifiedDirective($type); }, - static function ($type) { + static function ($type): bool { return ! Type::isBuiltInType($type); }, $options @@ -62,16 +60,11 @@ static function ($type) { } /** - * @param bool[] $options + * @param array $options */ - private static function printFilteredSchema(Schema $schema, $directiveFilter, $typeFilter, $options) : string + private static function printFilteredSchema(Schema $schema, callable $directiveFilter, callable $typeFilter, array $options) : string { - $directives = array_filter( - $schema->getDirectives(), - static function ($directive) use ($directiveFilter) { - return $directiveFilter($directive); - } - ); + $directives = array_filter($schema->getDirectives(), $directiveFilter); $types = $schema->getTypeMap(); ksort($types); @@ -85,13 +78,13 @@ static function ($directive) use ($directiveFilter) { array_merge( [self::printSchemaDefinition($schema)], array_map( - static function ($directive) use ($options) { + static function (Directive $directive) use ($options) : string { return self::printDirective($directive, $options); }, $directives ), array_map( - static function ($type) use ($options) { + static function ($type) use ($options): bool { return self::printType($type, $options); }, $types @@ -157,14 +150,22 @@ private static function isSchemaOfCommonNames(Schema $schema) : bool return $subscriptionType === null || $subscriptionType->name === 'Subscription'; } - private static function printDirective($directive, $options) : string + /** + * @param array $options + */ + private static function printDirective(Directive $directive, array $options) : string { - return self::printDescription($options, $directive) . - 'directive @' . $directive->name . self::printArgs($options, $directive->args) . - ' on ' . implode(' | ', $directive->locations); + return self::printDescription($options, $directive) + . 'directive @' . $directive->name + . self::printArgs($options, $directive->args) + . ($directive->isRepeatable ? ' repeatable' : '') + . ' on ' . implode(' | ', $directive->locations); } - private static function printDescription($options, $def, $indentation = '', $firstInBlock = true) : string + /** + * @param array $options + */ + private static function printDescription(array $options, $def, $indentation = '', $firstInBlock = true) : string { if (! $def->description) { return ''; @@ -264,7 +265,10 @@ private static function escapeQuote($line) : string return str_replace('"""', '\\"""', $line); } - private static function printArgs($options, $args, $indentation = '') : string + /** + * @param array $options + */ + private static function printArgs(array $options, $args, $indentation = '') : string { if (! $args) { return ''; @@ -273,7 +277,7 @@ private static function printArgs($options, $args, $indentation = '') : string // If every arg does not have a description, print them on one line. if (Utils::every( $args, - static function ($arg) { + static function ($arg): bool { return empty($arg->description); } )) { @@ -285,7 +289,7 @@ static function ($arg) { implode( "\n", array_map( - static function ($arg, $i) use ($indentation, $options) { + static function ($arg, $i) use ($indentation, $options): string { return self::printDescription($options, $arg, ' ' . $indentation, ! $i) . ' ' . $indentation . self::printInputValue($arg); }, @@ -308,7 +312,7 @@ private static function printInputValue($arg) : string } /** - * @param bool[] $options + * @param array $options */ public static function printType(Type $type, array $options = []) : string { @@ -340,7 +344,7 @@ public static function printType(Type $type, array $options = []) : string } /** - * @param bool[] $options + * @param array $options */ private static function printScalar(ScalarType $type, array $options) : string { @@ -348,7 +352,7 @@ private static function printScalar(ScalarType $type, array $options) : string } /** - * @param bool[] $options + * @param array $options */ private static function printObject(ObjectType $type, array $options) : string { @@ -357,8 +361,8 @@ private static function printObject(ObjectType $type, array $options) : string ? ' implements ' . implode( ' & ', array_map( - static function ($i) { - return $i->name; + static function (InterfaceType $interface): string { + return $interface->name; }, $interfaces ) @@ -370,16 +374,16 @@ static function ($i) { } /** - * @param bool[] $options + * @param array $options */ - private static function printFields($options, $type) : string + private static function printFields(array $options, $type) : string { $fields = array_values($type->getFields()); return implode( "\n", array_map( - static function ($f, $i) use ($options) { + static function ($f, $i) use ($options): string { return self::printDescription($options, $f, ' ', ! $i) . ' ' . $f->name . self::printArgs($options, $f->args, ' ') . ': ' . (string) $f->getType() . self::printDeprecated($f); @@ -405,7 +409,7 @@ private static function printDeprecated($fieldOrEnumVal) : string } /** - * @param bool[] $options + * @param array $options */ private static function printInterface(InterfaceType $type, array $options) : string { @@ -414,7 +418,7 @@ private static function printInterface(InterfaceType $type, array $options) : st } /** - * @param bool[] $options + * @param array $options */ private static function printUnion(UnionType $type, array $options) : string { @@ -423,7 +427,7 @@ private static function printUnion(UnionType $type, array $options) : string } /** - * @param bool[] $options + * @param array $options */ private static function printEnum(EnumType $type, array $options) : string { @@ -432,14 +436,14 @@ private static function printEnum(EnumType $type, array $options) : string } /** - * @param bool[] $options + * @param array $options */ - private static function printEnumValues($values, $options) : string + private static function printEnumValues($values, array $options) : string { return implode( "\n", array_map( - static function ($value, $i) use ($options) { + static function ($value, $i) use ($options): string { return self::printDescription($options, $value, ' ', ! $i) . ' ' . $value->name . self::printDeprecated($value); }, @@ -450,7 +454,7 @@ static function ($value, $i) use ($options) { } /** - * @param bool[] $options + * @param array $options */ private static function printInputObject(InputObjectType $type, array $options) : string { @@ -463,7 +467,7 @@ private static function printInputObject(InputObjectType $type, array $options) implode( "\n", array_map( - static function ($f, $i) use ($options) { + static function ($f, $i) use ($options): string { return self::printDescription($options, $f, ' ', ! $i) . ' ' . self::printInputValue($f); }, $fields, @@ -474,7 +478,7 @@ static function ($f, $i) use ($options) { } /** - * @param bool[] $options + * @param array $options * * @api */ diff --git a/src/Validator/Rules/UniqueDirectivesPerLocation.php b/src/Validator/Rules/UniqueDirectivesPerLocation.php index 9b784f19c..7c6eb9c07 100644 --- a/src/Validator/Rules/UniqueDirectivesPerLocation.php +++ b/src/Validator/Rules/UniqueDirectivesPerLocation.php @@ -5,13 +5,21 @@ namespace GraphQL\Validator\Rules; use GraphQL\Error\Error; +use GraphQL\Language\AST\DirectiveDefinitionNode; use GraphQL\Language\AST\DirectiveNode; use GraphQL\Language\AST\Node; +use GraphQL\Type\Definition\Directive; use GraphQL\Validator\ASTValidationContext; use GraphQL\Validator\SDLValidationContext; use GraphQL\Validator\ValidationContext; use function sprintf; +/** + * Unique directive names per location + * + * A GraphQL document is only valid if all non-repeatable directives at + * a given location are uniquely named. + */ class UniqueDirectivesPerLocation extends ValidationRule { public function getVisitor(ValidationContext $context) @@ -26,16 +34,41 @@ public function getSDLVisitor(SDLValidationContext $context) public function getASTVisitor(ASTValidationContext $context) { + $uniqueDirectiveMap = []; + + $schema = $context->getSchema(); + $definedDirectives = $schema !== null + ? $schema->getDirectives() + : Directive::getInternalDirectives(); + foreach ($definedDirectives as $directive) { + $uniqueDirectiveMap[$directive->name] = ! $directive->isRepeatable; + } + + $astDefinitions = $context->getDocument()->definitions; + foreach ($astDefinitions as $definition) { + if (! ($definition instanceof DirectiveDefinitionNode)) { + continue; + } + + $uniqueDirectiveMap[$definition->name->value] = $definition->repeatable; + } + return [ - 'enter' => static function (Node $node) use ($context) { + 'enter' => static function (Node $node) use ($uniqueDirectiveMap, $context) { if (! isset($node->directives)) { return; } $knownDirectives = []; + /** @var DirectiveNode $directive */ foreach ($node->directives as $directive) { $directiveName = $directive->name->value; + + if (! isset($uniqueDirectiveMap[$directiveName])) { + continue; + } + if (isset($knownDirectives[$directiveName])) { $context->reportError(new Error( self::duplicateDirectiveMessage($directiveName), diff --git a/tests/Language/SchemaParserTest.php b/tests/Language/SchemaParserTest.php index ca2677a97..f0ce10627 100644 --- a/tests/Language/SchemaParserTest.php +++ b/tests/Language/SchemaParserTest.php @@ -6,6 +6,7 @@ use GraphQL\Error\SyntaxError; use GraphQL\Language\AST\NodeKind; +use GraphQL\Language\DirectiveLocation; use GraphQL\Language\Parser; use GraphQL\Language\SourceLocation; use GraphQL\Tests\PHPUnit\ArraySubsetAsserts; @@ -1045,6 +1046,86 @@ public function testSimpleInputObjectWithArgsShouldFail() : void ); } + /** + * @see it('Directive definition', () => { + */ + public function testDirectiveDefinition() : void + { + $body = 'directive @foo on OBJECT | INTERFACE'; + $doc = Parser::parse($body); + $loc = static function ($start, $end): array { + return TestUtils::locArray($start, $end); + }; + + $expected = [ + 'kind' => NodeKind::DOCUMENT, + 'definitions' => [ + [ + 'kind' => NodeKind::DIRECTIVE_DEFINITION, + 'description' => null, + 'name' => $this->nameNode('foo', $loc(11, 14)), + 'arguments' => [], + 'repeatable' => false, + 'locations' => [ + [ + 'kind' => NodeKind::NAME, + 'value' => DirectiveLocation::OBJECT, + 'loc' => $loc(18, 24), + ], + [ + 'kind' => NodeKind::NAME, + 'value' => DirectiveLocation::IFACE, + 'loc' => $loc(27, 36), + ], + ], + 'loc' => $loc(0, 36), + ], + ], + 'loc' => $loc(0, 36), + ]; + self::assertEquals($expected, TestUtils::nodeToArray($doc)); + } + + /** + * @see it('Repeatable directive definition', () => { + */ + public function testRepeatableDirectiveDefinition() : void + { + $body = 'directive @foo repeatable on OBJECT | INTERFACE'; + $doc = Parser::parse($body); + $loc = static function ($start, $end): array { + return TestUtils::locArray($start, $end); + }; + + $expected = [ + 'kind' => NodeKind::DOCUMENT, + 'definitions' => [ + [ + 'kind' => NodeKind::DIRECTIVE_DEFINITION, + 'description' => null, + 'name' => $this->nameNode('foo', $loc(11, 14)), + 'arguments' => [], + 'repeatable' => true, + 'locations' => [ + [ + 'kind' => NodeKind::NAME, + 'value' => DirectiveLocation::OBJECT, + 'loc' => $loc(29, 35), + ], + [ + 'kind' => NodeKind::NAME, + 'value' => DirectiveLocation::IFACE, + 'loc' => $loc(38, 47), + ], + ], + 'loc' => $loc(0, 47), + ], + ], + 'loc' => $loc(0, 47), + ]; + self::assertEquals($expected, TestUtils::nodeToArray($doc)); + } + /** * @see it('Directive with incorrect locations') */ diff --git a/tests/Language/SchemaPrinterTest.php b/tests/Language/SchemaPrinterTest.php index b5769835a..05a326189 100644 --- a/tests/Language/SchemaPrinterTest.php +++ b/tests/Language/SchemaPrinterTest.php @@ -176,6 +176,8 @@ enum UndefinedEnum directive @include(if: Boolean!) on FIELD | FRAGMENT_SPREAD | INLINE_FRAGMENT directive @include2(if: Boolean!) on FIELD | FRAGMENT_SPREAD | INLINE_FRAGMENT + +directive @myRepeatableDir(name: String!) repeatable on OBJECT | INTERFACE '; self::assertEquals($expected, $printed); } diff --git a/tests/Language/schema-kitchen-sink.graphql b/tests/Language/schema-kitchen-sink.graphql index 3ad830cbf..b912f06d5 100644 --- a/tests/Language/schema-kitchen-sink.graphql +++ b/tests/Language/schema-kitchen-sink.graphql @@ -123,3 +123,7 @@ directive @include2(if: Boolean!) on | FIELD | FRAGMENT_SPREAD | INLINE_FRAGMENT + +directive @myRepeatableDir(name: String!) repeatable on + | OBJECT + | INTERFACE diff --git a/tests/Type/DirectiveTest.php b/tests/Type/DirectiveTest.php new file mode 100644 index 000000000..b94efe136 --- /dev/null +++ b/tests/Type/DirectiveTest.php @@ -0,0 +1,15 @@ + { + */ +class DirectiveTest extends TestCase +{ + // TODO implement https://github.com/graphql/graphql-js/blob/master/src/type/__tests__/directive-test.js +} diff --git a/tests/Utils/BuildSchemaTest.php b/tests/Utils/BuildSchemaTest.php index 3990093fa..7da00b10a 100644 --- a/tests/Utils/BuildSchemaTest.php +++ b/tests/Utils/BuildSchemaTest.php @@ -106,6 +106,8 @@ public function testWithDirectives() : void $body = ' directive @foo(arg: Int) on FIELD +directive @repeatableFoo(arg: Int) repeatable on FIELD + type Query { str: String } diff --git a/tests/Utils/SchemaExtenderTest.php b/tests/Utils/SchemaExtenderTest.php index 61cdb18a0..3a8656fb1 100644 --- a/tests/Utils/SchemaExtenderTest.php +++ b/tests/Utils/SchemaExtenderTest.php @@ -522,7 +522,7 @@ interfaceField: String type TestType implements TestInterface { interfaceField: String } - directive @test(arg: Int) on FIELD | SCALAR + directive @test(arg: Int) repeatable on FIELD | SCALAR '); $extendedTwiceSchema = SchemaExtender::extend($extendedSchema, $ast); @@ -1263,7 +1263,7 @@ public function testSetsCorrectDescriptionUsingLegacyComments() public function testMayExtendDirectivesWithNewComplexDirective() { $extendedSchema = $this->extendTestSchema(' - directive @profile(enable: Boolean! tag: String) on QUERY | FIELD + directive @profile(enable: Boolean! tag: String) repeatable on QUERY | FIELD '); $extendedDirective = $extendedSchema->getDirective('profile'); diff --git a/tests/Utils/SchemaPrinterTest.php b/tests/Utils/SchemaPrinterTest.php index ab54cdf1b..15b7500f6 100644 --- a/tests/Utils/SchemaPrinterTest.php +++ b/tests/Utils/SchemaPrinterTest.php @@ -706,26 +706,50 @@ public function testPrintsCustomDirectives() : void $query = new ObjectType([ 'name' => 'Query', 'fields' => [ - 'field' => ['type' => Type::string()], + 'field' => [ + 'type' => Type::string(), + ], ], ]); - $customDirectives = new Directive([ - 'name' => 'customDirective', + $simpleDirective = new Directive([ + 'name' => 'simpleDirective', 'locations' => [ DirectiveLocation::FIELD, ], ]); + $complexDirective = new Directive([ + 'name' => 'complexDirective', + 'description' => 'Complex Directive', + 'args' => [ + 'stringArg' => [ + 'type' => Type::string(), + ], + 'intArg' => [ + 'type' => Type::int(), + 'defaultValue' => -1, + ], + ], + 'isRepeatable' => true, + 'locations' => [ + DirectiveLocation::FIELD, + DirectiveLocation::QUERY, + ], + ]); + $schema = new Schema([ 'query' => $query, - 'directives' => [$customDirectives], + 'directives' => [$simpleDirective, $complexDirective], ]); $output = $this->printForTest($schema); self::assertEquals( ' -directive @customDirective on FIELD +directive @simpleDirective on FIELD + +"""Complex Directive""" +directive @complexDirective(stringArg: String, intArg: Int = -1) repeatable on FIELD | QUERY type Query { field: String diff --git a/tests/Validator/UniqueDirectivesPerLocationTest.php b/tests/Validator/UniqueDirectivesPerLocationTest.php index c71d02c65..5e7d656c6 100644 --- a/tests/Validator/UniqueDirectivesPerLocationTest.php +++ b/tests/Validator/UniqueDirectivesPerLocationTest.php @@ -89,6 +89,39 @@ public function testSameDirectivesInSimilarLocations() : void ); } + /** + * @see it('repeatable directives in same location', () => { + */ + public function testRepeatableDirectivesInSameLocation() : void + { + $this->expectPassesRule( + new UniqueDirectivesPerLocation(), + ' + fragment Test on Type @repeatable @repeatable { + field @repeatable @repeatable + } + ' + ); + } + + /** + * @see it('unknown directives must be ignored', () => { + */ + public function testUnknownDirectivesMustBeIgnored() : void + { + $this->expectPassesRule( + new UniqueDirectivesPerLocation(), + ' + type Test @unknown @unknown { + field: String! @unknown @unknown + } + extend type Test @unknown { + anotherField: String! + } + ' + ); + } + /** * @see it('duplicate directives in one location') */ @@ -180,38 +213,25 @@ public function testDuplicateDirectivesOnSDLDefinitions() { $this->expectSDLErrors( ' - schema @directive @directive { query: Dummy } - extend schema @directive @directive - - scalar TestScalar @directive @directive - extend scalar TestScalar @directive @directive - - type TestObject @directive @directive - extend type TestObject @directive @directive - - interface TestInterface @directive @directive - extend interface TestInterface @directive @directive + directive @nonRepeatable on + SCHEMA | SCALAR | OBJECT | INTERFACE | UNION | INPUT_OBJECT - union TestUnion @directive @directive - extend union TestUnion @directive @directive + schema @nonRepeatable @nonRepeatable { query: Dummy } - input TestInput @directive @directive - extend input TestInput @directive @directive + scalar TestScalar @nonRepeatable @nonRepeatable + type TestObject @nonRepeatable @nonRepeatable + interface TestInterface @nonRepeatable @nonRepeatable + union TestUnion @nonRepeatable @nonRepeatable + input TestInput @nonRepeatable @nonRepeatable ', null, [ - $this->duplicateDirective('directive', 2, 14, 2, 25), - $this->duplicateDirective('directive', 3, 21, 3, 32), - $this->duplicateDirective('directive', 5, 25, 5, 36), - $this->duplicateDirective('directive', 6, 32, 6, 43), - $this->duplicateDirective('directive', 8, 23, 8, 34), - $this->duplicateDirective('directive', 9, 30, 9, 41), - $this->duplicateDirective('directive', 11, 31, 11, 42), - $this->duplicateDirective('directive', 12, 38, 12, 49), - $this->duplicateDirective('directive', 14, 23, 14, 34), - $this->duplicateDirective('directive', 15, 30, 15, 41), - $this->duplicateDirective('directive', 17, 23, 17, 34), - $this->duplicateDirective('directive', 18, 30, 18, 41), + $this->duplicateDirective('nonRepeatable', 5, 14, 5, 29), + $this->duplicateDirective('nonRepeatable', 7, 25, 7, 40), + $this->duplicateDirective('nonRepeatable', 8, 23, 8, 38), + $this->duplicateDirective('nonRepeatable', 9, 31, 9, 46), + $this->duplicateDirective('nonRepeatable', 10, 23, 10, 38), + $this->duplicateDirective('nonRepeatable', 11, 23, 11, 38), ] ); } diff --git a/tests/Validator/ValidatorTestCase.php b/tests/Validator/ValidatorTestCase.php index 641039835..69062838c 100644 --- a/tests/Validator/ValidatorTestCase.php +++ b/tests/Validator/ValidatorTestCase.php @@ -5,6 +5,7 @@ namespace GraphQL\Tests\Validator; use Exception; +use GraphQL\Language\DirectiveLocation; use GraphQL\Language\Parser; use GraphQL\Type\Definition\CustomScalarType; use GraphQL\Type\Definition\Directive; @@ -354,37 +355,49 @@ public static function getTestSchema() 'directives' => [ Directive::includeDirective(), Directive::skipDirective(), + new Directive([ + 'name' => 'directive', + 'locations' => [DirectiveLocation::FIELD], + ]), + new Directive([ + 'name' => 'directiveA', + 'locations' => [DirectiveLocation::FIELD], + ]), + new Directive([ + 'name' => 'directiveB', + 'locations' => [DirectiveLocation::FIELD], + ]), new Directive([ 'name' => 'onQuery', - 'locations' => ['QUERY'], + 'locations' => [DirectiveLocation::QUERY], ]), new Directive([ 'name' => 'onMutation', - 'locations' => ['MUTATION'], + 'locations' => [DirectiveLocation::MUTATION], ]), new Directive([ 'name' => 'onSubscription', - 'locations' => ['SUBSCRIPTION'], + 'locations' => [DirectiveLocation::SUBSCRIPTION], ]), new Directive([ 'name' => 'onField', - 'locations' => ['FIELD'], + 'locations' => [DirectiveLocation::FIELD], ]), new Directive([ 'name' => 'onFragmentDefinition', - 'locations' => ['FRAGMENT_DEFINITION'], + 'locations' => [DirectiveLocation::FRAGMENT_DEFINITION], ]), new Directive([ 'name' => 'onFragmentSpread', - 'locations' => ['FRAGMENT_SPREAD'], + 'locations' => [DirectiveLocation::FRAGMENT_SPREAD], ]), new Directive([ 'name' => 'onInlineFragment', - 'locations' => ['INLINE_FRAGMENT'], + 'locations' => [DirectiveLocation::INLINE_FRAGMENT], ]), new Directive([ 'name' => 'onVariableDefinition', - 'locations' => ['VARIABLE_DEFINITION'], + 'locations' => [DirectiveLocation::VARIABLE_DEFINITION], ]), ], ]); From 8722b27c77a9444f99662509037fa7fc3e7705dd Mon Sep 17 00:00:00 2001 From: spawnia Date: Mon, 13 Apr 2020 21:00:13 +0200 Subject: [PATCH 2/7] Fix introspection See https://github.com/graphql/graphql-js/pull/2416 --- src/Type/Introspection.php | 84 ++++++++++++++++----------- tests/Type/IntrospectionTest.php | 72 +++++++++++++++-------- tests/Utils/BuildClientSchemaTest.php | 10 ++-- 3 files changed, 104 insertions(+), 62 deletions(-) diff --git a/src/Type/Introspection.php b/src/Type/Introspection.php index 5bd928b31..519c43557 100644 --- a/src/Type/Introspection.php +++ b/src/Type/Introspection.php @@ -43,28 +43,28 @@ class Introspection private static $map = []; /** - * Options: - * - descriptions - * Whether to include descriptions in the introspection result. - * Default: true - * - * @param bool[]|bool $options + * @param array $options + * Available options: + * - descriptions + * Whether to include descriptions in the introspection result. + * Default: true + * - directiveIsRepeatable + * Whether to include `isRepeatable` flag on directives. + * Default: false * * @return string + * + * @api */ - public static function getIntrospectionQuery($options = []) + public static function getIntrospectionQuery(array $options = []) { - if (is_bool($options)) { - trigger_error( - 'Calling Introspection::getIntrospectionQuery(boolean) is deprecated. ' . - 'Please use Introspection::getIntrospectionQuery(["descriptions" => boolean]).', - E_USER_DEPRECATED - ); - $descriptions = $options; - } else { - $descriptions = ! array_key_exists('descriptions', $options) || $options['descriptions'] === true; - } - $descriptionField = $descriptions ? 'description' : ''; + $optionsWithDefaults = array_merge([ + 'descriptions' => true, + 'directiveIsRepeatable' => false, + ], $options); + + $descriptions = $optionsWithDefaults['descriptions'] ? 'description' : ''; + $directiveIsRepeatable = $optionsWithDefaults['directiveIsRepeatable'] ? 'isRepeatable' : ''; return << $options + * Available options: + * - descriptions + * Whether to include `isRepeatable` flag on directives. + * Default: true + * - directiveIsRepeatable + * Whether to include descriptions in the introspection result. + * Default: true * * @return array>|null + * + * @api */ public static function fromSchema(Schema $schema, array $options = []) : ?array { + $optionsWithDefaults = array_merge([ + 'directiveIsRepeatable' => true, + ], $options); + $result = GraphQL::executeQuery( $schema, - self::getIntrospectionQuery($options) + self::getIntrospectionQuery($optionsWithDefaults) ); return $result->data; @@ -651,12 +661,10 @@ public static function _directive() return $obj->description; }, ], - 'locations' => [ - 'type' => Type::nonNull(Type::listOf(Type::nonNull( - self::_directiveLocation() - ))), - 'resolve' => static function ($obj) { - return $obj->locations; + 'isRepeatable' => [ + 'type' => Type::nonNull(Type::boolean()), + 'resolve' => static function (Directive $directive): bool { + return $directive->isRepeatable; }, ], 'args' => [ @@ -665,6 +673,14 @@ public static function _directive() return $directive->args ?: []; }, ], + 'locations' => [ + 'type' => Type::nonNull(Type::listOf(Type::nonNull( + self::_directiveLocation() + ))), + 'resolve' => static function ($obj) { + return $obj->locations; + }, + ], ], ]); } diff --git a/tests/Type/IntrospectionTest.php b/tests/Type/IntrospectionTest.php index fac1c9934..20bc425a4 100644 --- a/tests/Type/IntrospectionTest.php +++ b/tests/Type/IntrospectionTest.php @@ -35,7 +35,10 @@ public function testExecutesAnIntrospectionQuery() : void ]), ]); - $request = Introspection::getIntrospectionQuery(['descriptions' => false]); + $request = Introspection::getIntrospectionQuery([ + 'descriptions' => false, + 'directiveIsRepeatable' => true, + ]); $expected = [ 'data' => [ @@ -794,7 +797,25 @@ public function testExecutesAnIntrospectionQuery() : void ], 2 => [ - 'name' => 'locations', + 'name' => 'isRepeatable', + 'args' => + [], + 'type' => + [ + 'kind' => 'NON_NULL', + 'name' => 'null', + 'ofType' => [ + 'kind' => 'SCALAR', + 'name' => 'Boolean', + 'ofType' => null, + ] + ], + 'isDeprecated' => false, + 'deprecationReason' => null, + ], + 3 => + [ + 'name' => 'args', 'args' => [], 'type' => @@ -811,8 +832,8 @@ public function testExecutesAnIntrospectionQuery() : void 'name' => null, 'ofType' => [ - 'kind' => 'ENUM', - 'name' => '__DirectiveLocation', + 'kind' => 'OBJECT', + 'name' => '__InputValue', ], ], ], @@ -820,9 +841,9 @@ public function testExecutesAnIntrospectionQuery() : void 'isDeprecated' => false, 'deprecationReason' => null, ], - 3 => + 4 => [ - 'name' => 'args', + 'name' => 'locations', 'args' => [], 'type' => @@ -839,8 +860,8 @@ public function testExecutesAnIntrospectionQuery() : void 'name' => null, 'ofType' => [ - 'kind' => 'OBJECT', - 'name' => '__InputValue', + 'kind' => 'ENUM', + 'name' => '__DirectiveLocation', ], ], ], @@ -975,12 +996,7 @@ public function testExecutesAnIntrospectionQuery() : void 0 => [ 'name' => 'include', - 'locations' => - [ - 0 => 'FIELD', - 1 => 'FRAGMENT_SPREAD', - 2 => 'INLINE_FRAGMENT', - ], + 'isRepeatable' => false, 'args' => [ 0 => @@ -999,16 +1015,17 @@ public function testExecutesAnIntrospectionQuery() : void ], ], ], - ], - 1 => - [ - 'name' => 'skip', 'locations' => [ 0 => 'FIELD', 1 => 'FRAGMENT_SPREAD', 2 => 'INLINE_FRAGMENT', ], + ], + 1 => + [ + 'name' => 'skip', + 'isRepeatable' => false, 'args' => [ 0 => @@ -1027,15 +1044,17 @@ public function testExecutesAnIntrospectionQuery() : void ], ], ], + 'locations' => + [ + 0 => 'FIELD', + 1 => 'FRAGMENT_SPREAD', + 2 => 'INLINE_FRAGMENT', + ], ], 2 => [ 'name' => 'deprecated', - 'locations' => - [ - 0 => 'FIELD_DEFINITION', - 1 => 'ENUM_VALUE', - ], + 'isRepeatable' => false, 'args' => [ 0 => @@ -1050,6 +1069,11 @@ public function testExecutesAnIntrospectionQuery() : void ], ], ], + 'locations' => + [ + 0 => 'FIELD_DEFINITION', + 1 => 'ENUM_VALUE', + ], ], ], ], @@ -1608,7 +1632,7 @@ public function testExecutesAnIntrospectionQueryWithoutCallingGlobalFieldResolve ]); $schema = new Schema([ 'query' => $QueryRoot ]); - $source = Introspection::getIntrospectionQuery(); + $source = Introspection::getIntrospectionQuery(['directiveIsRepeatable' => true]); $calledForFields = []; /* istanbul ignore next */ diff --git a/tests/Utils/BuildClientSchemaTest.php b/tests/Utils/BuildClientSchemaTest.php index 814f69dd7..0d82fd4f3 100644 --- a/tests/Utils/BuildClientSchemaTest.php +++ b/tests/Utils/BuildClientSchemaTest.php @@ -23,10 +23,12 @@ class BuildClientSchemaTest extends TestCase { protected static function assertCycleIntrospection(string $sdl) : void { + $options = ['directiveIsRepeatable' => true]; + $serverSchema = BuildSchema::build($sdl); - $initialIntrospection = Introspection::fromSchema($serverSchema); + $initialIntrospection = Introspection::fromSchema($serverSchema, $options); $clientSchema = BuildClientSchema::build($initialIntrospection); - $secondIntrospection = Introspection::fromSchema($clientSchema); + $secondIntrospection = Introspection::fromSchema($clientSchema, $options); self::assertSame($initialIntrospection, $secondIntrospection); } @@ -489,8 +491,8 @@ public function testBuildsASchemaWithCustomDirectives() : void { self::assertCycleIntrospection(' """This is a custom directive""" - directive @customDirective on FIELD - + directive @customDirective repeatable on FIELD + type Query { string: String } From ecfa59e6b2083f573fc94abd906c44269e7ccb2f Mon Sep 17 00:00:00 2001 From: spawnia Date: Mon, 13 Apr 2020 21:01:50 +0200 Subject: [PATCH 3/7] Fix codestyle --- src/Type/Introspection.php | 10 ++++------ src/Utils/SchemaPrinter.php | 18 +++++++++--------- tests/Language/SchemaParserTest.php | 4 ++-- tests/Type/IntrospectionTest.php | 2 +- 4 files changed, 16 insertions(+), 18 deletions(-) diff --git a/src/Type/Introspection.php b/src/Type/Introspection.php index 519c43557..ff04ef20e 100644 --- a/src/Type/Introspection.php +++ b/src/Type/Introspection.php @@ -27,6 +27,7 @@ use GraphQL\Utils\Utils; use function array_filter; use function array_key_exists; +use function array_merge; use function array_values; use function is_bool; use function method_exists; @@ -63,7 +64,7 @@ public static function getIntrospectionQuery(array $options = []) 'directiveIsRepeatable' => false, ], $options); - $descriptions = $optionsWithDefaults['descriptions'] ? 'description' : ''; + $descriptions = $optionsWithDefaults['descriptions'] ? 'description' : ''; $directiveIsRepeatable = $optionsWithDefaults['directiveIsRepeatable'] ? 'isRepeatable' : ''; return << $options * Available options: * - descriptions @@ -211,9 +211,7 @@ public static function getTypes() */ public static function fromSchema(Schema $schema, array $options = []) : ?array { - $optionsWithDefaults = array_merge([ - 'directiveIsRepeatable' => true, - ], $options); + $optionsWithDefaults = array_merge(['directiveIsRepeatable' => true], $options); $result = GraphQL::executeQuery( $schema, @@ -663,7 +661,7 @@ public static function _directive() ], 'isRepeatable' => [ 'type' => Type::nonNull(Type::boolean()), - 'resolve' => static function (Directive $directive): bool { + 'resolve' => static function (Directive $directive) : bool { return $directive->isRepeatable; }, ], diff --git a/src/Utils/SchemaPrinter.php b/src/Utils/SchemaPrinter.php index acc612ae6..2de0c8456 100644 --- a/src/Utils/SchemaPrinter.php +++ b/src/Utils/SchemaPrinter.php @@ -49,10 +49,10 @@ public static function doPrint(Schema $schema, array $options = []) : string { return self::printFilteredSchema( $schema, - static function ($type): bool { + static function ($type) : bool { return ! Directive::isSpecifiedDirective($type); }, - static function ($type): bool { + static function ($type) : bool { return ! Type::isBuiltInType($type); }, $options @@ -84,7 +84,7 @@ static function (Directive $directive) use ($options) : string { $directives ), array_map( - static function ($type) use ($options): bool { + static function ($type) use ($options) : bool { return self::printType($type, $options); }, $types @@ -277,7 +277,7 @@ private static function printArgs(array $options, $args, $indentation = '') : st // If every arg does not have a description, print them on one line. if (Utils::every( $args, - static function ($arg): bool { + static function ($arg) : bool { return empty($arg->description); } )) { @@ -289,7 +289,7 @@ static function ($arg): bool { implode( "\n", array_map( - static function ($arg, $i) use ($indentation, $options): string { + static function ($arg, $i) use ($indentation, $options) : string { return self::printDescription($options, $arg, ' ' . $indentation, ! $i) . ' ' . $indentation . self::printInputValue($arg); }, @@ -361,7 +361,7 @@ private static function printObject(ObjectType $type, array $options) : string ? ' implements ' . implode( ' & ', array_map( - static function (InterfaceType $interface): string { + static function (InterfaceType $interface) : string { return $interface->name; }, $interfaces @@ -383,7 +383,7 @@ private static function printFields(array $options, $type) : string return implode( "\n", array_map( - static function ($f, $i) use ($options): string { + static function ($f, $i) use ($options) : string { return self::printDescription($options, $f, ' ', ! $i) . ' ' . $f->name . self::printArgs($options, $f->args, ' ') . ': ' . (string) $f->getType() . self::printDeprecated($f); @@ -443,7 +443,7 @@ private static function printEnumValues($values, array $options) : string return implode( "\n", array_map( - static function ($value, $i) use ($options): string { + static function ($value, $i) use ($options) : string { return self::printDescription($options, $value, ' ', ! $i) . ' ' . $value->name . self::printDeprecated($value); }, @@ -467,7 +467,7 @@ private static function printInputObject(InputObjectType $type, array $options) implode( "\n", array_map( - static function ($f, $i) use ($options): string { + static function ($f, $i) use ($options) : string { return self::printDescription($options, $f, ' ', ! $i) . ' ' . self::printInputValue($f); }, $fields, diff --git a/tests/Language/SchemaParserTest.php b/tests/Language/SchemaParserTest.php index f0ce10627..7941e4872 100644 --- a/tests/Language/SchemaParserTest.php +++ b/tests/Language/SchemaParserTest.php @@ -1053,7 +1053,7 @@ public function testDirectiveDefinition() : void { $body = 'directive @foo on OBJECT | INTERFACE'; $doc = Parser::parse($body); - $loc = static function ($start, $end): array { + $loc = static function ($start, $end) : array { return TestUtils::locArray($start, $end); }; @@ -1093,7 +1093,7 @@ public function testRepeatableDirectiveDefinition() : void { $body = 'directive @foo repeatable on OBJECT | INTERFACE'; $doc = Parser::parse($body); - $loc = static function ($start, $end): array { + $loc = static function ($start, $end) : array { return TestUtils::locArray($start, $end); }; diff --git a/tests/Type/IntrospectionTest.php b/tests/Type/IntrospectionTest.php index 20bc425a4..2f577c11d 100644 --- a/tests/Type/IntrospectionTest.php +++ b/tests/Type/IntrospectionTest.php @@ -808,7 +808,7 @@ public function testExecutesAnIntrospectionQuery() : void 'kind' => 'SCALAR', 'name' => 'Boolean', 'ofType' => null, - ] + ], ], 'isDeprecated' => false, 'deprecationReason' => null, From bccb5b761b5e7d671a4131689fce236750142c90 Mon Sep 17 00:00:00 2001 From: spawnia Date: Mon, 13 Apr 2020 21:36:35 +0200 Subject: [PATCH 4/7] Canonical ordering --- src/Language/AST/DirectiveDefinitionNode.php | 6 ++-- src/Language/Parser.php | 2 +- src/Type/Definition/Directive.php | 16 ++++----- src/Type/Introspection.php | 16 ++++----- src/Utils/ASTDefinitionBuilder.php | 4 +-- src/Utils/BuildClientSchema.php | 3 +- src/Utils/SchemaPrinter.php | 2 +- tests/Language/SchemaParserTest.php | 4 +-- tests/Type/IntrospectionTest.php | 36 ++++++++++---------- tests/Utils/SchemaPrinterTest.php | 6 ++-- 10 files changed, 49 insertions(+), 46 deletions(-) diff --git a/src/Language/AST/DirectiveDefinitionNode.php b/src/Language/AST/DirectiveDefinitionNode.php index 96b0c542c..26aa23e3c 100644 --- a/src/Language/AST/DirectiveDefinitionNode.php +++ b/src/Language/AST/DirectiveDefinitionNode.php @@ -12,6 +12,9 @@ class DirectiveDefinitionNode extends Node implements TypeSystemDefinitionNode /** @var NameNode */ public $name; + /** @var StringValueNode|null */ + public $description; + /** @var ArgumentNode[] */ public $arguments; @@ -20,7 +23,4 @@ class DirectiveDefinitionNode extends Node implements TypeSystemDefinitionNode /** @var NameNode[] */ public $locations; - - /** @var StringValueNode|null */ - public $description; } diff --git a/src/Language/Parser.php b/src/Language/Parser.php index 202d8ede5..52ea5882a 100644 --- a/src/Language/Parser.php +++ b/src/Language/Parser.php @@ -1693,11 +1693,11 @@ private function parseDirectiveDefinition() : DirectiveDefinitionNode return new DirectiveDefinitionNode([ 'name' => $name, + 'description' => $description, 'arguments' => $args, 'repeatable' => $repeatable, 'locations' => $locations, 'loc' => $this->loc($start), - 'description' => $description, ]); } diff --git a/src/Type/Definition/Directive.php b/src/Type/Definition/Directive.php index 727ca63e7..c27025c50 100644 --- a/src/Type/Definition/Directive.php +++ b/src/Type/Definition/Directive.php @@ -31,15 +31,15 @@ class Directive /** @var string|null */ public $description; - /** @var string[] */ - public $locations; - /** @var FieldArgument[] */ public $args = []; /** @var bool */ public $isRepeatable; + /** @var string[] */ + public $locations; + /** @var DirectiveDefinitionNode|null */ public $astNode; @@ -58,11 +58,6 @@ public function __construct(array $config) $this->description = $config['description'] ?? null; - if (! isset($config['locations']) || ! is_array($config['locations'])) { - throw new InvariantViolation('Must provide locations for directive.'); - } - $this->locations = $config['locations']; - if (isset($config['args'])) { $args = []; foreach ($config['args'] as $name => $arg) { @@ -75,6 +70,11 @@ public function __construct(array $config) $this->args = $args; } + if (! isset($config['locations']) || ! is_array($config['locations'])) { + throw new InvariantViolation('Must provide locations for directive.'); + } + $this->locations = $config['locations']; + $this->isRepeatable = $config['isRepeatable'] ?? false; $this->astNode = $config['astNode'] ?? null; diff --git a/src/Type/Introspection.php b/src/Type/Introspection.php index ff04ef20e..8508fda2f 100644 --- a/src/Type/Introspection.php +++ b/src/Type/Introspection.php @@ -79,11 +79,11 @@ public static function getIntrospectionQuery(array $options = []) directives { name {$descriptions} - {$directiveIsRepeatable} - locations args { ...InputValue } + {$directiveIsRepeatable} + locations } } } @@ -659,18 +659,18 @@ public static function _directive() return $obj->description; }, ], - 'isRepeatable' => [ - 'type' => Type::nonNull(Type::boolean()), - 'resolve' => static function (Directive $directive) : bool { - return $directive->isRepeatable; - }, - ], 'args' => [ 'type' => Type::nonNull(Type::listOf(Type::nonNull(self::_inputValue()))), 'resolve' => static function (Directive $directive) { return $directive->args ?: []; }, ], + 'isRepeatable' => [ + 'type' => Type::nonNull(Type::boolean()), + 'resolve' => static function (Directive $directive) : bool { + return $directive->isRepeatable; + }, + ], 'locations' => [ 'type' => Type::nonNull(Type::listOf(Type::nonNull( self::_directiveLocation() diff --git a/src/Utils/ASTDefinitionBuilder.php b/src/Utils/ASTDefinitionBuilder.php index 71e934961..d5157fe96 100644 --- a/src/Utils/ASTDefinitionBuilder.php +++ b/src/Utils/ASTDefinitionBuilder.php @@ -79,14 +79,14 @@ public function buildDirective(DirectiveDefinitionNode $directiveNode) return new Directive([ 'name' => $directiveNode->name->value, 'description' => $this->getDescription($directiveNode), + 'args' => $directiveNode->arguments ? FieldArgument::createMap($this->makeInputValues($directiveNode->arguments)) : null, + 'isRepeatable' => $directiveNode->repeatable, 'locations' => Utils::map( $directiveNode->locations, static function ($node) { return $node->value; } ), - 'args' => $directiveNode->arguments ? FieldArgument::createMap($this->makeInputValues($directiveNode->arguments)) : null, - 'isRepeatable' => $directiveNode->repeatable, 'astNode' => $directiveNode, ]); } diff --git a/src/Utils/BuildClientSchema.php b/src/Utils/BuildClientSchema.php index 57cef3284..a013c1ac1 100644 --- a/src/Utils/BuildClientSchema.php +++ b/src/Utils/BuildClientSchema.php @@ -472,8 +472,9 @@ public function buildDirective(array $directive) : Directive return new Directive([ 'name' => $directive['name'], 'description' => $directive['description'], - 'locations' => $directive['locations'], 'args' => $this->buildInputValueDefMap($directive['args']), + 'isRepeatable' => $directive['isRepeatable'], + 'locations' => $directive['locations'], ]); } } diff --git a/src/Utils/SchemaPrinter.php b/src/Utils/SchemaPrinter.php index 2de0c8456..cc68d26c8 100644 --- a/src/Utils/SchemaPrinter.php +++ b/src/Utils/SchemaPrinter.php @@ -84,7 +84,7 @@ static function (Directive $directive) use ($options) : string { $directives ), array_map( - static function ($type) use ($options) : bool { + static function ($type) use ($options) : string { return self::printType($type, $options); }, $types diff --git a/tests/Language/SchemaParserTest.php b/tests/Language/SchemaParserTest.php index 7941e4872..e6695c802 100644 --- a/tests/Language/SchemaParserTest.php +++ b/tests/Language/SchemaParserTest.php @@ -1062,8 +1062,8 @@ public function testDirectiveDefinition() : void 'definitions' => [ [ 'kind' => NodeKind::DIRECTIVE_DEFINITION, - 'description' => null, 'name' => $this->nameNode('foo', $loc(11, 14)), + 'description' => null, 'arguments' => [], 'repeatable' => false, 'locations' => [ @@ -1102,8 +1102,8 @@ public function testRepeatableDirectiveDefinition() : void 'definitions' => [ [ 'kind' => NodeKind::DIRECTIVE_DEFINITION, - 'description' => null, 'name' => $this->nameNode('foo', $loc(11, 14)), + 'description' => null, 'arguments' => [], 'repeatable' => true, 'locations' => [ diff --git a/tests/Type/IntrospectionTest.php b/tests/Type/IntrospectionTest.php index 2f577c11d..de8363172 100644 --- a/tests/Type/IntrospectionTest.php +++ b/tests/Type/IntrospectionTest.php @@ -796,24 +796,6 @@ public function testExecutesAnIntrospectionQuery() : void 'deprecationReason' => null, ], 2 => - [ - 'name' => 'isRepeatable', - 'args' => - [], - 'type' => - [ - 'kind' => 'NON_NULL', - 'name' => 'null', - 'ofType' => [ - 'kind' => 'SCALAR', - 'name' => 'Boolean', - 'ofType' => null, - ], - ], - 'isDeprecated' => false, - 'deprecationReason' => null, - ], - 3 => [ 'name' => 'args', 'args' => @@ -841,6 +823,24 @@ public function testExecutesAnIntrospectionQuery() : void 'isDeprecated' => false, 'deprecationReason' => null, ], + 3 => + [ + 'name' => 'isRepeatable', + 'args' => + [], + 'type' => + [ + 'kind' => 'NON_NULL', + 'name' => null, + 'ofType' => [ + 'kind' => 'SCALAR', + 'name' => 'Boolean', + 'ofType' => null, + ], + ], + 'isDeprecated' => false, + 'deprecationReason' => null, + ], 4 => [ 'name' => 'locations', diff --git a/tests/Utils/SchemaPrinterTest.php b/tests/Utils/SchemaPrinterTest.php index 15b7500f6..be8c4d7db 100644 --- a/tests/Utils/SchemaPrinterTest.php +++ b/tests/Utils/SchemaPrinterTest.php @@ -895,8 +895,9 @@ public function testPrintIntrospectionSchema() : void type __Directive { name: String! description: String - locations: [__DirectiveLocation!]! args: [__InputValue!]! + isRepeatable: Boolean! + locations: [__DirectiveLocation!]! } """ @@ -1135,8 +1136,9 @@ public function testPrintIntrospectionSchemaWithCommentDescriptions() : void type __Directive { name: String! description: String - locations: [__DirectiveLocation!]! args: [__InputValue!]! + isRepeatable: Boolean! + locations: [__DirectiveLocation!]! } # A Directive can be adjacent to many parts of the GraphQL language, a From 705e604cfd616aa4d8bd7c12f168775fdee01b91 Mon Sep 17 00:00:00 2001 From: spawnia Date: Mon, 13 Apr 2020 21:55:48 +0200 Subject: [PATCH 5/7] Update baseline --- phpstan-baseline.neon | 5 ----- 1 file changed, 5 deletions(-) diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index 6ad8a8e67..6050526af 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -885,11 +885,6 @@ parameters: count: 1 path: src/Utils/SchemaExtender.php - - - message: "#^Anonymous function should return bool but returns string\\.$#" - count: 1 - path: src/Utils/SchemaPrinter.php - - message: "#^Construct empty\\(\\) is not allowed\\. Use more strict comparison\\.$#" count: 2 From 4963d013d31821cd43e4310429fcaffe6a0bf9e3 Mon Sep 17 00:00:00 2001 From: Benedikt Franke Date: Mon, 27 Apr 2020 14:28:52 +0200 Subject: [PATCH 6/7] Make DirectiveTest.php final MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-Authored-By: Šimon Podlipský --- tests/Type/DirectiveTest.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/Type/DirectiveTest.php b/tests/Type/DirectiveTest.php index b94efe136..c98265379 100644 --- a/tests/Type/DirectiveTest.php +++ b/tests/Type/DirectiveTest.php @@ -9,7 +9,7 @@ /** * @see describe('Type System: Directive', () => { */ -class DirectiveTest extends TestCase +final class DirectiveTest extends TestCase { // TODO implement https://github.com/graphql/graphql-js/blob/master/src/type/__tests__/directive-test.js } From fda2f80adc1140943944258d83523ac92231f247 Mon Sep 17 00:00:00 2001 From: Benedikt Franke Date: Mon, 8 Jun 2020 19:28:39 +0200 Subject: [PATCH 7/7] Fix baseline --- phpstan-baseline.neon | 5 ----- 1 file changed, 5 deletions(-) diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index f57b1c63d..e47bdc0d7 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -265,11 +265,6 @@ parameters: count: 2 path: src/Server/OperationParams.php - - - message: "#^Variable property access on \\$this\\(GraphQL\\\\Type\\\\Definition\\\\Directive\\)\\.$#" - count: 1 - path: src/Type/Definition/Directive.php - - message: "#^Only booleans are allowed in a negated boolean, ArrayObject\\ given\\.$#" count: 1