-
-
Notifications
You must be signed in to change notification settings - Fork 574
Expand file tree
/
Copy pathCustomScalarType.php
More file actions
125 lines (107 loc) · 4.35 KB
/
CustomScalarType.php
File metadata and controls
125 lines (107 loc) · 4.35 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
<?php declare(strict_types=1);
namespace GraphQL\Type\Definition;
use GraphQL\Error\Error;
use GraphQL\Error\InvariantViolation;
use GraphQL\Language\AST\DirectiveNode;
use GraphQL\Language\AST\Node;
use GraphQL\Language\AST\ScalarTypeDefinitionNode;
use GraphQL\Language\AST\ScalarTypeExtensionNode;
use GraphQL\Language\AST\ValueNode;
use GraphQL\Utils\AST;
use GraphQL\Utils\Utils;
/**
* @phpstan-type InputCustomScalarConfig array{
* name?: string|null,
* description?: string|null,
* serialize?: callable(mixed): mixed,
* parseValue: callable(mixed): mixed,
* parseLiteral: callable(ValueNode&Node, array<string, mixed>|null): mixed,
* directives?: array<DirectiveNode>|null,
* astNode?: ScalarTypeDefinitionNode|null,
* extensionASTNodes?: array<ScalarTypeExtensionNode>|null
* }
* @phpstan-type OutputCustomScalarConfig array{
* name?: string|null,
* description?: string|null,
* serialize: callable(mixed): mixed,
* parseValue?: callable(mixed): mixed,
* parseLiteral?: callable(ValueNode&Node, array<string, mixed>|null): mixed,
* directives?: array<DirectiveNode>|null,
* astNode?: ScalarTypeDefinitionNode|null,
* extensionASTNodes?: array<ScalarTypeExtensionNode>|null
* }
* @phpstan-type CustomScalarConfig InputCustomScalarConfig|OutputCustomScalarConfig
*/
class CustomScalarType extends ScalarType
{
/** @phpstan-var CustomScalarConfig */
// @phpstan-ignore-next-line specialize type
public array $config;
/**
* @param array<string, mixed> $config
*
* @phpstan-param CustomScalarConfig $config
*/
public function __construct(array $config)
{
parent::__construct($config);
}
public function serialize($value)
{
if (isset($this->config['serialize'])) {
return $this->config['serialize']($value);
}
return $value;
}
public function parseValue($value)
{
if (isset($this->config['parseValue'])) {
return $this->config['parseValue']($value);
}
return $value;
}
/** @throws \Exception */
public function parseLiteral(Node $valueNode, ?array $variables = null)
{
if (isset($this->config['parseLiteral'])) {
return $this->config['parseLiteral']($valueNode, $variables);
}
return AST::valueFromASTUntyped($valueNode, $variables);
}
/**
* @throws Error
* @throws InvariantViolation
*/
public function assertValid(): void
{
parent::assertValid();
$serialize = $this->config['serialize'] ?? null;
$parseValue = $this->config['parseValue'] ?? null;
$parseLiteral = $this->config['parseLiteral'] ?? null;
$hasSerialize = $serialize !== null;
$hasParseValue = $parseValue !== null;
$hasParseLiteral = $parseLiteral !== null;
$hasParse = $hasParseValue && $hasParseLiteral;
if ($hasParseValue !== $hasParseLiteral) {
throw new InvariantViolation("{$this->name} must provide both \"parseValue\" and \"parseLiteral\" functions to work as an input type.");
}
if (! $hasSerialize && ! $hasParse) {
throw new InvariantViolation("{$this->name} must provide \"parseValue\" and \"parseLiteral\" functions, \"serialize\" function, or both.");
}
// @phpstan-ignore-next-line unnecessary according to types, but can happen during runtime
if ($hasSerialize && ! is_callable($serialize)) {
$notCallable = Utils::printSafe($serialize);
throw new InvariantViolation("{$this->name} must provide \"serialize\" as a callable if given, but got: {$notCallable}.");
}
// @phpstan-ignore-next-line unnecessary according to types, but can happen during runtime
if ($hasParseValue && ! is_callable($parseValue)) {
$notCallable = Utils::printSafe($parseValue);
throw new InvariantViolation("{$this->name} must provide \"parseValue\" as a callable if given, but got: {$notCallable}.");
}
// @phpstan-ignore-next-line unnecessary according to types, but can happen during runtime
if ($hasParseLiteral && ! is_callable($parseLiteral)) {
$notCallable = Utils::printSafe($parseLiteral);
throw new InvariantViolation("{$this->name} must provide \"parseLiteral\" as a callable if given, but got: {$notCallable}.");
}
}
}