Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for specifiedBy directive #1616

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion src/Language/Printer.php
Original file line number Diff line number Diff line change
Expand Up @@ -414,7 +414,8 @@ protected function p(?Node $node): string
return BlockString::print($node->value);
}

return \json_encode($node->value, JSON_THROW_ON_ERROR);
// Do not escape unicode or slashes in order to keep urls valid
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// Do not escape unicode or slashes in order to keep urls valid
// Do not escape unicode or slashes in order to keep URLs valid

return \json_encode($node->value, JSON_THROW_ON_ERROR | JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);

case $node instanceof UnionTypeDefinitionNode:
$typesStr = $this->printList($node->types, ' | ');
Expand Down
2 changes: 2 additions & 0 deletions src/Type/Definition/CustomScalarType.php
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
* serialize?: callable(mixed): mixed,
* parseValue: callable(mixed): mixed,
* parseLiteral: callable(ValueNode&Node, array<string, mixed>|null): mixed,
* specifiedByURL?: string|null,
* astNode?: ScalarTypeDefinitionNode|null,
* extensionASTNodes?: array<ScalarTypeExtensionNode>|null
* }
Expand All @@ -27,6 +28,7 @@
* serialize: callable(mixed): mixed,
* parseValue?: callable(mixed): mixed,
* parseLiteral?: callable(ValueNode&Node, array<string, mixed>|null): mixed,
* specifiedByURL?: string|null,
* astNode?: ScalarTypeDefinitionNode|null,
* extensionASTNodes?: array<ScalarTypeExtensionNode>|null
* }
Expand Down
23 changes: 23 additions & 0 deletions src/Type/Definition/Directive.php
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,9 @@ class Directive
public const IF_ARGUMENT_NAME = 'if';
public const SKIP_NAME = 'skip';
public const DEPRECATED_NAME = 'deprecated';
public const SPECIFIED_BY_NAME = 'specifiedBy';
public const REASON_ARGUMENT_NAME = 'reason';
public const URL_ARGUMENT_NAME = 'url';
Comment on lines 26 to +31
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
public const IF_ARGUMENT_NAME = 'if';
public const SKIP_NAME = 'skip';
public const DEPRECATED_NAME = 'deprecated';
public const SPECIFIED_BY_NAME = 'specifiedBy';
public const REASON_ARGUMENT_NAME = 'reason';
public const URL_ARGUMENT_NAME = 'url';
public const SKIP_NAME = 'skip';
public const IF_ARGUMENT_NAME = 'if';
public const DEPRECATED_NAME = 'deprecated';
public const REASON_ARGUMENT_NAME = 'reason';
public const SPECIFIED_BY_NAME = 'specifiedBy';
public const URL_ARGUMENT_NAME = 'url';


/**
* Lazily initialized.
Expand Down Expand Up @@ -138,6 +140,19 @@ public static function getInternalDirectives(): array
],
],
]),
'specifiedBy' => new self([
'name' => self::SPECIFIED_BY_NAME,
'description' => 'Exposes a URL that specifies the behaviour of this scalar.',
'locations' => [
DirectiveLocation::SCALAR,
],
'args' => [
self::URL_ARGUMENT_NAME => [
'type' => Type::nonNull(Type::string()),
'description' => 'The URL that specifies the behaviour of this scalar and points to a human-readable specification of the data format, serialization, and coercion rules. It must not appear on built-in scalar types.',
],
],
]),
];
}

Expand All @@ -157,6 +172,14 @@ public static function deprecatedDirective(): Directive
return $internal['deprecated'];
}

/** @throws InvariantViolation */
public static function specifiedByDirective(): Directive
{
$internal = self::getInternalDirectives();

return $internal['specifiedBy'];
}

/** @throws InvariantViolation */
public static function isSpecifiedDirective(Directive $directive): bool
{
Expand Down
4 changes: 4 additions & 0 deletions src/Type/Definition/ScalarType.php
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
* @phpstan-type ScalarConfig array{
* name?: string|null,
* description?: string|null,
* specifiedByURL?: string|null,
* astNode?: ScalarTypeDefinitionNode|null,
* extensionASTNodes?: array<ScalarTypeExtensionNode>|null
* }
Expand All @@ -38,6 +39,8 @@ abstract class ScalarType extends Type implements OutputType, InputType, LeafTyp

public ?ScalarTypeDefinitionNode $astNode;

public ?string $specifiedByURL;

/** @var array<ScalarTypeExtensionNode> */
public array $extensionASTNodes;

Expand All @@ -55,6 +58,7 @@ public function __construct(array $config = [])
$this->description = $config['description'] ?? $this->description ?? null;
$this->astNode = $config['astNode'] ?? null;
$this->extensionASTNodes = $config['extensionASTNodes'] ?? [];
$this->specifiedByURL = $config['specifiedByURL'] ?? null;

$this->config = $config;
}
Expand Down
26 changes: 24 additions & 2 deletions src/Utils/ASTDefinitionBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -392,7 +392,6 @@ public function buildField(FieldDefinitionNode $field): array
* @param EnumValueDefinitionNode|FieldDefinitionNode|InputValueDefinitionNode $node
*
* @throws \Exception
* @throws \ReflectionException
* @throws InvariantViolation
*/
private function getDeprecationReason(Node $node): ?string
Expand All @@ -405,6 +404,25 @@ private function getDeprecationReason(Node $node): ?string
return $deprecated['reason'] ?? null;
}

/**
* Given a collection of directives, returns the string value for the
* specifiedBy url.
Comment on lines +408 to +409
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
* Given a collection of directives, returns the string value for the
* specifiedBy url.
* Given a collection of directives, returns the string value for the specifiedBy URL.

*
* @param ScalarTypeDefinitionNode $node
*
* @throws \Exception
* @throws InvariantViolation
*/
private function getSpecifiedByURL(Node $node): ?string
{
$specifiedBy = Values::getDirectiveValues(
Directive::specifiedByDirective(),
$node
);

return $specifiedBy['url'] ?? null;
}

/**
* @param array<ObjectTypeDefinitionNode|ObjectTypeExtensionNode|InterfaceTypeDefinitionNode|InterfaceTypeExtensionNode> $nodes
*
Expand Down Expand Up @@ -509,7 +527,10 @@ private function makeUnionDef(UnionTypeDefinitionNode $def): UnionType
]);
}

/** @throws InvariantViolation */
/**
* @throws \Exception
* @throws InvariantViolation
*/
private function makeScalarDef(ScalarTypeDefinitionNode $def): CustomScalarType
{
$name = $def->name->value;
Expand All @@ -522,6 +543,7 @@ private function makeScalarDef(ScalarTypeDefinitionNode $def): CustomScalarType
'astNode' => $def,
'extensionASTNodes' => $extensionASTNodes,
'serialize' => static fn ($value) => $value,
'specifiedByURL' => $this->getSpecifiedByURL($def),
]);
}

Expand Down
3 changes: 3 additions & 0 deletions src/Utils/BuildSchema.php
Original file line number Diff line number Diff line change
Expand Up @@ -223,6 +223,9 @@ static function (string $typeName): Type {
if (! isset($directivesByName['deprecated'])) {
$directives[] = Directive::deprecatedDirective();
}
if (! isset($directivesByName['specifiedBy'])) {
$directives[] = Directive::specifiedByDirective();
}

// Note: While this could make early assertions to get the correctly
// typed values below, that would throw immediately while type system
Expand Down
1 change: 1 addition & 0 deletions src/Utils/SchemaExtender.php
Original file line number Diff line number Diff line change
Expand Up @@ -228,6 +228,7 @@ protected function extendScalarType(ScalarType $type): CustomScalarType
'serialize' => [$type, 'serialize'],
'parseValue' => [$type, 'parseValue'],
'parseLiteral' => [$type, 'parseLiteral'],
'specifiedByURL' => $type->specifiedByURL,
'astNode' => $type->astNode,
'extensionASTNodes' => $extensionASTNodes,
]);
Expand Down
25 changes: 24 additions & 1 deletion src/Utils/SchemaPrinter.php
Original file line number Diff line number Diff line change
Expand Up @@ -361,11 +361,14 @@ protected static function printInputValue($arg): string
* @phpstan-param Options $options
*
* @throws \JsonException
* @throws InvariantViolation
* @throws SerializationError
*/
protected static function printScalar(ScalarType $type, array $options): string
{
return static::printDescription($options, $type)
. "scalar {$type->name}";
. "scalar {$type->name}"
. static::printSpecifiedBy($type);
}

/**
Expand Down Expand Up @@ -452,6 +455,26 @@ protected static function printDeprecated($deprecation): string
return " @deprecated(reason: {$reasonASTString})";
}

/**
* @throws \JsonException
* @throws InvariantViolation
* @throws SerializationError
*/
protected static function printSpecifiedBy(ScalarType $type): string
{
$url = $type->specifiedByURL;
if ($url === null) {
return '';
}

$urlAST = AST::astFromValue($url, Type::string());
assert($urlAST instanceof StringValueNode);

$urlASTString = Printer::doPrint($urlAST);

return " @specifiedBy(url: {$urlASTString})";
}

protected static function printImplementedInterfaces(ImplementingType $type): string
{
$interfaces = $type->getInterfaces();
Expand Down
4 changes: 4 additions & 0 deletions src/Validator/Rules/QueryComplexity.php
Original file line number Diff line number Diff line change
Expand Up @@ -173,6 +173,10 @@ protected function directiveExcludesField(FieldNode $node): bool
return false;
}

if ($directiveNode->name->value === Directive::SPECIFIED_BY_NAME) {
return false;
}

[$errors, $variableValues] = Values::getVariableValues(
$this->context->getSchema(),
$this->variableDefs,
Expand Down
24 changes: 24 additions & 0 deletions tests/Type/IntrospectionTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -962,6 +962,30 @@ public function testExecutesAnIntrospectionQuery(): void
3 => 'INPUT_FIELD_DEFINITION',
],
],
[
'name' => 'specifiedBy',
'args' => [
0 => [
'name' => 'url',
'type' => [
'kind' => 'NON_NULL',
'name' => null,
'ofType' => [
'kind' => 'SCALAR',
'name' => 'String',
'ofType' => null,
],
],
'defaultValue' => null,
'isDeprecated' => false,
'deprecationReason' => null,
],
],
'isRepeatable' => false,
'locations' => [
0 => 'SCALAR',
],
],
],
],
],
Expand Down
2 changes: 1 addition & 1 deletion tests/Utils/BreakingChangesFinderTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -1329,7 +1329,7 @@ public function testShouldDetectIfADirectiveWasImplicitlyRemoved(): void
$oldSchema = new Schema([]);

$newSchema = new Schema([
'directives' => [Directive::skipDirective(), Directive::includeDirective()],
'directives' => [Directive::skipDirective(), Directive::includeDirective(), Directive::specifiedByDirective()],
]);

$deprecatedDirective = Directive::deprecatedDirective();
Expand Down
19 changes: 6 additions & 13 deletions tests/Utils/BuildSchemaTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -269,13 +269,10 @@ public function testMaintainsIncludeSkipAndSpecifiedBy(): void
{
$schema = BuildSchema::buildAST(Parser::parse('type Query'));

// TODO switch to 4 when adding @specifiedBy - see https://github.com/webonyx/graphql-php/issues/1140
self::assertCount(3, $schema->getDirectives());
self::assertCount(4, $schema->getDirectives());
self::assertSame(Directive::skipDirective(), $schema->getDirective('skip'));
self::assertSame(Directive::includeDirective(), $schema->getDirective('include'));
self::assertSame(Directive::deprecatedDirective(), $schema->getDirective('deprecated'));

self::markTestIncomplete('See https://github.com/webonyx/graphql-php/issues/1140');
self::assertSame(Directive::specifiedByDirective(), $schema->getDirective('specifiedBy'));
}

Expand All @@ -286,15 +283,13 @@ public function testOverridingDirectivesExcludesSpecified(): void
directive @skip on FIELD
directive @include on FIELD
directive @deprecated on FIELD_DEFINITION
directive @specifiedBy on FIELD_DEFINITION
directive @specifiedBy on SCALAR
'));

self::assertCount(4, $schema->getDirectives());
self::assertNotEquals(Directive::skipDirective(), $schema->getDirective('skip'));
self::assertNotEquals(Directive::includeDirective(), $schema->getDirective('include'));
self::assertNotEquals(Directive::deprecatedDirective(), $schema->getDirective('deprecated'));

self::markTestIncomplete('See https://github.com/webonyx/graphql-php/issues/1140');
self::assertNotEquals(Directive::specifiedByDirective(), $schema->getDirective('specifiedBy'));
}

Expand All @@ -307,14 +302,11 @@ public function testAddingDirectivesMaintainsIncludeSkipAndSpecifiedBy(): void
GRAPHQL;
$schema = BuildSchema::buildAST(Parser::parse($sdl));

// TODO switch to 5 when adding @specifiedBy - see https://github.com/webonyx/graphql-php/issues/1140
self::assertCount(4, $schema->getDirectives());
self::assertCount(5, $schema->getDirectives());
self::assertNotNull($schema->getDirective('foo'));
self::assertNotNull($schema->getDirective('skip'));
self::assertNotNull($schema->getDirective('include'));
self::assertNotNull($schema->getDirective('deprecated'));

self::markTestIncomplete('See https://github.com/webonyx/graphql-php/issues/1140');
self::assertNotNull($schema->getDirective('specifiedBy'));
}

Expand Down Expand Up @@ -815,7 +807,6 @@ enum: MyEnum
/** @see it('Supports @specifiedBy') */
public function testSupportsSpecifiedBy(): void
{
self::markTestSkipped('See https://github.com/webonyx/graphql-php/issues/1140');
$sdl = <<<GRAPHQL
scalar Foo @specifiedBy(url: "https://example.com/foo_spec")

Expand All @@ -828,8 +819,10 @@ public function testSupportsSpecifiedBy(): void
self::assertCycle($sdl);

$schema = BuildSchema::build($sdl);
$type = $schema->getType('Foo');

self::assertSame('https://example.com/foo_spec', $schema->getType('Foo')->specifiedByURL);
self::assertInstanceOf(ScalarType::class, $type);
self::assertSame('https://example.com/foo_spec', $type->specifiedByURL);
}

/** @see it('Correctly extend scalar type') */
Expand Down
2 changes: 2 additions & 0 deletions tests/Utils/SchemaExtenderTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -461,8 +461,10 @@ public function testExtendsScalarsByAddingSpecifiedByDirective(): void

extend scalar Foo @specifiedBy(url: "https://example.com/foo_spec")
GRAPHQL;

$extendedSchema = SchemaExtender::extend($schema, Parser::parse($extensionSDL));
$foo = $extendedSchema->getType('Foo');
assert($foo instanceof ScalarType);

self::assertSame('https://example.com/foo_spec', $foo->specifiedByURL);
self::assertEmpty($extendedSchema->validate());
Expand Down
6 changes: 6 additions & 0 deletions tests/Utils/SchemaPrinterTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -1010,6 +1010,12 @@ public function testPrintIntrospectionSchema(): void
"Explains why this element was deprecated, usually also including a suggestion for how to access supported similar data. Formatted using the Markdown syntax, as specified by [CommonMark](https:\/\/commonmark.org\/)."
reason: String = "No longer supported"
) on FIELD_DEFINITION | ENUM_VALUE | ARGUMENT_DEFINITION | INPUT_FIELD_DEFINITION

"Exposes a URL that specifies the behaviour of this scalar."
directive @specifiedBy(
"The URL that specifies the behaviour of this scalar and points to a human-readable specification of the data format, serialization, and coercion rules. It must not appear on built-in scalar types."
url: String!
) on SCALAR

"A GraphQL Schema defines the capabilities of a GraphQL server. It exposes all available types and directives on the server, as well as the entry points for query, mutation, and subscription operations."
type __Schema {
Expand Down
2 changes: 1 addition & 1 deletion tests/Validator/KnownDirectivesTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -318,7 +318,7 @@ public function testWSLWithWellPlacedDirectives(): void

extend type MyObj @onObject

scalar MyScalar @onScalar
scalar MyScalar @onScalar
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
scalar MyScalar @onScalar
scalar MyScalar @onScalar


extend scalar MyScalar @onScalar

Expand Down
7 changes: 2 additions & 5 deletions tests/Validator/ValidatorTestCase.php
Original file line number Diff line number Diff line change
Expand Up @@ -367,10 +367,7 @@ public static function getTestSchema(): Schema
return new Schema([
'query' => $queryRoot,
'subscription' => $subscriptionRoot,
'directives' => [
Directive::includeDirective(),
Directive::skipDirective(),
Directive::deprecatedDirective(),
'directives' => array_merge(Directive::getInternalDirectives(), [
new Directive([
'name' => 'directive',
'locations' => [DirectiveLocation::FIELD, DirectiveLocation::FRAGMENT_DEFINITION],
Expand Down Expand Up @@ -420,7 +417,7 @@ public static function getTestSchema(): Schema
'name' => 'onVariableDefinition',
'locations' => [DirectiveLocation::VARIABLE_DEFINITION],
]),
],
]),
]);
}

Expand Down