mirror of
https://github.com/retailcrm/graphql-php.git
synced 2025-04-03 12:43:31 +03:00
622 lines
22 KiB
PHP
622 lines
22 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace GraphQL\Utils;
|
|
|
|
use GraphQL\Error\Error;
|
|
use GraphQL\Language\AST\DirectiveDefinitionNode;
|
|
use GraphQL\Language\AST\DocumentNode;
|
|
use GraphQL\Language\AST\Node;
|
|
use GraphQL\Language\AST\NodeKind;
|
|
use GraphQL\Language\AST\ObjectTypeExtensionNode;
|
|
use GraphQL\Language\AST\SchemaDefinitionNode;
|
|
use GraphQL\Language\AST\SchemaTypeExtensionNode;
|
|
use GraphQL\Language\AST\TypeDefinitionNode;
|
|
use GraphQL\Language\AST\TypeExtensionNode;
|
|
use GraphQL\Type\Definition\CustomScalarType;
|
|
use GraphQL\Type\Definition\Directive;
|
|
use GraphQL\Type\Definition\EnumType;
|
|
use GraphQL\Type\Definition\EnumValueDefinition;
|
|
use GraphQL\Type\Definition\FieldArgument;
|
|
use GraphQL\Type\Definition\InputObjectType;
|
|
use GraphQL\Type\Definition\InterfaceType;
|
|
use GraphQL\Type\Definition\ListOfType;
|
|
use GraphQL\Type\Definition\NamedType;
|
|
use GraphQL\Type\Definition\NonNull;
|
|
use GraphQL\Type\Definition\ObjectType;
|
|
use GraphQL\Type\Definition\Type;
|
|
use GraphQL\Type\Definition\UnionType;
|
|
use GraphQL\Type\Introspection;
|
|
use GraphQL\Type\Schema;
|
|
use GraphQL\Validator\DocumentValidator;
|
|
use function array_keys;
|
|
use function array_map;
|
|
use function array_merge;
|
|
use function array_values;
|
|
use function count;
|
|
|
|
class SchemaExtender
|
|
{
|
|
const SCHEMA_EXTENSION = 'SchemaExtension';
|
|
|
|
/** @var Type[] */
|
|
protected static $extendTypeCache;
|
|
|
|
/** @var mixed[] */
|
|
protected static $typeExtensionsMap;
|
|
|
|
/** @var ASTDefinitionBuilder */
|
|
protected static $astBuilder;
|
|
|
|
/**
|
|
* @return TypeExtensionNode[]|null
|
|
*/
|
|
protected static function getExtensionASTNodes(NamedType $type) : ?array
|
|
{
|
|
if (! $type instanceof Type) {
|
|
return null;
|
|
}
|
|
|
|
$name = $type->name;
|
|
if ($type->extensionASTNodes !== null) {
|
|
if (isset(static::$typeExtensionsMap[$name])) {
|
|
return array_merge($type->extensionASTNodes, static::$typeExtensionsMap[$name]);
|
|
}
|
|
|
|
return $type->extensionASTNodes;
|
|
}
|
|
return static::$typeExtensionsMap[$name] ?? null;
|
|
}
|
|
|
|
/**
|
|
* @throws Error
|
|
*/
|
|
protected static function checkExtensionNode(Type $type, Node $node) : void
|
|
{
|
|
switch ($node->kind) {
|
|
case NodeKind::OBJECT_TYPE_EXTENSION:
|
|
if (! ($type instanceof ObjectType)) {
|
|
throw new Error(
|
|
'Cannot extend non-object type "' . $type->name . '".',
|
|
[$node]
|
|
);
|
|
}
|
|
break;
|
|
case NodeKind::INTERFACE_TYPE_EXTENSION:
|
|
if (! ($type instanceof InterfaceType)) {
|
|
throw new Error(
|
|
'Cannot extend non-interface type "' . $type->name . '".',
|
|
[$node]
|
|
);
|
|
}
|
|
break;
|
|
case NodeKind::ENUM_TYPE_EXTENSION:
|
|
if (! ($type instanceof EnumType)) {
|
|
throw new Error(
|
|
'Cannot extend non-enum type "' . $type->name . '".',
|
|
[$node]
|
|
);
|
|
}
|
|
break;
|
|
case NodeKind::UNION_TYPE_EXTENSION:
|
|
if (! ($type instanceof UnionType)) {
|
|
throw new Error(
|
|
'Cannot extend non-union type "' . $type->name . '".',
|
|
[$node]
|
|
);
|
|
}
|
|
break;
|
|
case NodeKind::INPUT_OBJECT_TYPE_EXTENSION:
|
|
if (! ($type instanceof InputObjectType)) {
|
|
throw new Error(
|
|
'Cannot extend non-input object type "' . $type->name . '".',
|
|
[$node]
|
|
);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
protected static function extendCustomScalarType(CustomScalarType $type) : CustomScalarType
|
|
{
|
|
return new CustomScalarType([
|
|
'name' => $type->name,
|
|
'description' => $type->description,
|
|
'astNode' => $type->astNode,
|
|
'serialize' => $type->config['serialize'] ?? null,
|
|
'parseValue' => $type->config['parseValue'] ?? null,
|
|
'parseLiteral' => $type->config['parseLiteral'] ?? null,
|
|
'extensionASTNodes' => static::getExtensionASTNodes($type),
|
|
]);
|
|
}
|
|
|
|
protected static function extendUnionType(UnionType $type) : UnionType
|
|
{
|
|
return new UnionType([
|
|
'name' => $type->name,
|
|
'description' => $type->description,
|
|
'types' => static function () use ($type) {
|
|
return static::extendPossibleTypes($type);
|
|
},
|
|
'astNode' => $type->astNode,
|
|
'resolveType' => $type->config['resolveType'] ?? null,
|
|
'extensionASTNodes' => static::getExtensionASTNodes($type),
|
|
]);
|
|
}
|
|
|
|
protected static function extendEnumType(EnumType $type) : EnumType
|
|
{
|
|
return new EnumType([
|
|
'name' => $type->name,
|
|
'description' => $type->description,
|
|
'values' => static::extendValueMap($type),
|
|
'astNode' => $type->astNode,
|
|
'extensionASTNodes' => static::getExtensionASTNodes($type),
|
|
]);
|
|
}
|
|
|
|
protected static function extendInputObjectType(InputObjectType $type) : InputObjectType
|
|
{
|
|
return new InputObjectType([
|
|
'name' => $type->name,
|
|
'description' => $type->description,
|
|
'fields' => static function () use ($type) {
|
|
return static::extendInputFieldMap($type);
|
|
},
|
|
'astNode' => $type->astNode,
|
|
'extensionASTNodes' => static::getExtensionASTNodes($type),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* @return mixed[]
|
|
*/
|
|
protected static function extendInputFieldMap(InputObjectType $type) : array
|
|
{
|
|
$newFieldMap = [];
|
|
$oldFieldMap = $type->getFields();
|
|
foreach ($oldFieldMap as $fieldName => $field) {
|
|
$newFieldMap[$fieldName] = [
|
|
'description' => $field->description,
|
|
'type' => static::extendType($field->type),
|
|
'astNode' => $field->astNode,
|
|
];
|
|
|
|
if (! $field->defaultValueExists()) {
|
|
continue;
|
|
}
|
|
|
|
$newFieldMap[$fieldName]['defaultValue'] = $field->defaultValue;
|
|
}
|
|
|
|
$extensions = static::$typeExtensionsMap[$type->name] ?? null;
|
|
if ($extensions !== null) {
|
|
foreach ($extensions as $extension) {
|
|
foreach ($extension->fields as $field) {
|
|
$fieldName = $field->name->value;
|
|
if (isset($oldFieldMap[$fieldName])) {
|
|
throw new Error('Field "' . $type->name . '.' . $fieldName . '" already exists in the schema. It cannot also be defined in this type extension.', [$field]);
|
|
}
|
|
|
|
$newFieldMap[$fieldName] = static::$astBuilder->buildInputField($field);
|
|
}
|
|
}
|
|
}
|
|
|
|
return $newFieldMap;
|
|
}
|
|
|
|
/**
|
|
* @return mixed[]
|
|
*/
|
|
protected static function extendValueMap(EnumType $type) : array
|
|
{
|
|
$newValueMap = [];
|
|
/** @var EnumValueDefinition[] $oldValueMap */
|
|
$oldValueMap = [];
|
|
foreach ($type->getValues() as $value) {
|
|
$oldValueMap[$value->name] = $value;
|
|
}
|
|
|
|
foreach ($oldValueMap as $key => $value) {
|
|
$newValueMap[$key] = [
|
|
'name' => $value->name,
|
|
'description' => $value->description,
|
|
'value' => $value->value,
|
|
'deprecationReason' => $value->deprecationReason,
|
|
'astNode' => $value->astNode,
|
|
];
|
|
}
|
|
|
|
$extensions = static::$typeExtensionsMap[$type->name] ?? null;
|
|
if ($extensions !== null) {
|
|
foreach ($extensions as $extension) {
|
|
foreach ($extension->values as $value) {
|
|
$valueName = $value->name->value;
|
|
if (isset($oldValueMap[$valueName])) {
|
|
throw new Error('Enum value "' . $type->name . '.' . $valueName . '" already exists in the schema. It cannot also be defined in this type extension.', [$value]);
|
|
}
|
|
$newValueMap[$valueName] = static::$astBuilder->buildEnumValue($value);
|
|
}
|
|
}
|
|
}
|
|
|
|
return $newValueMap;
|
|
}
|
|
|
|
/**
|
|
* @return ObjectType[]
|
|
*/
|
|
protected static function extendPossibleTypes(UnionType $type) : array
|
|
{
|
|
$possibleTypes = array_map(static function ($type) {
|
|
return static::extendNamedType($type);
|
|
}, $type->getTypes());
|
|
|
|
$extensions = static::$typeExtensionsMap[$type->name] ?? null;
|
|
if ($extensions !== null) {
|
|
foreach ($extensions as $extension) {
|
|
foreach ($extension->types as $namedType) {
|
|
$possibleTypes[] = static::$astBuilder->buildType($namedType);
|
|
}
|
|
}
|
|
}
|
|
|
|
return $possibleTypes;
|
|
}
|
|
|
|
/**
|
|
* @return InterfaceType[]
|
|
*/
|
|
protected static function extendImplementedInterfaces(ObjectType $type) : array
|
|
{
|
|
$interfaces = array_map(static function (InterfaceType $interfaceType) {
|
|
return static::extendNamedType($interfaceType);
|
|
}, $type->getInterfaces());
|
|
|
|
$extensions = static::$typeExtensionsMap[$type->name] ?? null;
|
|
if ($extensions !== null) {
|
|
/** @var ObjectTypeExtensionNode $extension */
|
|
foreach ($extensions as $extension) {
|
|
foreach ($extension->interfaces as $namedType) {
|
|
$interfaces[] = static::$astBuilder->buildType($namedType);
|
|
}
|
|
}
|
|
}
|
|
return $interfaces;
|
|
}
|
|
|
|
protected static function extendType($typeDef)
|
|
{
|
|
if ($typeDef instanceof ListOfType) {
|
|
return Type::listOf(static::extendType($typeDef->ofType));
|
|
}
|
|
|
|
if ($typeDef instanceof NonNull) {
|
|
return Type::nonNull(static::extendType($typeDef->getWrappedType()));
|
|
}
|
|
|
|
return static::extendNamedType($typeDef);
|
|
}
|
|
|
|
/**
|
|
* @param FieldArgument[] $args
|
|
*
|
|
* @return mixed[]
|
|
*/
|
|
protected static function extendArgs(array $args) : array
|
|
{
|
|
return Utils::keyValMap(
|
|
$args,
|
|
static function (FieldArgument $arg) {
|
|
return $arg->name;
|
|
},
|
|
static function (FieldArgument $arg) {
|
|
$def = [
|
|
'type' => static::extendType($arg->getType()),
|
|
'description' => $arg->description,
|
|
'astNode' => $arg->astNode,
|
|
];
|
|
|
|
if ($arg->defaultValueExists()) {
|
|
$def['defaultValue'] = $arg->defaultValue;
|
|
}
|
|
|
|
return $def;
|
|
}
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @param InterfaceType|ObjectType $type
|
|
*
|
|
* @return mixed[]
|
|
*
|
|
* @throws Error
|
|
*/
|
|
protected static function extendFieldMap($type) : array
|
|
{
|
|
$newFieldMap = [];
|
|
$oldFieldMap = $type->getFields();
|
|
|
|
foreach (array_keys($oldFieldMap) as $fieldName) {
|
|
$field = $oldFieldMap[$fieldName];
|
|
|
|
$newFieldMap[$fieldName] = [
|
|
'name' => $fieldName,
|
|
'description' => $field->description,
|
|
'deprecationReason' => $field->deprecationReason,
|
|
'type' => static::extendType($field->getType()),
|
|
'args' => static::extendArgs($field->args),
|
|
'astNode' => $field->astNode,
|
|
'resolveFn' => $field->resolveFn,
|
|
];
|
|
}
|
|
|
|
$extensions = static::$typeExtensionsMap[$type->name] ?? null;
|
|
if ($extensions !== null) {
|
|
foreach ($extensions as $extension) {
|
|
foreach ($extension->fields as $field) {
|
|
$fieldName = $field->name->value;
|
|
if (isset($oldFieldMap[$fieldName])) {
|
|
throw new Error('Field "' . $type->name . '.' . $fieldName . '" already exists in the schema. It cannot also be defined in this type extension.', [$field]);
|
|
}
|
|
|
|
$newFieldMap[$fieldName] = static::$astBuilder->buildField($field);
|
|
}
|
|
}
|
|
}
|
|
|
|
return $newFieldMap;
|
|
}
|
|
|
|
protected static function extendObjectType(ObjectType $type) : ObjectType
|
|
{
|
|
return new ObjectType([
|
|
'name' => $type->name,
|
|
'description' => $type->description,
|
|
'interfaces' => static function () use ($type) {
|
|
return static::extendImplementedInterfaces($type);
|
|
},
|
|
'fields' => static function () use ($type) {
|
|
return static::extendFieldMap($type);
|
|
},
|
|
'astNode' => $type->astNode,
|
|
'extensionASTNodes' => static::getExtensionASTNodes($type),
|
|
'isTypeOf' => $type->config['isTypeOf'] ?? null,
|
|
]);
|
|
}
|
|
|
|
protected static function extendInterfaceType(InterfaceType $type) : InterfaceType
|
|
{
|
|
return new InterfaceType([
|
|
'name' => $type->name,
|
|
'description' => $type->description,
|
|
'fields' => static function () use ($type) {
|
|
return static::extendFieldMap($type);
|
|
},
|
|
'astNode' => $type->astNode,
|
|
'extensionASTNodes' => static::getExtensionASTNodes($type),
|
|
'resolveType' => $type->config['resolveType'] ?? null,
|
|
]);
|
|
}
|
|
|
|
protected static function isSpecifiedScalarType(Type $type) : bool
|
|
{
|
|
return $type instanceof NamedType &&
|
|
(
|
|
$type->name === Type::STRING ||
|
|
$type->name === Type::INT ||
|
|
$type->name === Type::FLOAT ||
|
|
$type->name === Type::BOOLEAN ||
|
|
$type->name === Type::ID
|
|
);
|
|
}
|
|
|
|
protected static function extendNamedType(Type $type)
|
|
{
|
|
if (Introspection::isIntrospectionType($type) || static::isSpecifiedScalarType($type)) {
|
|
return $type;
|
|
}
|
|
|
|
$name = $type->name;
|
|
if (! isset(static::$extendTypeCache[$name])) {
|
|
if ($type instanceof CustomScalarType) {
|
|
static::$extendTypeCache[$name] = static::extendCustomScalarType($type);
|
|
} elseif ($type instanceof ObjectType) {
|
|
static::$extendTypeCache[$name] = static::extendObjectType($type);
|
|
} elseif ($type instanceof InterfaceType) {
|
|
static::$extendTypeCache[$name] = static::extendInterfaceType($type);
|
|
} elseif ($type instanceof UnionType) {
|
|
static::$extendTypeCache[$name] = static::extendUnionType($type);
|
|
} elseif ($type instanceof EnumType) {
|
|
static::$extendTypeCache[$name] = static::extendEnumType($type);
|
|
} elseif ($type instanceof InputObjectType) {
|
|
static::$extendTypeCache[$name] = static::extendInputObjectType($type);
|
|
}
|
|
}
|
|
|
|
return static::$extendTypeCache[$name];
|
|
}
|
|
|
|
/**
|
|
* @return mixed|null
|
|
*/
|
|
protected static function extendMaybeNamedType(?NamedType $type = null)
|
|
{
|
|
if ($type !== null) {
|
|
return static::extendNamedType($type);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @param DirectiveDefinitionNode[] $directiveDefinitions
|
|
*
|
|
* @return Directive[]
|
|
*/
|
|
protected static function getMergedDirectives(Schema $schema, array $directiveDefinitions) : array
|
|
{
|
|
$existingDirectives = array_map(static function (Directive $directive) {
|
|
return static::extendDirective($directive);
|
|
}, $schema->getDirectives());
|
|
|
|
Utils::invariant(count($existingDirectives) > 0, 'schema must have default directives');
|
|
|
|
return array_merge(
|
|
$existingDirectives,
|
|
array_map(static function (DirectiveDefinitionNode $directive) {
|
|
return static::$astBuilder->buildDirective($directive);
|
|
}, $directiveDefinitions)
|
|
);
|
|
}
|
|
|
|
protected static function extendDirective(Directive $directive) : Directive
|
|
{
|
|
return new Directive([
|
|
'name' => $directive->name,
|
|
'description' => $directive->description,
|
|
'locations' => $directive->locations,
|
|
'args' => static::extendArgs($directive->args),
|
|
'astNode' => $directive->astNode,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* @param mixed[]|null $options
|
|
*/
|
|
public static function extend(Schema $schema, DocumentNode $documentAST, ?array $options = null) : Schema
|
|
{
|
|
if ($options === null || ! (isset($options['assumeValid']) || isset($options['assumeValidSDL']))) {
|
|
DocumentValidator::assertValidSDLExtension($documentAST, $schema);
|
|
}
|
|
|
|
$typeDefinitionMap = [];
|
|
static::$typeExtensionsMap = [];
|
|
$directiveDefinitions = [];
|
|
/** @var SchemaDefinitionNode|null $schemaDef */
|
|
$schemaDef = null;
|
|
/** @var SchemaTypeExtensionNode[] $schemaExtensions */
|
|
$schemaExtensions = [];
|
|
|
|
$definitionsCount = count($documentAST->definitions);
|
|
for ($i = 0; $i < $definitionsCount; $i++) {
|
|
|
|
/** @var Node $def */
|
|
$def = $documentAST->definitions[$i];
|
|
|
|
if ($def instanceof SchemaDefinitionNode) {
|
|
$schemaDef = $def;
|
|
} elseif ($def instanceof SchemaTypeExtensionNode) {
|
|
$schemaExtensions[] = $def;
|
|
} elseif ($def instanceof TypeDefinitionNode) {
|
|
$typeName = isset($def->name) ? $def->name->value : null;
|
|
if ($schema->getType($typeName)) {
|
|
throw new Error('Type "' . $typeName . '" already exists in the schema. It cannot also be defined in this type definition.', [$def]);
|
|
}
|
|
$typeDefinitionMap[$typeName] = $def;
|
|
} elseif ($def instanceof TypeExtensionNode) {
|
|
$extendedTypeName = isset($def->name) ? $def->name->value : null;
|
|
$existingType = $schema->getType($extendedTypeName);
|
|
if ($existingType === null) {
|
|
throw new Error('Cannot extend type "' . $extendedTypeName . '" because it does not exist in the existing schema.', [$def]);
|
|
}
|
|
|
|
static::checkExtensionNode($existingType, $def);
|
|
|
|
$existingTypeExtensions = static::$typeExtensionsMap[$extendedTypeName] ?? null;
|
|
static::$typeExtensionsMap[$extendedTypeName] = $existingTypeExtensions !== null ? array_merge($existingTypeExtensions, [$def]) : [$def];
|
|
} elseif ($def instanceof DirectiveDefinitionNode) {
|
|
$directiveName = $def->name->value;
|
|
$existingDirective = $schema->getDirective($directiveName);
|
|
if ($existingDirective !== null) {
|
|
throw new Error('Directive "' . $directiveName . '" already exists in the schema. It cannot be redefined.', [$def]);
|
|
}
|
|
$directiveDefinitions[] = $def;
|
|
}
|
|
}
|
|
|
|
if (count(static::$typeExtensionsMap) === 0 &&
|
|
count($typeDefinitionMap) === 0 &&
|
|
count($directiveDefinitions) === 0 &&
|
|
count($schemaExtensions) === 0 &&
|
|
$schemaDef === null
|
|
) {
|
|
return $schema;
|
|
}
|
|
|
|
static::$astBuilder = new ASTDefinitionBuilder(
|
|
$typeDefinitionMap,
|
|
$options,
|
|
static function (string $typeName) use ($schema) {
|
|
/** @var NamedType $existingType */
|
|
$existingType = $schema->getType($typeName);
|
|
if ($existingType !== null) {
|
|
return static::extendNamedType($existingType);
|
|
}
|
|
|
|
throw new Error('Unknown type: "' . $typeName . '". Ensure that this type exists either in the original schema, or is added in a type definition.', [$typeName]);
|
|
}
|
|
);
|
|
|
|
static::$extendTypeCache = [];
|
|
|
|
$operationTypes = [
|
|
'query' => static::extendMaybeNamedType($schema->getQueryType()),
|
|
'mutation' => static::extendMaybeNamedType($schema->getMutationType()),
|
|
'subscription' => static::extendMaybeNamedType($schema->getSubscriptionType()),
|
|
];
|
|
|
|
if ($schemaDef) {
|
|
foreach ($schemaDef->operationTypes as $operationType) {
|
|
$operation = $operationType->operation;
|
|
$type = $operationType->type;
|
|
|
|
if (isset($operationTypes[$operation])) {
|
|
throw new Error('Must provide only one ' . $operation . ' type in schema.');
|
|
}
|
|
|
|
$operationTypes[$operation] = static::$astBuilder->buildType($type);
|
|
}
|
|
}
|
|
|
|
foreach ($schemaExtensions as $schemaExtension) {
|
|
if (! $schemaExtension->operationTypes) {
|
|
continue;
|
|
}
|
|
|
|
foreach ($schemaExtension->operationTypes as $operationType) {
|
|
$operation = $operationType->operation;
|
|
if (isset($operationTypes[$operation])) {
|
|
throw new Error('Must provide only one ' . $operation . ' type in schema.');
|
|
}
|
|
$operationTypes[$operation] = static::$astBuilder->buildType($operationType->type);
|
|
}
|
|
}
|
|
|
|
$schemaExtensionASTNodes = count($schemaExtensions) > 0
|
|
? ($schema->extensionASTNodes ? array_merge($schema->extensionASTNodes, $schemaExtensions) : $schemaExtensions)
|
|
: $schema->extensionASTNodes;
|
|
|
|
$types = array_merge(
|
|
array_map(static function ($type) {
|
|
return static::extendType($type);
|
|
}, array_values($schema->getTypeMap())),
|
|
array_map(static function ($type) {
|
|
return static::$astBuilder->buildType($type);
|
|
}, array_values($typeDefinitionMap))
|
|
);
|
|
|
|
return new Schema([
|
|
'query' => $operationTypes['query'],
|
|
'mutation' => $operationTypes['mutation'],
|
|
'subscription' => $operationTypes['subscription'],
|
|
'types' => $types,
|
|
'directives' => static::getMergedDirectives($schema, $directiveDefinitions),
|
|
'astNode' => $schema->getAstNode(),
|
|
'extensionASTNodes' => $schemaExtensionASTNodes,
|
|
]);
|
|
}
|
|
}
|