mirror of
https://github.com/retailcrm/graphql-php.git
synced 2025-03-12 15:56:06 +03:00
279 lines
10 KiB
PHP
279 lines
10 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace GraphQL\Executor;
|
|
|
|
use GraphQL\Error\Error;
|
|
use GraphQL\Language\AST\ArgumentNode;
|
|
use GraphQL\Language\AST\DirectiveNode;
|
|
use GraphQL\Language\AST\EnumValueDefinitionNode;
|
|
use GraphQL\Language\AST\FieldDefinitionNode;
|
|
use GraphQL\Language\AST\FieldNode;
|
|
use GraphQL\Language\AST\FragmentSpreadNode;
|
|
use GraphQL\Language\AST\InlineFragmentNode;
|
|
use GraphQL\Language\AST\Node;
|
|
use GraphQL\Language\AST\NodeList;
|
|
use GraphQL\Language\AST\ValueNode;
|
|
use GraphQL\Language\AST\VariableDefinitionNode;
|
|
use GraphQL\Language\AST\VariableNode;
|
|
use GraphQL\Language\Printer;
|
|
use GraphQL\Type\Definition\Directive;
|
|
use GraphQL\Type\Definition\FieldDefinition;
|
|
use GraphQL\Type\Definition\InputType;
|
|
use GraphQL\Type\Definition\NonNull;
|
|
use GraphQL\Type\Definition\Type;
|
|
use GraphQL\Type\Schema;
|
|
use GraphQL\Utils\AST;
|
|
use GraphQL\Utils\TypeInfo;
|
|
use GraphQL\Utils\Utils;
|
|
use GraphQL\Utils\Value;
|
|
use stdClass;
|
|
use Throwable;
|
|
use function array_key_exists;
|
|
use function array_map;
|
|
use function sprintf;
|
|
|
|
class Values
|
|
{
|
|
/**
|
|
* Prepares an object map of variables of the correct type based on the provided
|
|
* variable definitions and arbitrary input. If the input cannot be coerced
|
|
* to match the variable definitions, a Error will be thrown.
|
|
*
|
|
* @param VariableDefinitionNode[] $varDefNodes
|
|
* @param mixed[] $inputs
|
|
*
|
|
* @return mixed[]
|
|
*/
|
|
public static function getVariableValues(Schema $schema, $varDefNodes, array $inputs)
|
|
{
|
|
$errors = [];
|
|
$coercedValues = [];
|
|
foreach ($varDefNodes as $varDefNode) {
|
|
$varName = $varDefNode->variable->name->value;
|
|
/** @var InputType|Type $varType */
|
|
$varType = TypeInfo::typeFromAST($schema, $varDefNode->type);
|
|
|
|
if (Type::isInputType($varType)) {
|
|
if (array_key_exists($varName, $inputs)) {
|
|
$value = $inputs[$varName];
|
|
$coerced = Value::coerceValue($value, $varType, $varDefNode);
|
|
/** @var Error[] $coercionErrors */
|
|
$coercionErrors = $coerced['errors'];
|
|
if (empty($coercionErrors)) {
|
|
$coercedValues[$varName] = $coerced['value'];
|
|
} else {
|
|
$messagePrelude = sprintf(
|
|
'Variable "$%s" got invalid value %s; ',
|
|
$varName,
|
|
Utils::printSafeJson($value)
|
|
);
|
|
|
|
foreach ($coercionErrors as $error) {
|
|
$errors[] = new Error(
|
|
$messagePrelude . $error->getMessage(),
|
|
$error->getNodes(),
|
|
$error->getSource(),
|
|
$error->getPositions(),
|
|
$error->getPath(),
|
|
$error,
|
|
$error->getExtensions()
|
|
);
|
|
}
|
|
}
|
|
} else {
|
|
if ($varType instanceof NonNull) {
|
|
$errors[] = new Error(
|
|
sprintf(
|
|
'Variable "$%s" of required type "%s" was not provided.',
|
|
$varName,
|
|
$varType
|
|
),
|
|
[$varDefNode]
|
|
);
|
|
} elseif ($varDefNode->defaultValue !== null) {
|
|
$coercedValues[$varName] = AST::valueFromAST($varDefNode->defaultValue, $varType);
|
|
}
|
|
}
|
|
} else {
|
|
$errors[] = new Error(
|
|
sprintf(
|
|
'Variable "$%s" expected value of type "%s" which cannot be used as an input type.',
|
|
$varName,
|
|
Printer::doPrint($varDefNode->type)
|
|
),
|
|
[$varDefNode->type]
|
|
);
|
|
}
|
|
}
|
|
|
|
if (! empty($errors)) {
|
|
return [$errors, null];
|
|
}
|
|
|
|
return [null, $coercedValues];
|
|
}
|
|
|
|
/**
|
|
* Prepares an object map of argument values given a directive definition
|
|
* and a AST node which may contain directives. Optionally also accepts a map
|
|
* of variable values.
|
|
*
|
|
* If the directive does not exist on the node, returns undefined.
|
|
*
|
|
* @param FragmentSpreadNode|FieldNode|InlineFragmentNode|EnumValueDefinitionNode|FieldDefinitionNode $node
|
|
* @param mixed[]|null $variableValues
|
|
*
|
|
* @return mixed[]|null
|
|
*/
|
|
public static function getDirectiveValues(Directive $directiveDef, $node, $variableValues = null)
|
|
{
|
|
if (isset($node->directives) && $node->directives instanceof NodeList) {
|
|
$directiveNode = Utils::find(
|
|
$node->directives,
|
|
static function (DirectiveNode $directive) use ($directiveDef) {
|
|
return $directive->name->value === $directiveDef->name;
|
|
}
|
|
);
|
|
|
|
if ($directiveNode !== null) {
|
|
return self::getArgumentValues($directiveDef, $directiveNode, $variableValues);
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Prepares an object map of argument values given a list of argument
|
|
* definitions and list of argument AST nodes.
|
|
*
|
|
* @param FieldDefinition|Directive $def
|
|
* @param FieldNode|DirectiveNode $node
|
|
* @param mixed[] $variableValues
|
|
*
|
|
* @return mixed[]
|
|
*
|
|
* @throws Error
|
|
*/
|
|
public static function getArgumentValues($def, $node, $variableValues = null)
|
|
{
|
|
if (empty($def->args)) {
|
|
return [];
|
|
}
|
|
|
|
$argumentNodes = $node->arguments;
|
|
if (empty($argumentNodes)) {
|
|
return [];
|
|
}
|
|
|
|
$argumentValueMap = [];
|
|
foreach ($argumentNodes as $argumentNode) {
|
|
$argumentValueMap[$argumentNode->name->value] = $argumentNode->value;
|
|
}
|
|
|
|
return static::getArgumentValuesForMap($def, $argumentValueMap, $variableValues, $node);
|
|
}
|
|
|
|
/**
|
|
* @param FieldDefinition|Directive $fieldDefinition
|
|
* @param ArgumentNode[] $argumentValueMap
|
|
* @param mixed[] $variableValues
|
|
* @param Node|null $referenceNode
|
|
*
|
|
* @return mixed[]
|
|
*
|
|
* @throws Error
|
|
*/
|
|
public static function getArgumentValuesForMap($fieldDefinition, $argumentValueMap, $variableValues = null, $referenceNode = null)
|
|
{
|
|
$argumentDefinitions = $fieldDefinition->args;
|
|
$coercedValues = [];
|
|
|
|
foreach ($argumentDefinitions as $argumentDefinition) {
|
|
$name = $argumentDefinition->name;
|
|
$argType = $argumentDefinition->getType();
|
|
$argumentValueNode = $argumentValueMap[$name] ?? null;
|
|
|
|
if ($argumentValueNode === null) {
|
|
if ($argumentDefinition->defaultValueExists()) {
|
|
$coercedValues[$name] = $argumentDefinition->defaultValue;
|
|
} elseif ($argType instanceof NonNull) {
|
|
throw new Error(
|
|
'Argument "' . $name . '" of required type ' .
|
|
'"' . Utils::printSafe($argType) . '" was not provided.',
|
|
$referenceNode
|
|
);
|
|
}
|
|
} elseif ($argumentValueNode instanceof VariableNode) {
|
|
$variableName = $argumentValueNode->name->value;
|
|
|
|
if ($variableValues !== null && array_key_exists($variableName, $variableValues)) {
|
|
// Note: this does not check that this variable value is correct.
|
|
// This assumes that this query has been validated and the variable
|
|
// usage here is of the correct type.
|
|
$coercedValues[$name] = $variableValues[$variableName];
|
|
} elseif ($argumentDefinition->defaultValueExists()) {
|
|
$coercedValues[$name] = $argumentDefinition->defaultValue;
|
|
} elseif ($argType instanceof NonNull) {
|
|
throw new Error(
|
|
'Argument "' . $name . '" of required type "' . Utils::printSafe($argType) . '" was ' .
|
|
'provided the variable "$' . $variableName . '" which was not provided ' .
|
|
'a runtime value.',
|
|
[$argumentValueNode]
|
|
);
|
|
}
|
|
} else {
|
|
$valueNode = $argumentValueNode;
|
|
$coercedValue = AST::valueFromAST($valueNode, $argType, $variableValues);
|
|
if (Utils::isInvalid($coercedValue)) {
|
|
// Note: ValuesOfCorrectType validation should catch this before
|
|
// execution. This is a runtime check to ensure execution does not
|
|
// continue with an invalid argument value.
|
|
throw new Error(
|
|
'Argument "' . $name . '" has invalid value ' . Printer::doPrint($valueNode) . '.',
|
|
[$argumentValueNode]
|
|
);
|
|
}
|
|
$coercedValues[$name] = $coercedValue;
|
|
}
|
|
}
|
|
|
|
return $coercedValues;
|
|
}
|
|
|
|
/**
|
|
* @deprecated as of 8.0 (Moved to \GraphQL\Utils\AST::valueFromAST)
|
|
*
|
|
* @param ValueNode $valueNode
|
|
* @param mixed[]|null $variables
|
|
*
|
|
* @return mixed[]|stdClass|null
|
|
*/
|
|
public static function valueFromAST($valueNode, InputType $type, ?array $variables = null)
|
|
{
|
|
return AST::valueFromAST($valueNode, $type, $variables);
|
|
}
|
|
|
|
/**
|
|
* @deprecated as of 0.12 (Use coerceValue() directly for richer information)
|
|
*
|
|
* @param mixed[] $value
|
|
*
|
|
* @return string[]
|
|
*/
|
|
public static function isValidPHPValue($value, InputType $type)
|
|
{
|
|
$errors = Value::coerceValue($value, $type)['errors'];
|
|
|
|
return $errors
|
|
? array_map(
|
|
static function (Throwable $error) {
|
|
return $error->getMessage();
|
|
},
|
|
$errors
|
|
) : [];
|
|
}
|
|
}
|