-
Notifications
You must be signed in to change notification settings - Fork 10
Fix handling of errors in mutations #297
Changes from 3 commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -22,11 +22,9 @@ | |
use Digia\GraphQL\Type\Definition\SerializableTypeInterface; | ||
use Digia\GraphQL\Type\Definition\TypeInterface; | ||
use Digia\GraphQL\Type\Definition\UnionType; | ||
use InvalidArgumentException; | ||
use React\Promise\ExtendedPromiseInterface; | ||
use React\Promise\FulfilledPromise; | ||
use React\Promise\PromiseInterface; | ||
use Throwable; | ||
use function Digia\GraphQL\Type\SchemaMetaFieldDefinition; | ||
use function Digia\GraphQL\Type\TypeMetaFieldDefinition; | ||
use function Digia\GraphQL\Type\TypeNameMetaFieldDefinition; | ||
|
@@ -72,6 +70,7 @@ class Executor | |
|
||
/** | ||
* Executor constructor. | ||
* | ||
* @param ExecutionContext $context | ||
* @param FieldCollector $fieldCollector | ||
* @param ErrorHandlerInterface|null $errorHandler | ||
|
@@ -115,7 +114,7 @@ public function execute(): ?array | |
? $this->executeFieldsSerially($objectType, $rootValue, $path, $fields) | ||
: $this->executeFields($objectType, $rootValue, $path, $fields); | ||
} catch (\Throwable $ex) { | ||
$this->handleError(new ExecutionException($ex->getMessage(), $fields, null, null, null, null, $ex)); | ||
$this->handleError(new ExecutionException($ex->getMessage(), null, null, null, null, null, $ex)); | ||
return null; | ||
} | ||
|
||
|
@@ -125,6 +124,7 @@ public function execute(): ?array | |
/** | ||
* @param Schema $schema | ||
* @param OperationDefinitionNode $operation | ||
* | ||
* @return ObjectType|null | ||
* @throws ExecutionException | ||
*/ | ||
|
@@ -141,6 +141,7 @@ public function getOperationType(Schema $schema, OperationDefinitionNode $operat | |
[$operation] | ||
); | ||
} | ||
|
||
return $mutationType; | ||
case 'subscription': | ||
$subscriptionType = $schema->getSubscriptionType(); | ||
|
@@ -150,6 +151,7 @@ public function getOperationType(Schema $schema, OperationDefinitionNode $operat | |
[$operation] | ||
); | ||
} | ||
|
||
return $subscriptionType; | ||
default: | ||
throw new ExecutionException( | ||
|
@@ -166,9 +168,8 @@ public function getOperationType(Schema $schema, OperationDefinitionNode $operat | |
* @param mixed $rootValue | ||
* @param array $path | ||
* @param array $fields | ||
* | ||
* @return array | ||
* @throws InvalidArgumentException | ||
* @throws Throwable | ||
*/ | ||
public function executeFieldsSerially( | ||
ObjectType $objectType, | ||
|
@@ -189,10 +190,10 @@ public function executeFieldsSerially( | |
return null; | ||
} | ||
|
||
if ($this->isPromise($result)) { | ||
/** @var ExtendedPromiseInterface $result */ | ||
if ($result instanceof ExtendedPromiseInterface) { | ||
return $result->then(function ($resolvedResult) use ($fieldName, $results) { | ||
$results[$fieldName] = $resolvedResult; | ||
|
||
return $results; | ||
}); | ||
} | ||
|
@@ -217,8 +218,8 @@ public function executeFieldsSerially( | |
|
||
$promise->then(function ($resolvedResults) use (&$finalResults) { | ||
$finalResults = $resolvedResults ?? []; | ||
})->otherwise(function ($ex) { | ||
$this->context->addError($ex); | ||
})->otherwise(function (ExecutionException $ex) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. What if There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Sorry, clicked the wrong button. I'm also wondering if this is correct? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. You're right, I've fixed this now. |
||
$this->handleError($ex); | ||
}); | ||
|
||
return $finalResults; | ||
|
@@ -228,6 +229,7 @@ public function executeFieldsSerially( | |
* @param Schema $schema | ||
* @param ObjectType $parentType | ||
* @param string $fieldName | ||
* | ||
* @return Field|null | ||
*/ | ||
public function getFieldDefinition( | ||
|
@@ -251,6 +253,7 @@ public function getFieldDefinition( | |
return $typeNameMetaFieldDefinition; | ||
} | ||
|
||
/** @noinspection PhpUnhandledExceptionInspection */ | ||
$fields = $parentType->getFields(); | ||
|
||
return $fields[$fieldName] ?? null; | ||
|
@@ -262,6 +265,7 @@ public function getFieldDefinition( | |
* @param ResolveInfo $info | ||
* @param array $path | ||
* @param mixed $result | ||
* | ||
* @return array|mixed|null | ||
* @throws \Throwable | ||
*/ | ||
|
@@ -291,22 +295,22 @@ public function completeValueCatchingError( | |
$result | ||
); | ||
|
||
if ($this->isPromise($completed)) { | ||
$context = $this->context; | ||
/** @var ExtendedPromiseInterface $completed */ | ||
return $completed->then(null, function ($error) use ($context, $fieldNodes, $path) { | ||
//@TODO Handle $error better | ||
if ($completed instanceof ExtendedPromiseInterface) { | ||
return $completed->then(null, function ($error) use ($fieldNodes, $path) { | ||
if ($error instanceof \Exception) { | ||
$context->addError($this->buildLocatedError($error, $fieldNodes, $path)); | ||
$this->handleError( | ||
$this->buildLocatedError($error, $fieldNodes, $path) | ||
); | ||
} else { | ||
$context->addError( | ||
$this->handleError( | ||
$this->buildLocatedError( | ||
new ExecutionException($error ?? 'An unknown error occurred.'), | ||
$fieldNodes, | ||
$path | ||
) | ||
); | ||
} | ||
|
||
return new FulfilledPromise(null); | ||
}); | ||
} | ||
|
@@ -318,13 +322,13 @@ public function completeValueCatchingError( | |
} | ||
} | ||
|
||
|
||
/** | ||
* @param TypeInterface $fieldType | ||
* @param FieldNode[] $fieldNodes | ||
* @param ResolveInfo $info | ||
* @param array $path | ||
* @param mixed $result | ||
* | ||
* @return array|mixed | ||
* @throws \Throwable | ||
*/ | ||
|
@@ -357,7 +361,9 @@ public function completeValueWithLocatedError( | |
* @param mixed $rootValue | ||
* @param array $path | ||
* @param array $fields | ||
* | ||
* @return array | ||
* @throws ExecutionException | ||
* @throws \Throwable | ||
*/ | ||
protected function executeFields( | ||
|
@@ -379,7 +385,7 @@ protected function executeFields( | |
continue; | ||
} | ||
|
||
$doesContainPromise = $doesContainPromise || $this->isPromise($result); | ||
$doesContainPromise = $doesContainPromise || $result instanceof ExtendedPromiseInterface; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @crisu83 Not sure why you don't like the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Because it’s better to use There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @crisu83 If you insist, feel free to change, but we actually don't really use it in the code. |
||
|
||
$finalResults[$fieldName] = $result; | ||
} | ||
|
@@ -404,9 +410,10 @@ protected function executeFields( | |
* @param mixed $rootValue | ||
* @param FieldNode[] $fieldNodes | ||
* @param array $path | ||
* | ||
* @return array|mixed|null | ||
* @throws UndefinedException | ||
* @throws Throwable | ||
* @throws \Throwable | ||
*/ | ||
protected function resolveField( | ||
ObjectType $parentType, | ||
|
@@ -450,6 +457,7 @@ protected function resolveField( | |
/** | ||
* @param Field $field | ||
* @param ObjectType $objectType | ||
* | ||
* @return callable|mixed|null | ||
*/ | ||
protected function determineResolveCallback(Field $field, ObjectType $objectType) | ||
|
@@ -471,6 +479,7 @@ protected function determineResolveCallback(Field $field, ObjectType $objectType | |
* @param ResolveInfo $info | ||
* @param array $path | ||
* @param mixed $result | ||
* | ||
* @return array|mixed | ||
* @throws InvariantException | ||
* @throws InvalidTypeException | ||
|
@@ -484,8 +493,7 @@ protected function completeValue( | |
array $path, | ||
&$result | ||
) { | ||
if ($this->isPromise($result)) { | ||
/** @var ExtendedPromiseInterface $result */ | ||
if ($result instanceof ExtendedPromiseInterface) { | ||
return $result->then(function (&$value) use ($returnType, $fieldNodes, $info, $path) { | ||
return $this->completeValue($returnType, $fieldNodes, $info, $path, $value); | ||
}); | ||
|
@@ -552,6 +560,7 @@ protected function completeValue( | |
* @param ResolveInfo $info | ||
* @param array $path | ||
* @param mixed $result | ||
* | ||
* @return array|PromiseInterface | ||
* @throws ExecutionException | ||
* @throws InvalidTypeException | ||
|
@@ -572,8 +581,7 @@ protected function completeAbstractValue( | |
$runtimeType = $this->defaultTypeResolver($result, $this->context->getContextValue(), $info, $returnType); | ||
} | ||
|
||
if ($this->isPromise($runtimeType)) { | ||
/** @var ExtendedPromiseInterface $runtimeType */ | ||
if ($runtimeType instanceof ExtendedPromiseInterface) { | ||
return $runtimeType->then(function ($resolvedRuntimeType) use ( | ||
$returnType, | ||
$fieldNodes, | ||
|
@@ -605,6 +613,7 @@ protected function completeAbstractValue( | |
* @param NamedTypeInterface $returnType | ||
* @param ResolveInfo $info | ||
* @param mixed $result | ||
* | ||
* @return TypeInterface|ObjectType|null | ||
* @throws ExecutionException | ||
* @throws InvariantException | ||
|
@@ -666,6 +675,7 @@ protected function ensureValidRuntimeType( | |
* @param mixed $context | ||
* @param ResolveInfo $info | ||
* @param AbstractTypeInterface $abstractType | ||
* | ||
* @return NamedTypeInterface|mixed|null | ||
* @throws InvariantException | ||
*/ | ||
|
@@ -686,7 +696,7 @@ protected function defaultTypeResolver( | |
foreach ($possibleTypes as $index => $type) { | ||
$isTypeOfResult = $type->isTypeOf($value, $context, $info); | ||
|
||
if ($this->isPromise($isTypeOfResult)) { | ||
if ($isTypeOfResult instanceof ExtendedPromiseInterface) { | ||
$promisedIsTypeOfResults[$index] = $isTypeOfResult; | ||
continue; | ||
} | ||
|
@@ -713,6 +723,7 @@ protected function defaultTypeResolver( | |
return $possibleTypes[$index]; | ||
} | ||
} | ||
|
||
return null; | ||
}); | ||
} | ||
|
@@ -726,6 +737,7 @@ protected function defaultTypeResolver( | |
* @param ResolveInfo $info | ||
* @param array $path | ||
* @param mixed $result | ||
* | ||
* @return array|\React\Promise\Promise | ||
* @throws \Throwable | ||
*/ | ||
|
@@ -757,7 +769,7 @@ protected function completeListValue( | |
$fieldPath[] = $key; | ||
$completedItem = $this->completeValueCatchingError($itemType, $fieldNodes, $info, $fieldPath, $item); | ||
$completedItems[] = $completedItem; | ||
$doesContainPromise = $doesContainPromise || $this->isPromise($completedItem); | ||
$doesContainPromise = $doesContainPromise || $completedItem instanceof ExtendedPromiseInterface; | ||
} | ||
|
||
return $doesContainPromise | ||
|
@@ -768,6 +780,7 @@ protected function completeListValue( | |
/** | ||
* @param LeafTypeInterface|SerializableTypeInterface $returnType | ||
* @param mixed $result | ||
* | ||
* @return mixed | ||
* @throws ExecutionException | ||
*/ | ||
|
@@ -791,6 +804,7 @@ protected function completeLeafValue($returnType, &$result) | |
* @param ResolveInfo $info | ||
* @param array $path | ||
* @param mixed $result | ||
* | ||
* @return array | ||
* @throws ExecutionException | ||
* @throws InvalidTypeException | ||
|
@@ -825,6 +839,7 @@ protected function completeObjectValue( | |
* @param mixed $rootValue | ||
* @param ExecutionContext $context | ||
* @param ResolveInfo $info | ||
* | ||
* @return array|\Throwable | ||
*/ | ||
protected function resolveFieldValueOrError( | ||
|
@@ -854,11 +869,12 @@ protected function resolveFieldValueOrError( | |
* @param FieldNode[] $fieldNodes | ||
* @param array $path | ||
* @param mixed $result | ||
* | ||
* @return array | ||
* @throws ExecutionException | ||
* @throws InvalidTypeException | ||
* @throws InvariantException | ||
* @throws Throwable | ||
* @throws \Throwable | ||
*/ | ||
protected function executeSubFields( | ||
ObjectType $returnType, | ||
|
@@ -887,19 +903,11 @@ protected function executeSubFields( | |
return $result; | ||
} | ||
|
||
/** | ||
* @param mixed $value | ||
* @return bool | ||
*/ | ||
protected function isPromise($value): bool | ||
{ | ||
return $value instanceof ExtendedPromiseInterface; | ||
} | ||
|
||
/** | ||
* @param \Throwable $originalException | ||
* @param array $nodes | ||
* @param array $path | ||
* | ||
* @return ExecutionException | ||
*/ | ||
protected function buildLocatedError( | ||
|
@@ -933,6 +941,7 @@ protected function buildLocatedError( | |
* @param ObjectType $parentType | ||
* @param array|null $path | ||
* @param ExecutionContext $context | ||
* | ||
* @return ResolveInfo | ||
*/ | ||
protected function createResolveInfo( | ||
|
@@ -976,6 +985,7 @@ protected function handleError(ExecutionException $error) | |
* @param array $arguments | ||
* @param mixed $contextValues | ||
* @param ResolveInfo $info | ||
* | ||
* @return mixed|null | ||
*/ | ||
public static function defaultFieldResolver($rootValue, array $arguments, $contextValues, ResolveInfo $info) | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@crisu83 not sure why you removed the
$fields
?