graphql-php/src/Utils/SchemaExtender.php
Simon Podlipsky 90d0156291
Use PHPStan strict rules
Two rules excluded: missing type hints and empty() usage
2018-10-09 17:46:55 +02:00

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,
]);
}
}