From 313e2dcb8bdcc20ae23414e9b48c6b97ca28ec51 Mon Sep 17 00:00:00 2001 From: Ruud Kamphuis Date: Mon, 10 Jun 2024 10:58:01 +0200 Subject: [PATCH 1/3] Add unaliased path to resolve info Fixes #1345 --- docs/class-reference.md | 28 ++++++++- src/Error/Error.php | 37 +++++++++-- src/Executor/ReferenceExecutor.php | 95 ++++++++++++++++++++++------- src/Type/Definition/ResolveInfo.php | 20 +++++- tests/Type/ResolveInfoTest.php | 73 ++++++++++++++++++++++ 5 files changed, 222 insertions(+), 31 deletions(-) diff --git a/docs/class-reference.md b/docs/class-reference.md index 5f0f11c4c..97134cb97 100644 --- a/docs/class-reference.md +++ b/docs/class-reference.md @@ -309,6 +309,7 @@ Passed as 4th argument to every field resolver. See [docs on field resolving (da @phpstan-import-type QueryPlanOptions from QueryPlan @phpstan-type Path array +@phpstan-type UnaliasedPath list ### GraphQL\Type\Definition\ResolveInfo Props @@ -351,7 +352,7 @@ public $fieldNodes; public $parentType; /** - * Path to this field from the very root value. + * Path to this field from the very root value. When fields are aliased, the path includes aliases. * * @api * @@ -361,6 +362,17 @@ public $parentType; */ public $path; +/** + * Path to this field from the very root value. This will never include aliases. + * + * @api + * + * @var list + * + * @phpstan-var UnaliasedPath + */ +public $unaliasedPath; + /** * Instance of a schema used for execution. * @@ -1730,7 +1742,7 @@ function getLocations(): array ```php /** * Returns an array describing the path from the root value to the field which produced this error. - * Only included for execution errors. + * Only included for execution errors. When fields are aliased, the path includes aliases. * * @return array|null * @@ -1739,6 +1751,18 @@ function getLocations(): array function getPath(): ?array ``` +```php +/** + * Returns an array describing the path from the root value to the field which produced this error. + * Only included for execution errors. This will never include aliases. + * + * @return list|null + * + * @api + */ +function getUnaliasedPath(): ?array +``` + ## GraphQL\Error\Warning Encapsulates warnings produced by the library. diff --git a/src/Error/Error.php b/src/Error/Error.php index f30bf4fc9..20e82093d 100644 --- a/src/Error/Error.php +++ b/src/Error/Error.php @@ -34,11 +34,21 @@ class Error extends \Exception implements \JsonSerializable, ClientAware, Provid /** * An array describing the JSON-path into the execution response which * corresponds to this error. Only included for errors during execution. + * When fields are aliased, the path includes aliases. * * @var array|null */ public ?array $path; + /** + * An array describing the JSON-path into the execution response which + * corresponds to this error. Only included for errors during execution. + * This will never include aliases. + * + * @var list|null + */ + public ?array $unaliasedPath; + /** * An array of GraphQL AST Nodes corresponding to this error. * @@ -67,6 +77,7 @@ class Error extends \Exception implements \JsonSerializable, ClientAware, Provid * @param array|null $positions * @param array|null $path * @param array|null $extensions + * @param list|null $unaliasedPath */ public function __construct( string $message = '', @@ -75,7 +86,8 @@ public function __construct( ?array $positions = null, ?array $path = null, ?\Throwable $previous = null, - ?array $extensions = null + ?array $extensions = null, + ?array $unaliasedPath = null ) { parent::__construct($message, 0, $previous); @@ -93,6 +105,7 @@ public function __construct( $this->source = $source; $this->positions = $positions; $this->path = $path; + $this->unaliasedPath = $unaliasedPath; if (\is_array($extensions) && $extensions !== []) { $this->extensions = $extensions; @@ -115,8 +128,9 @@ public function __construct( * @param mixed $error * @param iterable|Node|null $nodes * @param array|null $path + * @param list|null $unaliasedPath */ - public static function createLocatedError($error, $nodes = null, ?array $path = null): Error + public static function createLocatedError($error, $nodes = null, ?array $path = null, ?array $unaliasedPath = null): Error { if ($error instanceof self) { if ($error->isLocated()) { @@ -125,6 +139,7 @@ public static function createLocatedError($error, $nodes = null, ?array $path = $nodes ??= $error->getNodes(); $path ??= $error->getPath(); + $unaliasedPath ??= $error->getUnaliasedPath(); } $source = null; @@ -159,7 +174,8 @@ public static function createLocatedError($error, $nodes = null, ?array $path = $positions, $path, $originalError, - $extensions + $extensions, + $unaliasedPath ); } @@ -251,7 +267,7 @@ public function getNodes(): ?array /** * Returns an array describing the path from the root value to the field which produced this error. - * Only included for execution errors. + * Only included for execution errors. When fields are aliased, the path includes aliases. * * @return array|null * @@ -262,6 +278,19 @@ public function getPath(): ?array return $this->path; } + /** + * Returns an array describing the path from the root value to the field which produced this error. + * Only included for execution errors. This will never include aliases. + * + * @return list|null + * + * @api + */ + public function getUnaliasedPath(): ?array + { + return $this->unaliasedPath; + } + /** @return array|null */ public function getExtensions(): ?array { diff --git a/src/Executor/ReferenceExecutor.php b/src/Executor/ReferenceExecutor.php index 434047a96..e3344853b 100644 --- a/src/Executor/ReferenceExecutor.php +++ b/src/Executor/ReferenceExecutor.php @@ -37,6 +37,7 @@ /** * @phpstan-import-type FieldResolver from Executor * @phpstan-import-type Path from ResolveInfo + * @phpstan-import-type UnaliasedPath from ResolveInfo * * @phpstan-type Fields \ArrayObject> */ @@ -287,6 +288,7 @@ protected function executeOperation(OperationDefinitionNode $operation, $rootVal $type = $this->getOperationRootType($this->exeContext->schema, $operation); $fields = $this->collectFields($type, $operation->selectionSet, new \ArrayObject(), new \ArrayObject()); $path = []; + $unaliasedPath = []; // Errors from sub-fields of a NonNull type may propagate to the top level, // at which point we still log the error and null the parent field, which // in this case is the entire response. @@ -294,8 +296,8 @@ protected function executeOperation(OperationDefinitionNode $operation, $rootVal // Similar to completeValueCatchingError. try { $result = $operation->operation === 'mutation' - ? $this->executeFieldsSerially($type, $rootValue, $path, $fields, $this->exeContext->contextValue) - : $this->executeFields($type, $rootValue, $path, $fields, $this->exeContext->contextValue); + ? $this->executeFieldsSerially($type, $rootValue, $path, $unaliasedPath, $fields, $this->exeContext->contextValue) + : $this->executeFields($type, $rootValue, $path, $unaliasedPath, $fields, $this->exeContext->contextValue); $promise = $this->getPromise($result); if ($promise !== null) { @@ -522,23 +524,31 @@ protected function doesFragmentConditionMatch(Node $fragment, ObjectType $type): * * @param mixed $rootValue * @param array $path + * @param list $unaliasedPath * @param mixed $contextValue * * @phpstan-param Fields $fields * * @return array|Promise|\stdClass */ - protected function executeFieldsSerially(ObjectType $parentType, $rootValue, array $path, \ArrayObject $fields, $contextValue) + protected function executeFieldsSerially(ObjectType $parentType, $rootValue, array $path, array $unaliasedPath, \ArrayObject $fields, $contextValue) { $result = $this->promiseReduce( \array_keys($fields->getArrayCopy()), - function ($results, $responseName) use ($contextValue, $path, $parentType, $rootValue, $fields) { + function ($results, $responseName) use ($contextValue, $path, $unaliasedPath, $parentType, $rootValue, $fields) { $fieldNodes = $fields[$responseName]; assert($fieldNodes instanceof \ArrayObject, 'The keys of $fields populate $responseName'); $fieldPath = $path; $fieldPath[] = $responseName; - $result = $this->resolveField($parentType, $rootValue, $fieldNodes, $fieldPath, $this->maybeScopeContext($contextValue)); + + $fieldNode = $fieldNodes[0]; + assert($fieldNode instanceof FieldNode, '$fieldNodes is non-empty'); + + $fieldUnaliasedPath = $unaliasedPath; + $fieldUnaliasedPath[] = $fieldNode->name->value; + + $result = $this->resolveField($parentType, $rootValue, $fieldNodes, $fieldPath, $fieldUnaliasedPath, $this->maybeScopeContext($contextValue)); if ($result === static::$UNDEFINED) { return $results; } @@ -578,9 +588,11 @@ function ($results, $responseName) use ($contextValue, $path, $parentType, $root * * @param mixed $rootValue * @param array $path + * @param list $unaliasedPath * @param mixed $contextValue * * @phpstan-param Path $path + * @phpstan-param UnaliasedPath $unaliasedPath * * @param \ArrayObject $fieldNodes * @@ -589,7 +601,7 @@ function ($results, $responseName) use ($contextValue, $path, $parentType, $root * * @return array|\Throwable|mixed|null */ - protected function resolveField(ObjectType $parentType, $rootValue, \ArrayObject $fieldNodes, array $path, $contextValue) + protected function resolveField(ObjectType $parentType, $rootValue, \ArrayObject $fieldNodes, array $path, array $unaliasedPath, $contextValue) { $exeContext = $this->exeContext; $fieldNode = $fieldNodes[0]; @@ -616,7 +628,8 @@ protected function resolveField(ObjectType $parentType, $rootValue, \ArrayObject $exeContext->fragments, $exeContext->rootValue, $exeContext->operation, - $exeContext->variableValues + $exeContext->variableValues, + $unaliasedPath ); if ($fieldDef->resolveFn !== null) { $resolveFn = $fieldDef->resolveFn; @@ -642,6 +655,7 @@ protected function resolveField(ObjectType $parentType, $rootValue, \ArrayObject $fieldNodes, $info, $path, + $unaliasedPath, $result, $contextValue ); @@ -723,9 +737,11 @@ protected function resolveFieldValueOrError( * * @param \ArrayObject $fieldNodes * @param array $path + * @param list $unaliasedPath * @param mixed $contextValue * * @phpstan-param Path $path + * @phpstan-param UnaliasedPath $unaliasedPath * * @param mixed $result * @@ -738,6 +754,7 @@ protected function completeValueCatchingError( \ArrayObject $fieldNodes, ResolveInfo $info, array $path, + array $unaliasedPath, $result, $contextValue ) { @@ -746,23 +763,23 @@ protected function completeValueCatchingError( try { $promise = $this->getPromise($result); if ($promise !== null) { - $completed = $promise->then(function (&$resolved) use ($contextValue, $returnType, $fieldNodes, $info, $path) { - return $this->completeValue($returnType, $fieldNodes, $info, $path, $resolved, $contextValue); + $completed = $promise->then(function (&$resolved) use ($contextValue, $returnType, $fieldNodes, $info, $path, $unaliasedPath) { + return $this->completeValue($returnType, $fieldNodes, $info, $path, $unaliasedPath, $resolved, $contextValue); }); } else { - $completed = $this->completeValue($returnType, $fieldNodes, $info, $path, $result, $contextValue); + $completed = $this->completeValue($returnType, $fieldNodes, $info, $path, $unaliasedPath, $result, $contextValue); } $promise = $this->getPromise($completed); if ($promise !== null) { - return $promise->then(null, function ($error) use ($fieldNodes, $path, $returnType): void { - $this->handleFieldError($error, $fieldNodes, $path, $returnType); + return $promise->then(null, function ($error) use ($fieldNodes, $path, $unaliasedPath, $returnType): void { + $this->handleFieldError($error, $fieldNodes, $path, $unaliasedPath, $returnType); }); } return $completed; } catch (\Throwable $err) { - $this->handleFieldError($err, $fieldNodes, $path, $returnType); + $this->handleFieldError($err, $fieldNodes, $path, $unaliasedPath, $returnType); return null; } @@ -772,15 +789,17 @@ protected function completeValueCatchingError( * @param mixed $rawError * @param \ArrayObject $fieldNodes * @param array $path + * @param list $unaliasedPath * * @throws Error */ - protected function handleFieldError($rawError, \ArrayObject $fieldNodes, array $path, Type $returnType): void + protected function handleFieldError($rawError, \ArrayObject $fieldNodes, array $path, array $unaliasedPath, Type $returnType): void { $error = Error::createLocatedError( $rawError, $fieldNodes, - $path + $path, + $unaliasedPath ); // If the field type is non-nullable, then it is resolved without any @@ -817,6 +836,7 @@ protected function handleFieldError($rawError, \ArrayObject $fieldNodes, array $ * * @param \ArrayObject $fieldNodes * @param array $path + * @param list $unaliasedPath * @param mixed $result * @param mixed $contextValue * @@ -830,6 +850,7 @@ protected function completeValue( \ArrayObject $fieldNodes, ResolveInfo $info, array $path, + array $unaliasedPath, &$result, $contextValue ) { @@ -846,6 +867,7 @@ protected function completeValue( $fieldNodes, $info, $path, + $unaliasedPath, $result, $contextValue ); @@ -868,7 +890,7 @@ protected function completeValue( throw new InvariantViolation("Expected field {$info->parentType}.{$info->fieldName} to return iterable, but got: {$resultType}."); } - return $this->completeListValue($returnType, $fieldNodes, $info, $path, $result, $contextValue); + return $this->completeListValue($returnType, $fieldNodes, $info, $path, $unaliasedPath, $result, $contextValue); } assert($returnType instanceof NamedType, 'Wrapping types should return early'); @@ -885,12 +907,12 @@ protected function completeValue( } if ($returnType instanceof AbstractType) { - return $this->completeAbstractValue($returnType, $fieldNodes, $info, $path, $result, $contextValue); + return $this->completeAbstractValue($returnType, $fieldNodes, $info, $path, $unaliasedPath, $result, $contextValue); } // Field type must be and Object, Interface or Union and expect sub-selections. if ($returnType instanceof ObjectType) { - return $this->completeObjectValue($returnType, $fieldNodes, $info, $path, $result, $contextValue); + return $this->completeObjectValue($returnType, $fieldNodes, $info, $path, $unaliasedPath, $result, $contextValue); } $safeReturnType = Utils::printSafe($returnType); @@ -958,6 +980,7 @@ function ($previous, $value) use ($callback) { * @param ListOfType $returnType * @param \ArrayObject $fieldNodes * @param list $path + * @param list $unaliasedPath * @param iterable $results * @param mixed $contextValue * @@ -970,6 +993,7 @@ protected function completeListValue( \ArrayObject $fieldNodes, ResolveInfo $info, array $path, + array $unaliasedPath, iterable &$results, $contextValue ) { @@ -979,10 +1003,13 @@ protected function completeListValue( $containsPromise = false; $completedItems = []; foreach ($results as $item) { - $fieldPath = [...$path, $i++]; + $fieldPath = [...$path, $i]; $info->path = $fieldPath; + $unaliasedPath = [...$unaliasedPath, $i]; + $info->unaliasedPath = $unaliasedPath; + ++$i; - $completedItem = $this->completeValueCatchingError($itemType, $fieldNodes, $info, $fieldPath, $item, $contextValue); + $completedItem = $this->completeValueCatchingError($itemType, $fieldNodes, $info, $fieldPath, $unaliasedPath, $item, $contextValue); if (! $containsPromise && $this->getPromise($completedItem) !== null) { $containsPromise = true; @@ -1027,6 +1054,7 @@ protected function completeLeafValue(LeafType $returnType, &$result) * @param AbstractType&Type $returnType * @param \ArrayObject $fieldNodes * @param array $path + * @param list $unaliasedPath * @param array $result * @param mixed $contextValue * @@ -1041,6 +1069,7 @@ protected function completeAbstractValue( \ArrayObject $fieldNodes, ResolveInfo $info, array $path, + array $unaliasedPath, &$result, $contextValue ) { @@ -1066,6 +1095,7 @@ protected function completeAbstractValue( $fieldNodes, $info, $path, + $unaliasedPath, $result, $contextValue )); @@ -1081,6 +1111,7 @@ protected function completeAbstractValue( $fieldNodes, $info, $path, + $unaliasedPath, $result, $contextValue ); @@ -1157,6 +1188,7 @@ protected function defaultTypeResolver($value, $contextValue, ResolveInfo $info, * * @param \ArrayObject $fieldNodes * @param array $path + * @param list $unaliasedPath * @param mixed $result * @param mixed $contextValue * @@ -1170,6 +1202,7 @@ protected function completeObjectValue( \ArrayObject $fieldNodes, ResolveInfo $info, array $path, + array $unaliasedPath, &$result, $contextValue ) { @@ -1185,6 +1218,7 @@ protected function completeObjectValue( $returnType, $fieldNodes, $path, + $unaliasedPath, &$result ) { if (! $isTypeOfResult) { @@ -1195,6 +1229,7 @@ protected function completeObjectValue( $returnType, $fieldNodes, $path, + $unaliasedPath, $result, $contextValue ); @@ -1211,6 +1246,7 @@ protected function completeObjectValue( $returnType, $fieldNodes, $path, + $unaliasedPath, $result, $contextValue ); @@ -1236,6 +1272,7 @@ protected function invalidReturnTypeError( /** * @param \ArrayObject $fieldNodes * @param array $path + * @param list $unaliasedPath * @param mixed $result * @param mixed $contextValue * @@ -1248,12 +1285,13 @@ protected function collectAndExecuteSubfields( ObjectType $returnType, \ArrayObject $fieldNodes, array $path, + array $unaliasedPath, &$result, $contextValue ) { $subFieldNodes = $this->collectSubFields($returnType, $fieldNodes); - return $this->executeFields($returnType, $result, $path, $subFieldNodes, $contextValue); + return $this->executeFields($returnType, $result, $path, $unaliasedPath, $subFieldNodes, $contextValue); } /** @@ -1298,6 +1336,7 @@ protected function collectSubFields(ObjectType $returnType, \ArrayObject $fieldN * * @param mixed $rootValue * @param array $path + * @param list $unaliasedPath * @param mixed $contextValue * * @phpstan-param Fields $fields @@ -1307,14 +1346,24 @@ protected function collectSubFields(ObjectType $returnType, \ArrayObject $fieldN * * @return Promise|\stdClass|array */ - protected function executeFields(ObjectType $parentType, $rootValue, array $path, \ArrayObject $fields, $contextValue) + protected function executeFields(ObjectType $parentType, $rootValue, array $path, array $unaliasedPath, \ArrayObject $fields, $contextValue) { $containsPromise = false; $results = []; foreach ($fields as $responseName => $fieldNodes) { + $fieldNodes = $fields[$responseName]; + assert($fieldNodes instanceof \ArrayObject, 'The keys of $fields populate $responseName'); + $fieldPath = $path; $fieldPath[] = $responseName; - $result = $this->resolveField($parentType, $rootValue, $fieldNodes, $fieldPath, $this->maybeScopeContext($contextValue)); + + $fieldNode = $fieldNodes[0]; + assert($fieldNode instanceof FieldNode, '$fieldNodes is non-empty'); + + $fieldUnaliasedPath = $unaliasedPath; + $fieldUnaliasedPath[] = $fieldNode->name->value; + + $result = $this->resolveField($parentType, $rootValue, $fieldNodes, $fieldPath, $fieldUnaliasedPath, $this->maybeScopeContext($contextValue)); if ($result === static::$UNDEFINED) { continue; } diff --git a/src/Type/Definition/ResolveInfo.php b/src/Type/Definition/ResolveInfo.php index 1d3ea8a00..0e4ed07bb 100644 --- a/src/Type/Definition/ResolveInfo.php +++ b/src/Type/Definition/ResolveInfo.php @@ -20,6 +20,7 @@ * @phpstan-import-type QueryPlanOptions from QueryPlan * * @phpstan-type Path array + * @phpstan-type UnaliasedPath list */ class ResolveInfo { @@ -61,7 +62,7 @@ class ResolveInfo public ObjectType $parentType; /** - * Path to this field from the very root value. + * Path to this field from the very root value. When fields are aliased, the path includes aliases. * * @api * @@ -71,6 +72,17 @@ class ResolveInfo */ public array $path; + /** + * Path to this field from the very root value. This will never include aliases. + * + * @api + * + * @var list + * + * @phpstan-var UnaliasedPath + */ + public array $unaliasedPath; + /** * Instance of a schema used for execution. * @@ -115,8 +127,10 @@ class ResolveInfo /** * @param \ArrayObject $fieldNodes * @param array $path + * @param array $unaliasedPath * * @phpstan-param Path $path + * @phpstan-param UnaliasedPath $unaliasedPath * * @param array $fragments * @param mixed|null $rootValue @@ -131,7 +145,8 @@ public function __construct( array $fragments, $rootValue, OperationDefinitionNode $operation, - array $variableValues + array $variableValues, + array $unaliasedPath = [] ) { $this->fieldDefinition = $fieldDefinition; $this->fieldName = $fieldDefinition->name; @@ -139,6 +154,7 @@ public function __construct( $this->fieldNodes = $fieldNodes; $this->parentType = $parentType; $this->path = $path; + $this->unaliasedPath = $unaliasedPath; $this->schema = $schema; $this->fragments = $fragments; $this->rootValue = $rootValue; diff --git a/tests/Type/ResolveInfoTest.php b/tests/Type/ResolveInfoTest.php index e640354c4..a62033bfc 100644 --- a/tests/Type/ResolveInfoTest.php +++ b/tests/Type/ResolveInfoTest.php @@ -445,4 +445,77 @@ public function testDeepFieldSelectionOnDuplicatedFields(): void self::assertEquals(['data' => ['level1' => null]], $result); self::assertEquals($expectedDeepSelection, $actualDeepSelection); } + + public function testPathAndUnaliasedPath(): void + { + $level2 = new ObjectType([ + 'name' => 'level2', + 'fields' => [ + 'scalar1' => [ + 'type' => Type::string(), + 'resolve' => static function ($value, array $args, $context, ResolveInfo $info) { + return 'path: ' . implode('.', $info->path) . ', unaliasedPath: ' . implode('.', $info->unaliasedPath); + }, + ], + 'scalar2' => [ + 'type' => Type::string(), + 'resolve' => static function ($value, array $args, $context, ResolveInfo $info) { + return 'path: ' . implode('.', $info->path) . ', unaliasedPath: ' . implode('.', $info->unaliasedPath); + }, + ], + ], + ]); + $level1 = new ObjectType([ + 'name' => 'level1', + 'fields' => [ + 'level2' => [ + 'type' => $level2, + 'resolve' => function () { + return true; + }, + ], + ], + ]); + + $query = new ObjectType([ + 'name' => 'Query', + 'fields' => [ + 'level1' => [ + 'type' => $level1, + 'resolve' => function () { + return true; + }, + ], + ], + ]); + + $result = GraphQL::executeQuery( + new Schema(['query' => $query]), + <<toArray(); + + self::assertEquals([ + 'data' => [ + 'level1' => [ + 'level2' => [ + 'scalar1' => 'path: level1.level2.scalar1, unaliasedPath: level1.level2.scalar1', + ], + 'level1000' => [ + 'scalar2' => 'path: level1.level1000.scalar2, unaliasedPath: level1.level2.scalar2', + ], + ], + ], + ], $result); + } } From 8a4113b269a426c79d858f3f40616f5234dafb28 Mon Sep 17 00:00:00 2001 From: Ruud Kamphuis Date: Mon, 10 Jun 2024 11:35:38 +0200 Subject: [PATCH 2/3] Change Path from array to list --- docs/class-reference.md | 3 +-- src/Error/Error.php | 12 ++++++------ src/Executor/ReferenceExecutor.php | 23 +++++++++++------------ src/Type/Definition/ResolveInfo.php | 13 ++++++------- 4 files changed, 24 insertions(+), 27 deletions(-) diff --git a/docs/class-reference.md b/docs/class-reference.md index 97134cb97..8c4e317f6 100644 --- a/docs/class-reference.md +++ b/docs/class-reference.md @@ -309,7 +309,6 @@ Passed as 4th argument to every field resolver. See [docs on field resolving (da @phpstan-import-type QueryPlanOptions from QueryPlan @phpstan-type Path array -@phpstan-type UnaliasedPath list ### GraphQL\Type\Definition\ResolveInfo Props @@ -369,7 +368,7 @@ public $path; * * @var list * - * @phpstan-var UnaliasedPath + * @phpstan-var Path */ public $unaliasedPath; diff --git a/src/Error/Error.php b/src/Error/Error.php index 20e82093d..d644ed103 100644 --- a/src/Error/Error.php +++ b/src/Error/Error.php @@ -36,7 +36,7 @@ class Error extends \Exception implements \JsonSerializable, ClientAware, Provid * corresponds to this error. Only included for errors during execution. * When fields are aliased, the path includes aliases. * - * @var array|null + * @var list|null */ public ?array $path; @@ -75,9 +75,9 @@ class Error extends \Exception implements \JsonSerializable, ClientAware, Provid /** * @param iterable|Node|null $nodes * @param array|null $positions - * @param array|null $path + * @param list|null $path * @param array|null $extensions - * @param list|null $unaliasedPath + * @param list|null $unaliasedPath */ public function __construct( string $message = '', @@ -127,8 +127,8 @@ public function __construct( * * @param mixed $error * @param iterable|Node|null $nodes - * @param array|null $path - * @param list|null $unaliasedPath + * @param list|null $path + * @param list|null $unaliasedPath */ public static function createLocatedError($error, $nodes = null, ?array $path = null, ?array $unaliasedPath = null): Error { @@ -269,7 +269,7 @@ public function getNodes(): ?array * Returns an array describing the path from the root value to the field which produced this error. * Only included for execution errors. When fields are aliased, the path includes aliases. * - * @return array|null + * @return list|null * * @api */ diff --git a/src/Executor/ReferenceExecutor.php b/src/Executor/ReferenceExecutor.php index e3344853b..baa0dd295 100644 --- a/src/Executor/ReferenceExecutor.php +++ b/src/Executor/ReferenceExecutor.php @@ -37,7 +37,6 @@ /** * @phpstan-import-type FieldResolver from Executor * @phpstan-import-type Path from ResolveInfo - * @phpstan-import-type UnaliasedPath from ResolveInfo * * @phpstan-type Fields \ArrayObject> */ @@ -523,7 +522,7 @@ protected function doesFragmentConditionMatch(Node $fragment, ObjectType $type): * Implements the "Evaluating selection sets" section of the spec for "write" mode. * * @param mixed $rootValue - * @param array $path + * @param list $path * @param list $unaliasedPath * @param mixed $contextValue * @@ -587,12 +586,12 @@ function ($results, $responseName) use ($contextValue, $path, $unaliasedPath, $p * serialize scalars, or execute the sub-selection-set for objects. * * @param mixed $rootValue - * @param array $path + * @param list $path * @param list $unaliasedPath * @param mixed $contextValue * * @phpstan-param Path $path - * @phpstan-param UnaliasedPath $unaliasedPath + * @phpstan-param Path $unaliasedPath * * @param \ArrayObject $fieldNodes * @@ -736,12 +735,12 @@ protected function resolveFieldValueOrError( * in the execution context. * * @param \ArrayObject $fieldNodes - * @param array $path + * @param list $path * @param list $unaliasedPath * @param mixed $contextValue * * @phpstan-param Path $path - * @phpstan-param UnaliasedPath $unaliasedPath + * @phpstan-param Path $unaliasedPath * * @param mixed $result * @@ -788,7 +787,7 @@ protected function completeValueCatchingError( /** * @param mixed $rawError * @param \ArrayObject $fieldNodes - * @param array $path + * @param list $path * @param list $unaliasedPath * * @throws Error @@ -835,7 +834,7 @@ protected function handleFieldError($rawError, \ArrayObject $fieldNodes, array $ * value by evaluating all sub-selections. * * @param \ArrayObject $fieldNodes - * @param array $path + * @param list $path * @param list $unaliasedPath * @param mixed $result * @param mixed $contextValue @@ -1053,7 +1052,7 @@ protected function completeLeafValue(LeafType $returnType, &$result) * * @param AbstractType&Type $returnType * @param \ArrayObject $fieldNodes - * @param array $path + * @param list $path * @param list $unaliasedPath * @param array $result * @param mixed $contextValue @@ -1187,7 +1186,7 @@ protected function defaultTypeResolver($value, $contextValue, ResolveInfo $info, * Complete an Object value by executing all sub-selections. * * @param \ArrayObject $fieldNodes - * @param array $path + * @param list $path * @param list $unaliasedPath * @param mixed $result * @param mixed $contextValue @@ -1271,7 +1270,7 @@ protected function invalidReturnTypeError( /** * @param \ArrayObject $fieldNodes - * @param array $path + * @param list $path * @param list $unaliasedPath * @param mixed $result * @param mixed $contextValue @@ -1335,7 +1334,7 @@ protected function collectSubFields(ObjectType $returnType, \ArrayObject $fieldN * Implements the "Evaluating selection sets" section of the spec for "read" mode. * * @param mixed $rootValue - * @param array $path + * @param list $path * @param list $unaliasedPath * @param mixed $contextValue * diff --git a/src/Type/Definition/ResolveInfo.php b/src/Type/Definition/ResolveInfo.php index 0e4ed07bb..dcae8eecf 100644 --- a/src/Type/Definition/ResolveInfo.php +++ b/src/Type/Definition/ResolveInfo.php @@ -19,8 +19,7 @@ * * @phpstan-import-type QueryPlanOptions from QueryPlan * - * @phpstan-type Path array - * @phpstan-type UnaliasedPath list + * @phpstan-type Path list */ class ResolveInfo { @@ -66,7 +65,7 @@ class ResolveInfo * * @api * - * @var array + * @var list * * @phpstan-var Path */ @@ -79,7 +78,7 @@ class ResolveInfo * * @var list * - * @phpstan-var UnaliasedPath + * @phpstan-var Path */ public array $unaliasedPath; @@ -126,11 +125,11 @@ class ResolveInfo /** * @param \ArrayObject $fieldNodes - * @param array $path - * @param array $unaliasedPath + * @param list $path + * @param list $unaliasedPath * * @phpstan-param Path $path - * @phpstan-param UnaliasedPath $unaliasedPath + * @phpstan-param Path $unaliasedPath * * @param array $fragments * @param mixed|null $rootValue From 4362ad2f03899794ea430e0274cf8e0394701070 Mon Sep 17 00:00:00 2001 From: ruudk Date: Mon, 10 Jun 2024 09:37:08 +0000 Subject: [PATCH 3/3] Prettify docs --- docs/class-reference.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/docs/class-reference.md b/docs/class-reference.md index 8c4e317f6..35f8d46a1 100644 --- a/docs/class-reference.md +++ b/docs/class-reference.md @@ -308,7 +308,7 @@ Passed as 4th argument to every field resolver. See [docs on field resolving (da @phpstan-import-type QueryPlanOptions from QueryPlan -@phpstan-type Path array +@phpstan-type Path list ### GraphQL\Type\Definition\ResolveInfo Props @@ -355,7 +355,7 @@ public $parentType; * * @api * - * @var array + * @var list * * @phpstan-var Path */ @@ -1743,7 +1743,7 @@ function getLocations(): array * Returns an array describing the path from the root value to the field which produced this error. * Only included for execution errors. When fields are aliased, the path includes aliases. * - * @return array|null + * @return list|null * * @api */