diff --git a/docs/reference.md b/docs/reference.md index 70f5614..76ac3ef 100644 --- a/docs/reference.md +++ b/docs/reference.md @@ -171,7 +171,7 @@ static function float() ```php /** * @api - * @param ObjectType|InterfaceType|UnionType|ScalarType|InputObjectType|EnumType|ListOfType|NonNull $wrappedType + * @param Type|ObjectType|InterfaceType|UnionType|ScalarType|InputObjectType|EnumType|ListOfType|NonNull $wrappedType * @return ListOfType */ static function listOf($wrappedType) @@ -1345,7 +1345,6 @@ Also it is possible to override warning handler (which is **trigger_error()** by **Class Constants:** ```php -const WARNING_NAME = 1; const WARNING_ASSIGN = 2; const WARNING_CONFIG = 4; const WARNING_FULL_SCHEMA_SCAN = 8; diff --git a/src/Error/Warning.php b/src/Error/Warning.php index fa6a666..4bdbf66 100644 --- a/src/Error/Warning.php +++ b/src/Error/Warning.php @@ -9,7 +9,6 @@ namespace GraphQL\Error; */ final class Warning { - const WARNING_NAME = 1; const WARNING_ASSIGN = 2; const WARNING_CONFIG = 4; const WARNING_FULL_SCHEMA_SCAN = 8; diff --git a/src/Language/AST/EnumTypeDefinitionNode.php b/src/Language/AST/EnumTypeDefinitionNode.php index e9be727..fc8eb66 100644 --- a/src/Language/AST/EnumTypeDefinitionNode.php +++ b/src/Language/AST/EnumTypeDefinitionNode.php @@ -19,7 +19,7 @@ class EnumTypeDefinitionNode extends Node implements TypeDefinitionNode public $directives; /** - * @var EnumValueDefinitionNode[]|null + * @var EnumValueDefinitionNode[]|null|NodeList */ public $values; diff --git a/src/Language/AST/FieldDefinitionNode.php b/src/Language/AST/FieldDefinitionNode.php index d081d7f..6baf498 100644 --- a/src/Language/AST/FieldDefinitionNode.php +++ b/src/Language/AST/FieldDefinitionNode.php @@ -14,7 +14,7 @@ class FieldDefinitionNode extends Node public $name; /** - * @var InputValueDefinitionNode[] + * @var InputValueDefinitionNode[]|NodeList */ public $arguments; @@ -24,7 +24,7 @@ class FieldDefinitionNode extends Node public $type; /** - * @var DirectiveNode[] + * @var DirectiveNode[]|NodeList */ public $directives; diff --git a/src/Language/Parser.php b/src/Language/Parser.php index 3cdc211..d40414a 100644 --- a/src/Language/Parser.php +++ b/src/Language/Parser.php @@ -980,6 +980,7 @@ class Parser /** * @return OperationTypeDefinitionNode + * @throws SyntaxError */ function parseOperationTypeDefinition() { @@ -1095,11 +1096,12 @@ class Parser /** * @return InputValueDefinitionNode[]|NodeList + * @throws SyntaxError */ function parseArgumentDefs() { if (!$this->peek(Token::PAREN_L)) { - return []; + return new NodeList([]); } return $this->many(Token::PAREN_L, [$this, 'parseInputValueDef'], Token::PAREN_R); } @@ -1357,7 +1359,7 @@ class Parser $fields = $this->parseFieldsDefinition(); if ( - count($interfaces) === 0 && + !$interfaces && count($directives) === 0 && count($fields) === 0 ) { @@ -1412,7 +1414,7 @@ class Parser $types = $this->parseMemberTypesDefinition(); if ( count($directives) === 0 && - count($types) === 0 + !$types ) { throw $this->unexpected(); } diff --git a/src/Type/Definition/Directive.php b/src/Type/Definition/Directive.php index fcf6c2a..b3b9a1a 100644 --- a/src/Type/Definition/Directive.php +++ b/src/Type/Definition/Directive.php @@ -3,6 +3,7 @@ namespace GraphQL\Type\Definition; use GraphQL\Language\AST\DirectiveDefinitionNode; use GraphQL\Language\DirectiveLocation; +use GraphQL\Utils\Utils; /** * Class Directive @@ -159,6 +160,9 @@ class Directive foreach ($config as $key => $value) { $this->{$key} = $value; } + + Utils::invariant($this->name, 'Directive must be named.'); + Utils::invariant(is_array($this->locations), 'Must provide locations for directive.'); $this->config = $config; } } diff --git a/src/Type/Definition/EnumType.php b/src/Type/Definition/EnumType.php index 0019d24..dbb7e33 100644 --- a/src/Type/Definition/EnumType.php +++ b/src/Type/Definition/EnumType.php @@ -11,7 +11,7 @@ use GraphQL\Utils\Utils; * Class EnumType * @package GraphQL\Type\Definition */ -class EnumType extends Type implements InputType, OutputType, LeafType +class EnumType extends Type implements InputType, OutputType, LeafType, NamedType { /** * @var EnumTypeDefinitionNode|null @@ -39,7 +39,7 @@ class EnumType extends Type implements InputType, OutputType, LeafType $config['name'] = $this->tryInferName(); } - Utils::assertValidName($config['name'], !empty($config['isIntrospection'])); + Utils::invariant(is_string($config['name']), 'Must provide name.'); Config::validate($config, [ 'name' => Config::NAME | Config::REQUIRED, @@ -188,24 +188,7 @@ class EnumType extends Type implements InputType, OutputType, LeafType ); $values = $this->getValues(); - - Utils::invariant( - !empty($values), - "{$this->name} values must be not empty." - ); foreach ($values as $value) { - try { - Utils::assertValidName($value->name); - } catch (InvariantViolation $e) { - throw new InvariantViolation( - "{$this->name} has value with invalid name: " . - Utils::printSafe($value->name) . " ({$e->getMessage()})" - ); - } - Utils::invariant( - !in_array($value->name, ['true', 'false', 'null']), - "{$this->name}: \"{$value->name}\" can not be used as an Enum value." - ); Utils::invariant( !isset($value->config['isDeprecated']), "{$this->name}.{$value->name} should provide \"deprecationReason\" instead of \"isDeprecated\"." diff --git a/src/Type/Definition/InputObjectType.php b/src/Type/Definition/InputObjectType.php index d795630..a3661e6 100644 --- a/src/Type/Definition/InputObjectType.php +++ b/src/Type/Definition/InputObjectType.php @@ -9,7 +9,7 @@ use GraphQL\Utils\Utils; * Class InputObjectType * @package GraphQL\Type\Definition */ -class InputObjectType extends Type implements InputType +class InputObjectType extends Type implements InputType, NamedType { /** * @var InputObjectField[] @@ -31,7 +31,7 @@ class InputObjectType extends Type implements InputType $config['name'] = $this->tryInferName(); } - Utils::assertValidName($config['name']); + Utils::invariant(is_string($config['name']), 'Must provide name.'); Config::validate($config, [ 'name' => Config::NAME | Config::REQUIRED, @@ -91,41 +91,4 @@ class InputObjectType extends Type implements InputType Utils::invariant(isset($this->fields[$name]), "Field '%s' is not defined for type '%s'", $name, $this->name); return $this->fields[$name]; } - - /** - * @throws InvariantViolation - */ - public function assertValid() - { - parent::assertValid(); - - $fields = $this->getFields(); - - Utils::invariant( - !empty($fields), - "{$this->name} fields must not be empty" - ); - - foreach ($fields as $field) { - try { - Utils::assertValidName($field->name); - } catch (InvariantViolation $e) { - throw new InvariantViolation("{$this->name}.{$field->name}: {$e->getMessage()}"); - } - - $fieldType = $field->type; - if ($fieldType instanceof WrappingType) { - $fieldType = $fieldType->getWrappedType(true); - } - Utils::invariant( - $fieldType instanceof InputType, - "{$this->name}.{$field->name} field type must be Input Type but got: %s.", - Utils::printSafe($field->type) - ); - Utils::invariant( - !isset($field->config['resolve']), - "{$this->name}.{$field->name} field type has a resolve property, but Input Types cannot define resolvers." - ); - } - } } diff --git a/src/Type/Definition/InputType.php b/src/Type/Definition/InputType.php index 7f90039..b2c3830 100644 --- a/src/Type/Definition/InputType.php +++ b/src/Type/Definition/InputType.php @@ -3,11 +3,16 @@ namespace GraphQL\Type\Definition; /* export type GraphQLInputType = - GraphQLScalarType | - GraphQLEnumType | - GraphQLInputObjectType | - GraphQLList | - GraphQLNonNull; + | GraphQLScalarType + | GraphQLEnumType + | GraphQLInputObjectType + | GraphQLList + | GraphQLNonNull< + | GraphQLScalarType + | GraphQLEnumType + | GraphQLInputObjectType + | GraphQLList, + >; */ interface InputType { diff --git a/src/Type/Definition/InterfaceType.php b/src/Type/Definition/InterfaceType.php index 33bcb67..3d57f88 100644 --- a/src/Type/Definition/InterfaceType.php +++ b/src/Type/Definition/InterfaceType.php @@ -10,7 +10,7 @@ use GraphQL\Utils\Utils; * Class InterfaceType * @package GraphQL\Type\Definition */ -class InterfaceType extends Type implements AbstractType, OutputType, CompositeType +class InterfaceType extends Type implements AbstractType, OutputType, CompositeType, NamedType { /** * @param mixed $type @@ -51,7 +51,7 @@ class InterfaceType extends Type implements AbstractType, OutputType, CompositeT $config['name'] = $this->tryInferName(); } - Utils::assertValidName($config['name']); + Utils::invariant(is_string($config['name']), 'Must provide name.'); Config::validate($config, [ 'name' => Config::NAME, @@ -120,23 +120,9 @@ class InterfaceType extends Type implements AbstractType, OutputType, CompositeT { parent::assertValid(); - $fields = $this->getFields(); - Utils::invariant( !isset($this->config['resolveType']) || is_callable($this->config['resolveType']), "{$this->name} must provide \"resolveType\" as a function." ); - - Utils::invariant( - !empty($fields), - "{$this->name} fields must not be empty" - ); - - foreach ($fields as $field) { - $field->assertValid($this); - foreach ($field->args as $arg) { - $arg->assertValid($field, $this); - } - } } } diff --git a/src/Type/Definition/NamedType.php b/src/Type/Definition/NamedType.php new file mode 100644 index 0000000..be9681a --- /dev/null +++ b/src/Type/Definition/NamedType.php @@ -0,0 +1,15 @@ +tryInferName(); } - Utils::assertValidName($config['name'], !empty($config['isIntrospection'])); + Utils::invariant(is_string($config['name']), 'Must provide name.'); // Note: this validation is disabled by default, because it is resource-consuming // TODO: add bin/validate script to check if schema is valid during development @@ -228,18 +228,5 @@ class ObjectType extends Type implements OutputType, CompositeType !isset($this->config['isTypeOf']) || is_callable($this->config['isTypeOf']), "{$this->name} must provide 'isTypeOf' as a function" ); - - // getFields() and getInterfaceMap() will do structural validation - $fields = $this->getFields(); - Utils::invariant( - !empty($fields), - "{$this->name} fields must not be empty" - ); - foreach ($fields as $field) { - $field->assertValid($this); - foreach ($field->args as $arg) { - $arg->assertValid($field, $this); - } - } } } diff --git a/src/Type/Definition/ScalarType.php b/src/Type/Definition/ScalarType.php index 6038796..eee6bb0 100644 --- a/src/Type/Definition/ScalarType.php +++ b/src/Type/Definition/ScalarType.php @@ -22,7 +22,7 @@ use GraphQL\Utils\Utils; * } * } */ -abstract class ScalarType extends Type implements OutputType, InputType, LeafType +abstract class ScalarType extends Type implements OutputType, InputType, LeafType, NamedType { /** * @var ScalarTypeDefinitionNode|null @@ -36,6 +36,6 @@ abstract class ScalarType extends Type implements OutputType, InputType, LeafTyp $this->astNode = isset($config['astNode']) ? $config['astNode'] : null; $this->config = $config; - Utils::assertValidName($this->name); + Utils::invariant(is_string($this->name), 'Must provide name.'); } } diff --git a/src/Type/Definition/Type.php b/src/Type/Definition/Type.php index ba797bc..5dd6b93 100644 --- a/src/Type/Definition/Type.php +++ b/src/Type/Definition/Type.php @@ -80,7 +80,7 @@ abstract class Type implements \JsonSerializable /** * @api - * @param ObjectType|InterfaceType|UnionType|ScalarType|InputObjectType|EnumType|ListOfType|NonNull $wrappedType + * @param Type|ObjectType|InterfaceType|UnionType|ScalarType|InputObjectType|EnumType|ListOfType|NonNull $wrappedType * @return ListOfType */ public static function listOf($wrappedType) @@ -161,8 +161,11 @@ abstract class Type implements \JsonSerializable */ public static function isInputType($type) { - $nakedType = self::getNamedType($type); - return $nakedType instanceof InputType; + return $type instanceof InputType && + ( + !$type instanceof WrappingType || + self::getNamedType($type) instanceof InputType + ); } /** @@ -172,8 +175,11 @@ abstract class Type implements \JsonSerializable */ public static function isOutputType($type) { - $nakedType = self::getNamedType($type); - return $nakedType instanceof OutputType; + return $type instanceof OutputType && + ( + !$type instanceof WrappingType || + self::getNamedType($type) instanceof OutputType + ); } /** @@ -311,6 +317,7 @@ abstract class Type implements \JsonSerializable */ public function assertValid() { + Utils::assertValidName($this->name); } /** diff --git a/src/Type/Definition/UnionType.php b/src/Type/Definition/UnionType.php index 06d57fc..f9f4863 100644 --- a/src/Type/Definition/UnionType.php +++ b/src/Type/Definition/UnionType.php @@ -9,7 +9,7 @@ use GraphQL\Utils\Utils; * Class UnionType * @package GraphQL\Type\Definition */ -class UnionType extends Type implements AbstractType, OutputType, CompositeType +class UnionType extends Type implements AbstractType, OutputType, CompositeType, NamedType { /** * @var UnionTypeDefinitionNode @@ -36,7 +36,7 @@ class UnionType extends Type implements AbstractType, OutputType, CompositeType $config['name'] = $this->tryInferName(); } - Utils::assertValidName($config['name']); + Utils::invariant(is_string($config['name']), 'Must provide name.'); Config::validate($config, [ 'name' => Config::NAME | Config::REQUIRED, @@ -81,7 +81,8 @@ class UnionType extends Type implements AbstractType, OutputType, CompositeType if (!is_array($types)) { throw new InvariantViolation( - "{$this->name} types must be an Array or a callable which returns an Array." + "Must provide Array of types or a callable which returns " . + "such an array for Union {$this->name}" ); } @@ -133,31 +134,11 @@ class UnionType extends Type implements AbstractType, OutputType, CompositeType { parent::assertValid(); - $types = $this->getTypes(); - Utils::invariant( - !empty($types), - "{$this->name} types must not be empty" - ); - if (isset($this->config['resolveType'])) { Utils::invariant( is_callable($this->config['resolveType']), "{$this->name} must provide \"resolveType\" as a function." ); } - - $includedTypeNames = []; - foreach ($types as $objType) { - Utils::invariant( - $objType instanceof ObjectType, - "{$this->name} may only contain Object types, it cannot contain: %s.", - Utils::printSafe($objType) - ); - Utils::invariant( - !isset($includedTypeNames[$objType->name]), - "{$this->name} can include {$objType->name} type only once." - ); - $includedTypeNames[$objType->name] = true; - } } } diff --git a/src/Type/Introspection.php b/src/Type/Introspection.php index 57be002..4fd41ca 100644 --- a/src/Type/Introspection.php +++ b/src/Type/Introspection.php @@ -168,7 +168,7 @@ EOD; * @param Type $type * @return bool */ - public static function isIntrospectionType(Type $type) + public static function isIntrospectionType($type) { return in_array($type->name, array_keys(self::getTypes())); } diff --git a/src/Type/SchemaValidationContext.php b/src/Type/SchemaValidationContext.php index a0a4312..7b89871 100644 --- a/src/Type/SchemaValidationContext.php +++ b/src/Type/SchemaValidationContext.php @@ -2,9 +2,11 @@ namespace GraphQL\Type; use GraphQL\Error\Error; -use GraphQL\Error\InvariantViolation; +use GraphQL\Language\AST\EnumValueDefinitionNode; use GraphQL\Language\AST\FieldDefinitionNode; use GraphQL\Language\AST\InputValueDefinitionNode; +use GraphQL\Language\AST\InterfaceTypeDefinitionNode; +use GraphQL\Language\AST\InterfaceTypeExtensionNode; use GraphQL\Language\AST\NamedTypeNode; use GraphQL\Language\AST\Node; use GraphQL\Language\AST\ObjectTypeDefinitionNode; @@ -13,20 +15,24 @@ use GraphQL\Language\AST\SchemaDefinitionNode; use GraphQL\Language\AST\TypeDefinitionNode; use GraphQL\Language\AST\TypeNode; use GraphQL\Type\Definition\Directive; +use GraphQL\Type\Definition\EnumType; +use GraphQL\Type\Definition\EnumValueDefinition; +use GraphQL\Type\Definition\FieldDefinition; +use GraphQL\Type\Definition\InputObjectField; +use GraphQL\Type\Definition\InputObjectType; use GraphQL\Type\Definition\InterfaceType; +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\Utils\TypeComparators; use GraphQL\Utils\Utils; -/** - * - */ class SchemaValidationContext { /** - * @var array + * @var Error[] */ private $errors = []; @@ -56,7 +62,7 @@ class SchemaValidationContext ); } else if (!$queryType instanceof ObjectType) { $this->reportError( - 'Query root type must be Object type but got: ' . Utils::getVariableType($queryType) . '.', + 'Query root type must be Object type, it cannot be ' . Utils::printSafe($queryType) . '.', $this->getOperationTypeNode($queryType, 'query') ); } @@ -64,7 +70,7 @@ class SchemaValidationContext $mutationType = $this->schema->getMutationType(); if ($mutationType && !$mutationType instanceof ObjectType) { $this->reportError( - 'Mutation root type must be Object type if provided but got: ' . Utils::getVariableType($mutationType) . '.', + 'Mutation root type must be Object type if provided, it cannot be ' . Utils::printSafe($mutationType) . '.', $this->getOperationTypeNode($mutationType, 'mutation') ); } @@ -72,282 +78,12 @@ class SchemaValidationContext $subscriptionType = $this->schema->getSubscriptionType(); if ($subscriptionType && !$subscriptionType instanceof ObjectType) { $this->reportError( - 'Subscription root type must be Object type if provided but got: ' . Utils::getVariableType($subscriptionType) . '.', + 'Subscription root type must be Object type if provided, it cannot be ' . Utils::printSafe($subscriptionType) . '.', $this->getOperationTypeNode($subscriptionType, 'subscription') ); } } - public function validateDirectives() - { - $directives = $this->schema->getDirectives(); - foreach($directives as $directive) { - if (!$directive instanceof Directive) { - $this->reportError( - "Expected directive but got: " . $directive, - is_object($directive) ? $directive->astNode : null - ); - } - } - } - - public function validateTypes() - { - $typeMap = $this->schema->getTypeMap(); - foreach($typeMap as $typeName => $type) { - // Ensure all provided types are in fact GraphQL type. - if (!Type::isType($type)) { - $this->reportError( - "Expected GraphQL type but got: " . Utils::getVariableType($type), - is_object($type) ? $type->astNode : null - ); - } - - // Ensure objects implement the interfaces they claim to. - if ($type instanceof ObjectType) { - $implementedTypeNames = []; - - foreach($type->getInterfaces() as $iface) { - if (isset($implementedTypeNames[$iface->name])) { - $this->reportError( - "{$type->name} must declare it implements {$iface->name} only once.", - $this->getAllImplementsInterfaceNode($type, $iface) - ); - } - $implementedTypeNames[$iface->name] = true; - $this->validateObjectImplementsInterface($type, $iface); - } - } - } - } - - /** - * @param ObjectType $object - * @param InterfaceType $iface - */ - private function validateObjectImplementsInterface(ObjectType $object, $iface) - { - if (!$iface instanceof InterfaceType) { - $this->reportError( - $object . - " must only implement Interface types, it cannot implement " . - $iface . ".", - $this->getImplementsInterfaceNode($object, $iface) - ); - return; - } - - $objectFieldMap = $object->getFields(); - $ifaceFieldMap = $iface->getFields(); - - // Assert each interface field is implemented. - foreach ($ifaceFieldMap as $fieldName => $ifaceField) { - $objectField = array_key_exists($fieldName, $objectFieldMap) - ? $objectFieldMap[$fieldName] - : null; - - // Assert interface field exists on object. - if (!$objectField) { - $this->reportError( - "\"{$iface->name}\" expects field \"{$fieldName}\" but \"{$object->name}\" does not provide it.", - [$this->getFieldNode($iface, $fieldName), $object->astNode] - ); - continue; - } - - // Assert interface field type is satisfied by object field type, by being - // a valid subtype. (covariant) - if ( - !TypeComparators::isTypeSubTypeOf( - $this->schema, - $objectField->getType(), - $ifaceField->getType() - ) - ) { - $this->reportError( - "{$iface->name}.{$fieldName} expects type ". - "\"{$ifaceField->getType()}\"" . - " but {$object->name}.{$fieldName} is type " . - "\"{$objectField->getType()}\".", - [ - $this->getFieldTypeNode($iface, $fieldName), - $this->getFieldTypeNode($object, $fieldName), - ] - ); - } - - // Assert each interface field arg is implemented. - foreach($ifaceField->args as $ifaceArg) { - $argName = $ifaceArg->name; - $objectArg = null; - - foreach($objectField->args as $arg) { - if ($arg->name === $argName) { - $objectArg = $arg; - break; - } - } - - // Assert interface field arg exists on object field. - if (!$objectArg) { - $this->reportError( - "{$iface->name}.{$fieldName} expects argument \"{$argName}\" but ". - "{$object->name}.{$fieldName} does not provide it.", - [ - $this->getFieldArgNode($iface, $fieldName, $argName), - $this->getFieldNode($object, $fieldName), - ] - ); - continue; - } - - // Assert interface field arg type matches object field arg type. - // (invariant) - // TODO: change to contravariant? - if (!TypeComparators::isEqualType($ifaceArg->getType(), $objectArg->getType())) { - $this->reportError( - "{$iface->name}.{$fieldName}({$argName}:) expects type ". - "\"{$ifaceArg->getType()}\"" . - " but {$object->name}.{$fieldName}({$argName}:) is type " . - "\"{$objectArg->getType()}\".", - [ - $this->getFieldArgTypeNode($iface, $fieldName, $argName), - $this->getFieldArgTypeNode($object, $fieldName, $argName), - ] - ); - } - - // TODO: validate default values? - } - - // Assert additional arguments must not be required. - foreach($objectField->args as $objectArg) { - $argName = $objectArg->name; - $ifaceArg = null; - - foreach($ifaceField->args as $arg) { - if ($arg->name === $argName) { - $ifaceArg = $arg; - break; - } - } - - if (!$ifaceArg && $objectArg->getType() instanceof NonNull) { - $this->reportError( - "{$object->name}.{$fieldName}({$argName}:) is of required type " . - "\"{$objectArg->getType()}\"" . - " but is not also provided by the interface {$iface->name}.{$fieldName}.", - [ - $this->getFieldArgTypeNode($object, $fieldName, $argName), - $this->getFieldNode($iface, $fieldName), - ] - ); - } - } - } - } - - /** - * @param ObjectType $type - * @param InterfaceType|null $iface - * @return NamedTypeNode|null - */ - private function getImplementsInterfaceNode(ObjectType $type, $iface) - { - $nodes = $this->getAllImplementsInterfaceNode($type, $iface); - return $nodes && isset($nodes[0]) ? $nodes[0] : null; - } - - /** - * @param ObjectType $type - * @param InterfaceType|null $iface - * @return NamedTypeNode[] - */ - private function getAllImplementsInterfaceNode(ObjectType $type, $iface) - { - $implementsNodes = []; - /** @var ObjectTypeDefinitionNode|ObjectTypeExtensionNode[] $astNodes */ - $astNodes = array_merge([$type->astNode], $type->extensionASTNodes ?: []); - - foreach($astNodes as $astNode) { - if ($astNode && $astNode->interfaces) { - foreach($astNode->interfaces as $node) { - if ($node->name->value === $iface->name) { - $implementsNodes[] = $node; - } - } - } - } - - return $implementsNodes; - } - - /** - * @param ObjectType|InterfaceType $type - * @param string $fieldName - * @return FieldDefinitionNode|null - */ - private function getFieldNode($type, $fieldName) - { - /** @var ObjectTypeDefinitionNode|ObjectTypeExtensionNode[] $astNodes */ - $astNodes = array_merge([$type->astNode], $type->extensionASTNodes ?: []); - - foreach($astNodes as $astNode) { - if ($astNode && $astNode->fields) { - foreach($astNode->fields as $node) { - if ($node->name->value === $fieldName) { - return $node; - } - } - } - } - } - - /** - * @param ObjectType|InterfaceType $type - * @param string $fieldName - * @return TypeNode|null - */ - private function getFieldTypeNode($type, $fieldName) - { - $fieldNode = $this->getFieldNode($type, $fieldName); - if ($fieldNode) { - return $fieldNode->type; - } - } - - /** - * @param ObjectType|InterfaceType $type - * @param string $fieldName - * @param string $argName - * @return InputValueDefinitionNode|null - */ - private function getFieldArgNode($type, $fieldName, $argName) - { - $fieldNode = $this->getFieldNode($type, $fieldName); - if ($fieldNode && $fieldNode->arguments) { - foreach ($fieldNode->arguments as $node) { - if ($node->name->value === $argName) { - return $node; - } - } - } - } - - /** - * @param ObjectType|InterfaceType $type - * @param string $fieldName - * @param string $argName - * @return TypeNode|null - */ - private function getFieldArgTypeNode($type, $fieldName, $argName) - { - $fieldArgNode = $this->getFieldArgNode($type, $fieldName, $argName); - if ($fieldArgNode) { - return $fieldArgNode->type; - } - } - /** * @param Type $type * @param string $operation @@ -373,12 +109,620 @@ class SchemaValidationContext return $operationTypeNode ? $operationTypeNode->type : ($type ? $type->astNode : null); } + public function validateDirectives() + { + $directives = $this->schema->getDirectives(); + foreach($directives as $directive) { + // Ensure all directives are in fact GraphQL directives. + if (!$directive instanceof Directive) { + $this->reportError( + "Expected directive but got: " . Utils::printSafe($directive) . '.', + is_object($directive) ? $directive->astNode : null + ); + continue; + } + + // Ensure they are named correctly. + $this->validateName($directive); + + // TODO: Ensure proper locations. + + $argNames = []; + foreach ($directive->args as $arg) { + $argName = $arg->name; + + // Ensure they are named correctly. + $this->validateName($directive); + + if (isset($argNames[$argName])) { + $this->reportError( + "Argument @{$directive->name}({$argName}:) can only be defined once.", + $this->getAllDirectiveArgNodes($directive, $argName) + ); + continue; + } + + $argNames[$argName] = true; + + // Ensure the type is an input type. + if (!Type::isInputType($arg->getType())) { + $this->reportError( + "The type of @{$directive->name}({$argName}:) must be Input Type " . + 'but got: ' . Utils::printSafe($arg->getType()) . '.', + $this->getDirectiveArgTypeNode($directive, $argName) + ); + } + } + } + } + + /** + * @param Type|Directive|FieldDefinition|EnumValueDefinition|InputObjectField $node + */ + private function validateName($node) + { + // Ensure names are valid, however introspection types opt out. + $error = Utils::isValidNameError($node->name, $node->astNode); + if ($error && !Introspection::isIntrospectionType($node)) { + $this->addError($error); + } + } + + public function validateTypes() + { + $typeMap = $this->schema->getTypeMap(); + foreach($typeMap as $typeName => $type) { + // Ensure all provided types are in fact GraphQL type. + if (!$type instanceof NamedType) { + $this->reportError( + "Expected GraphQL named type but got: " . Utils::printSafe($type) . '.', + is_object($type) ? $type->astNode : null + ); + continue; + } + + $this->validateName($type); + + if ($type instanceof ObjectType) { + // Ensure fields are valid + $this->validateFields($type); + + // Ensure objects implement the interfaces they claim to. + $this->validateObjectInterfaces($type); + } else if ($type instanceof InterfaceType) { + // Ensure fields are valid. + $this->validateFields($type); + } else if ($type instanceof UnionType) { + // Ensure Unions include valid member types. + $this->validateUnionMembers($type); + } else if ($type instanceof EnumType) { + // Ensure Enums have valid values. + $this->validateEnumValues($type); + } else if ($type instanceof InputObjectType) { + // Ensure Input Object fields are valid. + $this->validateInputFields($type); + } + } + } + + /** + * @param ObjectType|InterfaceType $type + */ + private function validateFields($type) { + $fieldMap = $type->getFields(); + + // Objects and Interfaces both must define one or more fields. + if (!$fieldMap) { + $this->reportError( + "Type {$type->name} must define one or more fields.", + $this->getAllObjectOrInterfaceNodes($type) + ); + } + + foreach ($fieldMap as $fieldName => $field) { + // Ensure they are named correctly. + $this->validateName($field); + + // Ensure they were defined at most once. + $fieldNodes = $this->getAllFieldNodes($type, $fieldName); + if ($fieldNodes && count($fieldNodes) > 1) { + $this->reportError( + "Field {$type->name}.{$fieldName} can only be defined once.", + $fieldNodes + ); + continue; + } + + // Ensure the type is an output type + if (!Type::isOutputType($field->getType())) { + $this->reportError( + "The type of {$type->name}.{$fieldName} must be Output Type " . + 'but got: ' . Utils::printSafe($field->getType()) . '.', + $this->getFieldTypeNode($type, $fieldName) + ); + } + + // Ensure the arguments are valid + $argNames = []; + foreach($field->args as $arg) { + $argName = $arg->name; + + // Ensure they are named correctly. + $this->validateName($arg); + + if (isset($argNames[$argName])) { + $this->reportError( + "Field argument {$type->name}.{$fieldName}({$argName}:) can only " . + 'be defined once.', + $this->getAllFieldArgNodes($type, $fieldName, $argName) + ); + } + $argNames[$argName] = true; + + // Ensure the type is an input type + if (!Type::isInputType($arg->getType())) { + $this->reportError( + "The type of {$type->name}.{$fieldName}({$argName}:) must be Input " . + 'Type but got: '. Utils::printSafe($arg->getType()) . '.', + $this->getFieldArgTypeNode($type, $fieldName, $argName) + ); + } + } + } + } + + private function validateObjectInterfaces(ObjectType $object) { + $implementedTypeNames = []; + foreach($object->getInterfaces() as $iface) { + if (isset($implementedTypeNames[$iface->name])) { + $this->reportError( + "Type {$object->name} can only implement {$iface->name} once.", + $this->getAllImplementsInterfaceNodes($object, $iface) + ); + continue; + } + $implementedTypeNames[$iface->name] = true; + $this->validateObjectImplementsInterface($object, $iface); + } + } + + /** + * @param ObjectType $object + * @param InterfaceType $iface + */ + private function validateObjectImplementsInterface(ObjectType $object, $iface) + { + if (!$iface instanceof InterfaceType) { + $this->reportError( + "Type {$object->name} must only implement Interface types, " . + "it cannot implement ". Utils::printSafe($iface) . ".", + $this->getImplementsInterfaceNode($object, $iface) + ); + return; + } + + $objectFieldMap = $object->getFields(); + $ifaceFieldMap = $iface->getFields(); + + // Assert each interface field is implemented. + foreach ($ifaceFieldMap as $fieldName => $ifaceField) { + $objectField = array_key_exists($fieldName, $objectFieldMap) + ? $objectFieldMap[$fieldName] + : null; + + // Assert interface field exists on object. + if (!$objectField) { + $this->reportError( + "Interface field {$iface->name}.{$fieldName} expected but " . + "{$object->name} does not provide it.", + [$this->getFieldNode($iface, $fieldName), $object->astNode] + ); + continue; + } + + // Assert interface field type is satisfied by object field type, by being + // a valid subtype. (covariant) + if ( + !TypeComparators::isTypeSubTypeOf( + $this->schema, + $objectField->getType(), + $ifaceField->getType() + ) + ) { + $this->reportError( + "Interface field {$iface->name}.{$fieldName} expects type ". + "{$ifaceField->getType()} but {$object->name}.{$fieldName} " . + "is type " . Utils::printSafe($objectField->getType()) . ".", + [ + $this->getFieldTypeNode($iface, $fieldName), + $this->getFieldTypeNode($object, $fieldName), + ] + ); + } + + // Assert each interface field arg is implemented. + foreach($ifaceField->args as $ifaceArg) { + $argName = $ifaceArg->name; + $objectArg = null; + + foreach($objectField->args as $arg) { + if ($arg->name === $argName) { + $objectArg = $arg; + break; + } + } + + // Assert interface field arg exists on object field. + if (!$objectArg) { + $this->reportError( + "Interface field argument {$iface->name}.{$fieldName}({$argName}:) " . + "expected but {$object->name}.{$fieldName} does not provide it.", + [ + $this->getFieldArgNode($iface, $fieldName, $argName), + $this->getFieldNode($object, $fieldName), + ] + ); + continue; + } + + // Assert interface field arg type matches object field arg type. + // (invariant) + // TODO: change to contravariant? + if (!TypeComparators::isEqualType($ifaceArg->getType(), $objectArg->getType())) { + $this->reportError( + "Interface field argument {$iface->name}.{$fieldName}({$argName}:) ". + "expects type " . Utils::printSafe($ifaceArg->getType()) . " but " . + "{$object->name}.{$fieldName}({$argName}:) is type " . + Utils::printSafe($objectArg->getType()) . ".", + [ + $this->getFieldArgTypeNode($iface, $fieldName, $argName), + $this->getFieldArgTypeNode($object, $fieldName, $argName), + ] + ); + } + + // TODO: validate default values? + } + + // Assert additional arguments must not be required. + foreach($objectField->args as $objectArg) { + $argName = $objectArg->name; + $ifaceArg = null; + + foreach($ifaceField->args as $arg) { + if ($arg->name === $argName) { + $ifaceArg = $arg; + break; + } + } + + if (!$ifaceArg && $objectArg->getType() instanceof NonNull) { + $this->reportError( + "Object field argument {$object->name}.{$fieldName}({$argName}:) " . + "is of required type " . Utils::printSafe($objectArg->getType()) . " but is not also " . + "provided by the Interface field {$iface->name}.{$fieldName}.", + [ + $this->getFieldArgTypeNode($object, $fieldName, $argName), + $this->getFieldNode($iface, $fieldName), + ] + ); + } + } + } + } + + private function validateUnionMembers(UnionType $union) + { + $memberTypes = $union->getTypes(); + + if (!$memberTypes) { + $this->reportError( + "Union type {$union->name} must define one or more member types.", + $union->astNode + ); + } + + $includedTypeNames = []; + + foreach($memberTypes as $memberType) { + if (isset($includedTypeNames[$memberType->name])) { + $this->reportError( + "Union type {$union->name} can only include type ". + "{$memberType->name} once.", + $this->getUnionMemberTypeNodes($union, $memberType->name) + ); + continue; + } + $includedTypeNames[$memberType->name] = true; + if (!$memberType instanceof ObjectType) { + $this->reportError( + "Union type {$union->name} can only include Object types, ". + "it cannot include " . Utils::printSafe($memberType) . ".", + $this->getUnionMemberTypeNodes($union, Utils::printSafe($memberType)) + ); + } + } + } + + private function validateEnumValues(EnumType $enumType) + { + $enumValues = $enumType->getValues(); + + if (!$enumValues) { + $this->reportError( + "Enum type {$enumType->name} must define one or more values.", + $enumType->astNode + ); + } + + foreach($enumValues as $enumValue) { + $valueName = $enumValue->name; + + // Ensure no duplicates + $allNodes = $this->getEnumValueNodes($enumType, $valueName); + if ($allNodes && count($allNodes) > 1) { + $this->reportError( + "Enum type {$enumType->name} can include value {$valueName} only once.", + $allNodes + ); + } + + // Ensure valid name. + $this->validateName($enumValue); + if ($valueName === 'true' || $valueName === 'false' || $valueName === 'null') { + $this->reportError( + "Enum type {$enumType->name} cannot include value: {$valueName}.", + $enumValue->astNode + ); + } + } + } + + private function validateInputFields(InputObjectType $inputObj) + { + $fieldMap = $inputObj->getFields(); + + if (!$fieldMap) { + $this->reportError( + "Input Object type {$inputObj->name} must define one or more fields.", + $inputObj->astNode + ); + } + + // Ensure the arguments are valid + foreach ($fieldMap as $fieldName => $field) { + // Ensure they are named correctly. + $this->validateName($field); + + // TODO: Ensure they are unique per field. + + // Ensure the type is an input type + if (!Type::isInputType($field->getType())) { + $this->reportError( + "The type of {$inputObj->name}.{$fieldName} must be Input Type " . + "but got: " . Utils::printSafe($field->getType()) . ".", + $field->astNode ? $field->astNode->type : null + ); + } + } + } + + /** + * @param ObjectType|InterfaceType $type + * @return ObjectTypeDefinitionNode[]|ObjectTypeExtensionNode[]|InterfaceTypeDefinitionNode[]|InterfaceTypeExtensionNode[] + */ + private function getAllObjectOrInterfaceNodes($type) + { + return $type->astNode + ? ($type->extensionASTNodes + ? array_merge([$type->astNode], $type->extensionASTNodes) + : [$type->astNode]) + : ($type->extensionASTNodes ?: []); + } + + /** + * @param ObjectType $type + * @param InterfaceType $iface + * @return NamedTypeNode|null + */ + private function getImplementsInterfaceNode(ObjectType $type, $iface) + { + $nodes = $this->getAllImplementsInterfaceNodes($type, $iface); + return $nodes && isset($nodes[0]) ? $nodes[0] : null; + } + + /** + * @param ObjectType $type + * @param InterfaceType $iface + * @return NamedTypeNode[] + */ + private function getAllImplementsInterfaceNodes(ObjectType $type, $iface) + { + $implementsNodes = []; + $astNodes = $this->getAllObjectOrInterfaceNodes($type); + + foreach($astNodes as $astNode) { + if ($astNode && $astNode->interfaces) { + foreach($astNode->interfaces as $node) { + if ($node->name->value === $iface->name) { + $implementsNodes[] = $node; + } + } + } + } + + return $implementsNodes; + } + + /** + * @param ObjectType|InterfaceType $type + * @param string $fieldName + * @return FieldDefinitionNode|null + */ + private function getFieldNode($type, $fieldName) + { + $nodes = $this->getAllFieldNodes($type, $fieldName); + return $nodes && isset($nodes[0]) ? $nodes[0] : null; + } + + /** + * @param ObjectType|InterfaceType $type + * @param string $fieldName + * @return FieldDefinitionNode[] + */ + private function getAllFieldNodes($type, $fieldName) + { + $fieldNodes = []; + $astNodes = $this->getAllObjectOrInterfaceNodes($type); + foreach($astNodes as $astNode) { + if ($astNode && $astNode->fields) { + foreach($astNode->fields as $node) { + if ($node->name->value === $fieldName) { + $fieldNodes[] = $node; + } + } + } + } + return $fieldNodes; + } + + /** + * @param ObjectType|InterfaceType $type + * @param string $fieldName + * @return TypeNode|null + */ + private function getFieldTypeNode($type, $fieldName) + { + $fieldNode = $this->getFieldNode($type, $fieldName); + return $fieldNode ? $fieldNode->type : null; + } + + /** + * @param ObjectType|InterfaceType $type + * @param string $fieldName + * @param string $argName + * @return InputValueDefinitionNode|null + */ + private function getFieldArgNode($type, $fieldName, $argName) + { + $nodes = $this->getAllFieldArgNodes($type, $fieldName, $argName); + return $nodes && isset($nodes[0]) ? $nodes[0] : null; + } + + /** + * @param ObjectType|InterfaceType $type + * @param string $fieldName + * @param string $argName + * @return InputValueDefinitionNode[] + */ + private function getAllFieldArgNodes($type, $fieldName, $argName) + { + $argNodes = []; + $fieldNode = $this->getFieldNode($type, $fieldName); + if ($fieldNode && $fieldNode->arguments) { + foreach ($fieldNode->arguments as $node) { + if ($node->name->value === $argName) { + $argNodes[] = $node; + } + } + } + return $argNodes; + } + + /** + * @param ObjectType|InterfaceType $type + * @param string $fieldName + * @param string $argName + * @return TypeNode|null + */ + private function getFieldArgTypeNode($type, $fieldName, $argName) + { + $fieldArgNode = $this->getFieldArgNode($type, $fieldName, $argName); + return $fieldArgNode ? $fieldArgNode->type : null; + } + + /** + * @param Directive $directive + * @param string $argName + * @return InputValueDefinitionNode[] + */ + private function getAllDirectiveArgNodes(Directive $directive, $argName) + { + $argNodes = []; + $directiveNode = $directive->astNode; + if ($directiveNode && $directiveNode->arguments) { + foreach($directiveNode->arguments as $node) { + if ($node->name->value === $argName) { + $argNodes[] = $node; + } + } + } + + return $argNodes; + } + + /** + * @param Directive $directive + * @param string $argName + * @return TypeNode|null + */ + private function getDirectiveArgTypeNode(Directive $directive, $argName) + { + $argNode = $this->getAllDirectiveArgNodes($directive, $argName)[0]; + return $argNode ? $argNode->type : null; + } + + /** + * @param UnionType $union + * @param string $typeName + * @return NamedTypeNode[] + */ + private function getUnionMemberTypeNodes(UnionType $union, $typeName) + { + if ($union->astNode && $union->astNode->types) { + return array_filter( + $union->astNode->types, + function (NamedTypeNode $value) use ($typeName) { + return $value->name->value === $typeName; + } + ); + } + return $union->astNode ? + $union->astNode->types : null; + } + + /** + * @param EnumType $enum + * @param string $valueName + * @return EnumValueDefinitionNode[] + */ + private function getEnumValueNodes(EnumType $enum, $valueName) + { + if ($enum->astNode && $enum->astNode->values) { + return array_filter( + iterator_to_array($enum->astNode->values), + function (EnumValueDefinitionNode $value) use ($valueName) { + return $value->name->value === $valueName; + } + ); + } + return $enum->astNode ? + $enum->astNode->values : null; + } + /** * @param string $message * @param array|Node|TypeNode|TypeDefinitionNode $nodes */ private function reportError($message, $nodes = null) { $nodes = array_filter($nodes && is_array($nodes) ? $nodes : [$nodes]); - $this->errors[] = new Error($message, $nodes); + $this->addError(new Error($message, $nodes)); + } + + /** + * @param Error $error + */ + private function addError($error) { + $this->errors[] = $error; } } diff --git a/src/Utils/ASTDefinitionBuilder.php b/src/Utils/ASTDefinitionBuilder.php index c5516dd..be90907 100644 --- a/src/Utils/ASTDefinitionBuilder.php +++ b/src/Utils/ASTDefinitionBuilder.php @@ -22,7 +22,6 @@ use GraphQL\Type\Definition\CustomScalarType; use GraphQL\Type\Definition\EnumType; use GraphQL\Type\Definition\InputObjectType; use GraphQL\Type\Definition\InputType; -use GraphQL\Type\Definition\OutputType; use GraphQL\Type\Definition\Directive; use GraphQL\Type\Definition\InterfaceType; use GraphQL\Type\Definition\FieldArgument; @@ -128,53 +127,7 @@ class ASTDefinitionBuilder /** * @param TypeNode $typeNode - * @return InputType|Type - * @throws Error - */ - public function buildInputType(TypeNode $typeNode) - { - $type = $this->internalBuildWrappedType($typeNode); - Utils::invariant(Type::isInputType($type), 'Expected Input type.'); - return $type; - } - - /** - * @param TypeNode $typeNode - * @return OutputType|Type - * @throws Error - */ - public function buildOutputType(TypeNode $typeNode) - { - $type = $this->internalBuildWrappedType($typeNode); - Utils::invariant(Type::isOutputType($type), 'Expected Output type.'); - return $type; - } - - /** - * @param TypeNode|string $typeNode - * @return ObjectType|Type - * @throws Error - */ - public function buildObjectType($typeNode) - { - $type = $this->buildType($typeNode); - return ObjectType::assertObjectType($type); - } - - /** - * @param TypeNode|string $typeNode - * @return InterfaceType|Type - * @throws Error - */ - public function buildInterfaceType($typeNode) - { - $type = $this->buildType($typeNode); - return InterfaceType::assertInterfaceType($type); - } - - /** - * @param TypeNode $typeNode - * @return Type + * @return Type|InputType * @throws Error */ private function internalBuildWrappedType(TypeNode $typeNode) @@ -199,7 +152,10 @@ class ASTDefinitionBuilder public function buildField(FieldDefinitionNode $field) { return [ - 'type' => $this->buildOutputType($field->type), + // Note: While this could make assertions to get the correctly typed + // value, that would throw immediately while type system validation + // with validateSchema() will produce more actionable results. + 'type' => $this->internalBuildWrappedType($field->type), 'description' => $this->getDescription($field), 'args' => $field->arguments ? $this->makeInputValues($field->arguments) : null, 'deprecationReason' => $this->getDeprecationReason($field), @@ -282,7 +238,10 @@ class ASTDefinitionBuilder return $value->name->value; }, function ($value) { - $type = $this->buildInputType($value->type); + // Note: While this could make assertions to get the correctly typed + // value, that would throw immediately while type system validation + // with validateSchema() will produce more actionable results. + $type = $this->internalBuildWrappedType($value->type); $config = [ 'name' => $value->name->value, 'type' => $type, @@ -339,9 +298,12 @@ class ASTDefinitionBuilder return new UnionType([ 'name' => $def->name->value, 'description' => $this->getDescription($def), + // Note: While this could make assertions to get the correctly typed + // values below, that would throw immediately while type system + // validation with validateSchema() will produce more actionable results. 'types' => $def->types ? Utils::map($def->types, function ($typeNode) { - return $this->buildObjectType($typeNode); + return $this->buildType($typeNode); }): [], 'astNode' => $def, @@ -409,7 +371,7 @@ class ASTDefinitionBuilder { $loc = $node->loc; if (!$loc || !$loc->startToken) { - return; + return null; } $comments = []; $token = $loc->startToken->prev; diff --git a/src/Utils/SchemaPrinter.php b/src/Utils/SchemaPrinter.php index e0a74b7..9685554 100644 --- a/src/Utils/SchemaPrinter.php +++ b/src/Utils/SchemaPrinter.php @@ -221,7 +221,7 @@ class SchemaPrinter private static function printArgs($options, $args, $indentation = '') { - if (count($args) === 0) { + if (!$args) { return ''; } diff --git a/src/Utils/Utils.php b/src/Utils/Utils.php index 0b48c41..c000c80 100644 --- a/src/Utils/Utils.php +++ b/src/Utils/Utils.php @@ -1,8 +1,10 @@ toString(); } if (is_object($var)) { - return 'instance of ' . get_class($var); + if (method_exists($var, '__toString')) { + return (string) $var; + } else { + return 'instance of ' . get_class($var); + } } if (is_array($var)) { return json_encode($var); @@ -399,34 +406,46 @@ class Utils } /** + * Upholds the spec rules about naming. + * * @param $name - * @param bool $isIntrospection - * @throws InvariantViolation + * @throws Error */ - public static function assertValidName($name, $isIntrospection = false) + public static function assertValidName($name) { - $regex = '/^[_a-zA-Z][_a-zA-Z0-9]*$/'; + $error = self::isValidNameError($name); + if ($error) { + throw $error; + } + } - if (!$name || !is_string($name)) { - throw new InvariantViolation( - "Must be named. Unexpected name: " . self::printSafe($name) + /** + * Returns an Error if a name is invalid. + * + * @param string $name + * @param Node|null $node + * @return Error|null + */ + public static function isValidNameError($name, $node = null) + { + Utils::invariant(is_string($name), 'Expected string'); + + if (isset($name[1]) && $name[0] === '_' && $name[1] === '_') { + return new Error( + "Name \"{$name}\" must not begin with \"__\", which is reserved by " . + "GraphQL introspection.", + $node ); } - if (!$isIntrospection && isset($name[1]) && $name[0] === '_' && $name[1] === '_') { - Warning::warnOnce( - 'Name "'.$name.'" must not begin with "__", which is reserved by ' . - 'GraphQL introspection. In a future release of graphql this will ' . - 'become an exception', - Warning::WARNING_NAME + if (!preg_match('/^[_a-zA-Z][_a-zA-Z0-9]*$/', $name)) { + return new Error( + "Names must match /^[_a-zA-Z][_a-zA-Z0-9]*\$/ but \"{$name}\" does not.", + $node ); } - if (!preg_match($regex, $name)) { - throw new InvariantViolation( - 'Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but "'.$name.'" does not.' - ); - } + return null; } /** diff --git a/tests/Type/DefinitionTest.php b/tests/Type/DefinitionTest.php index 9f1b8ce..9457d17 100644 --- a/tests/Type/DefinitionTest.php +++ b/tests/Type/DefinitionTest.php @@ -468,37 +468,6 @@ class DefinitionTest extends \PHPUnit_Framework_TestCase } } - /** - * @it prohibits putting non-Object types in unions - */ - public function testProhibitsPuttingNonObjectTypesInUnions() - { - $int = Type::int(); - - $badUnionTypes = [ - $int, - new NonNull($int), - new ListOfType($int), - $this->interfaceType, - $this->unionType, - $this->enumType, - $this->inputObjectType - ]; - - foreach ($badUnionTypes as $type) { - try { - $union = new UnionType(['name' => 'BadUnion', 'types' => [$type]]); - $union->assertValid(); - $this->fail('Expected exception not thrown'); - } catch (\Exception $e) { - $this->assertSame( - 'BadUnion may only contain Object types, it cannot contain: ' . Utils::printSafe($type) . '.', - $e->getMessage() - ); - } - } - } - /** * @it allows a thunk for Union\'s types */ diff --git a/tests/Type/ValidationTest.php b/tests/Type/ValidationTest.php index 1409da4..361607a 100644 --- a/tests/Type/ValidationTest.php +++ b/tests/Type/ValidationTest.php @@ -21,8 +21,6 @@ class ValidationTest extends \PHPUnit_Framework_TestCase public $SomeObjectType; - public $ObjectWithIsTypeOf; - public $SomeUnionType; public $SomeInterfaceType; @@ -39,11 +37,11 @@ class ValidationTest extends \PHPUnit_Framework_TestCase public $notInputTypes; - public $String; + public $Number; public function setUp() { - $this->String = 'TestString'; + $this->Number = 1; $this->SomeScalarType = new CustomScalarType([ 'name' => 'SomeScalar', @@ -57,11 +55,6 @@ class ValidationTest extends \PHPUnit_Framework_TestCase 'fields' => [ 'f' => [ 'type' => Type::string() ] ], 'interfaces' => function() {return [$this->SomeInterfaceType];} ]); - - $this->ObjectWithIsTypeOf = new ObjectType([ - 'name' => 'ObjectWithIsTypeOf', - 'fields' => [ 'f' => [ 'type' => Type::string() ]] - ]); $this->SomeUnionType = new UnionType([ 'name' => 'SomeUnion', 'types' => [ $this->SomeObjectType ] @@ -98,7 +91,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->notOutputTypes = $this->withModifiers([ $this->SomeInputObjectType, ]); - $this->notOutputTypes[] = $this->String; + $this->notOutputTypes[] = $this->Number; $this->inputTypes = $this->withModifiers([ Type::string(), @@ -113,7 +106,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->SomeInterfaceType, ]); - $this->notInputTypes[] = $this->String; + $this->notInputTypes[] = $this->Number; Warning::suppress(Warning::WARNING_NOT_A_TYPE); } @@ -126,29 +119,29 @@ class ValidationTest extends \PHPUnit_Framework_TestCase /** * @param InvariantViolation[]|Error[] $array - * @param string $message - * @param array|null $locations + * @param array $messages */ - private function assertContainsValidationMessage($array, $message, array $locations = null) { - foreach ($array as $error) { - if ($error->getMessage() === $message) { - if ($error instanceof Error) { - $errorLocations = []; - foreach ($error->getLocations() as $location) { - $errorLocations[] = $location->toArray(); - } - $this->assertEquals($locations, $errorLocations ?: null); - } - return; - } - } - - $this->fail( - 'Failed asserting that the array of validation messages contains ' . - 'the message "' . $message . '"' . "\n" . - 'Found the following messages in the array:' . "\n" . - join("\n", array_map(function($error) { return "\"{$error->getMessage()}\""; }, $array)) + private function assertContainsValidationMessage($array, $messages) { + $this->assertCount( + count($messages), + $array, + 'For messages: ' . $messages[0]['message'] . "\n" . + "Received: \n" . join("\n", array_map(function($error) { return $error->getMessage(); }, $array)) ); + foreach ($array as $index => $error) { + if(!isset($messages[$index]) || !$error instanceof Error) { + $this->fail('Received unexpected error: ' . $error->getMessage()); + } + $this->assertEquals($messages[$index]['message'], $error->getMessage()); + $errorLocations = []; + foreach ($error->getLocations() as $location) { + $errorLocations[] = $location->toArray(); + } + $this->assertEquals( + isset($messages[$index]['locations']) ? $messages[$index]['locations'] : [], + $errorLocations + ); + } } public function testRejectsTypesWithoutNames() @@ -169,70 +162,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase function() { return new InterfaceType([]); } - ], 'Must be named. Unexpected name: null'); - } - - public function testRejectsAnObjectTypeWithReservedName() - { - $this->assertWarnsOnce([ - function() { - return new ObjectType([ - 'name' => '__ReservedName', - ]); - }, - function() { - return new EnumType([ - 'name' => '__ReservedName', - ]); - }, - function() { - return new InputObjectType([ - 'name' => '__ReservedName', - ]); - }, - function() { - return new UnionType([ - 'name' => '__ReservedName', - 'types' => [new ObjectType(['name' => 'Test'])] - ]); - }, - function() { - return new InterfaceType([ - 'name' => '__ReservedName', - ]); - } - ], 'Name "__ReservedName" must not begin with "__", which is reserved by GraphQL introspection. In a future release of graphql this will become an exception'); - } - - public function testRejectsAnObjectTypeWithInvalidName() - { - $this->assertEachCallableThrows([ - function() { - return new ObjectType([ - 'name' => 'a-b-c', - ]); - }, - function() { - return new EnumType([ - 'name' => 'a-b-c', - ]); - }, - function() { - return new InputObjectType([ - 'name' => 'a-b-c', - ]); - }, - function() { - return new UnionType([ - 'name' => 'a-b-c', - ]); - }, - function() { - return new InterfaceType([ - 'name' => 'a-b-c', - ]); - } - ], 'Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but "a-b-c" does not.'); + ], 'Must provide name.'); } // DESCRIBE: Type System: A Schema must have Object root types @@ -339,7 +269,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'Query root type must be provided.' + [['message' => 'Query root type must be provided.']] ); @@ -355,8 +285,10 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schemaWithDef->validate(), - 'Query root type must be provided.', - [['line' => 2, 'column' => 7]] + [[ + 'message' => 'Query root type must be provided.', + 'locations' => [['line' => 2, 'column' => 7]], + ]] ); } @@ -373,8 +305,10 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'Query root type must be Object type but got: Query.', - [['line' => 2, 'column' => 7]] + [[ + 'message' => 'Query root type must be Object type, it cannot be Query.', + 'locations' => [['line' => 2, 'column' => 7]], + ]] ); @@ -390,8 +324,10 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schemaWithDef->validate(), - 'Query root type must be Object type but got: SomeInputObject.', - [['line' => 3, 'column' => 16]] + [[ + 'message' => 'Query root type must be Object type, it cannot be SomeInputObject.', + 'locations' => [['line' => 3, 'column' => 16]], + ]] ); } @@ -412,11 +348,12 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'Mutation root type must be Object type if provided but got: Mutation.', - [['line' => 6, 'column' => 7]] + [[ + 'message' => 'Mutation root type must be Object type if provided, it cannot be Mutation.', + 'locations' => [['line' => 6, 'column' => 7]], + ]] ); - $schemaWithDef = BuildSchema::build(' schema { query: Query @@ -434,8 +371,10 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schemaWithDef->validate(), - 'Mutation root type must be Object type if provided but got: SomeInputObject.', - [['line' => 4, 'column' => 19]] + [[ + 'message' => 'Mutation root type must be Object type if provided, it cannot be SomeInputObject.', + 'locations' => [['line' => 4, 'column' => 19]], + ]] ); } @@ -456,11 +395,12 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'Subscription root type must be Object type if provided but got: Subscription.', - [['line' => 6, 'column' => 7]] + [[ + 'message' => 'Subscription root type must be Object type if provided, it cannot be Subscription.', + 'locations' => [['line' => 6, 'column' => 7]], + ]] ); - $schemaWithDef = BuildSchema::build(' schema { query: Query @@ -478,8 +418,10 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schemaWithDef->validate(), - 'Subscription root type must be Object type if provided but got: SomeInputObject.', - [['line' => 4, 'column' => 23]] + [[ + 'message' => 'Subscription root type must be Object type if provided, it cannot be SomeInputObject.', + 'locations' => [['line' => 4, 'column' => 23]], + ]] ); @@ -497,113 +439,10 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'Expected directive but got: somedirective' + [['message' => 'Expected directive but got: somedirective.']] ); } - // DESCRIBE: Type System: A Schema must contain uniquely named types - /** - * @it rejects a Schema which redefines a built-in type - */ - public function testRejectsASchemaWhichRedefinesABuiltInType() - { - $FakeString = new CustomScalarType([ - 'name' => 'String', - 'serialize' => function() { - return null; - }, - ]); - - $QueryType = new ObjectType([ - 'name' => 'Query', - 'fields' => [ - 'normal' => [ 'type' => Type::string() ], - 'fake' => [ 'type' => $FakeString ], - ] - ]); - - $this->setExpectedException( - InvariantViolation::class, - 'Schema must contain unique named types but contains multiple types named "String" '. - '(see http://webonyx.github.io/graphql-php/type-system/#type-registry).' - ); - new Schema(['query' => $QueryType]); - } - - /** - * @it rejects a Schema which defines an object type twice - */ - public function testRejectsASchemaWhichDfinesAnObjectTypeTwice() - { - $A = new ObjectType([ - 'name' => 'SameName', - 'fields' => [ 'f' => [ 'type' => Type::string() ]], - ]); - - $B = new ObjectType([ - 'name' => 'SameName', - 'fields' => [ 'f' => [ 'type' => Type::string() ] ], - ]); - - $QueryType = new ObjectType([ - 'name' => 'Query', - 'fields' => [ - 'a' => [ 'type' => $A ], - 'b' => [ 'type' => $B ] - ] - ]); - - $this->setExpectedException( - InvariantViolation::class, - 'Schema must contain unique named types but contains multiple types named "SameName" '. - '(see http://webonyx.github.io/graphql-php/type-system/#type-registry).' - ); - - new Schema([ 'query' => $QueryType ]); - } - - /** - * @it rejects a Schema which have same named objects implementing an interface - */ - public function testRejectsASchemaWhichHaveSameNamedObjectsImplementingAnInterface() - { - $AnotherInterface = new InterfaceType([ - 'name' => 'AnotherInterface', - 'fields' => [ 'f' => [ 'type' => Type::string() ]], - ]); - - $FirstBadObject = new ObjectType([ - 'name' => 'BadObject', - 'interfaces' => [ $AnotherInterface ], - 'fields' => [ 'f' => [ 'type' => Type::string() ]], - ]); - - $SecondBadObject = new ObjectType([ - 'name' => 'BadObject', - 'interfaces' => [ $AnotherInterface ], - 'fields' => [ 'f' => [ 'type' => Type::string() ]], - ]); - - $QueryType = new ObjectType([ - 'name' => 'Query', - 'fields' => [ - 'iface' => [ 'type' => $AnotherInterface ], - ] - ]); - - $this->setExpectedException( - InvariantViolation::class, - 'Schema must contain unique named types but contains multiple types named "BadObject" '. - '(see http://webonyx.github.io/graphql-php/type-system/#type-registry).' - ); - - new Schema([ - 'query' => $QueryType, - 'types' => [ $FirstBadObject, $SecondBadObject ] - ]); - } - - // DESCRIBE: Type System: Objects must have fields /** @@ -611,31 +450,17 @@ class ValidationTest extends \PHPUnit_Framework_TestCase */ public function testAcceptsAnObjectTypeWithFieldsObject() { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => [ - 'f' => [ 'type' => Type::string() ] - ] - ])); + $schema = BuildSchema::build(' + type Query { + field: SomeObject + } - // Should not throw: - $schema->assertValid(); - } + type SomeObject { + field: String + } + '); - /** - * @it accepts an Object type with a field function - */ - public function testAcceptsAnObjectTypeWithAfieldFunction() - { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => function() { - return [ - 'f' => ['type' => Type::string()] - ]; - } - ])); - $schema->assertValid(); + $this->assertEquals([], $schema->validate()); } /** @@ -643,32 +468,45 @@ class ValidationTest extends \PHPUnit_Framework_TestCase */ public function testRejectsAnObjectTypeWithMissingFields() { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject' - ])); + $schema = BuildSchema::build(' + type Query { + test: IncompleteObject + } - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject fields must not be empty' - ); - $schema->assertValid(); - } + type IncompleteObject + '); - /** - * @it rejects an Object type field with undefined config - */ - public function testRejectsAnObjectTypeFieldWithUndefinedConfig() - { - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject.f field config must be an array, but got' + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Type IncompleteObject must define one or more fields.', + 'locations' => [['line' => 6, 'column' => 7]], + ]] + ); + + $manualSchema = $this->schemaWithFieldType( + new ObjectType([ + 'name' => 'IncompleteObject', + 'fields' => [], + ]) + ); + + $this->assertContainsValidationMessage( + $manualSchema->validate(), + [['message' => 'Type IncompleteObject must define one or more fields.']] + ); + + $manualSchema2 = $this->schemaWithFieldType( + new ObjectType([ + 'name' => 'IncompleteObject', + 'fields' => function () { return []; }, + ]) + ); + + $this->assertContainsValidationMessage( + $manualSchema2->validate(), + [['message' => 'Type IncompleteObject must define one or more fields.']] ); - $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => [ - 'f' => null - ] - ])); } /** @@ -676,124 +514,33 @@ class ValidationTest extends \PHPUnit_Framework_TestCase */ public function testRejectsAnObjectTypeWithIncorrectlyNamedFields() { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => [ - 'bad-name-with-dashes' => ['type' => Type::string()] - ] - ])); - - $this->setExpectedException( - InvariantViolation::class + $schema = $this->schemaWithFieldType( + new ObjectType([ + 'name' => 'SomeObject', + 'fields' => [ + 'bad-name-with-dashes' => ['type' => Type::string()] + ], + ]) ); - $schema->assertValid(); - } - - /** - * @it warns about an Object type with reserved named fields - */ - public function testWarnsAboutAnObjectTypeWithReservedNamedFields() - { - $lastMessage = null; - Warning::setWarningHandler(function($message) use (&$lastMessage) { - $lastMessage = $message; - }); - - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => [ - '__notPartOfIntrospection' => ['type' => Type::string()] - ] - ])); - - $schema->assertValid(); - - $this->assertEquals( - 'Name "__notPartOfIntrospection" must not begin with "__", which is reserved by GraphQL introspection. '. - 'In a future release of graphql this will become an exception', - $lastMessage + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but ' . + '"bad-name-with-dashes" does not.', + ]] ); - Warning::setWarningHandler(null); } public function testAcceptsShorthandNotationForFields() { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => [ - 'field' => Type::string() - ] - ])); - $schema->assertValid(); - } - - /** - * @it rejects an Object type with incorrectly typed fields - */ - public function testRejectsAnObjectTypeWithIncorrectlyTypedFields() - { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => [ - 'field' => new \stdClass(['type' => Type::string()]) - ] - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject.field field type must be Output Type but got: instance of stdClass' - ); - $schema->assertValid(); - } - - /** - * @it rejects an Object type with empty fields - */ - public function testRejectsAnObjectTypeWithEmptyFields() - { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => [] - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject fields must not be empty' - ); - $schema->assertValid(); - } - - /** - * @it rejects an Object type with a field function that returns nothing - */ - public function testRejectsAnObjectTypeWithAFieldFunctionThatReturnsNothing() - { - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject fields must be an array or a callable which returns such an array.' - ); - $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => function() {} - ])); - } - - /** - * @it rejects an Object type with a field function that returns empty - */ - public function testRejectsAnObjectTypeWithAFieldFunctionThatReturnsEmpty() - { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => function() { - return []; - } - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject fields must not be empty' + $schema = $this->schemaWithFieldType( + new ObjectType([ + 'name' => 'SomeObject', + 'fields' => [ + 'field' => Type::string() + ] + ]) ); $schema->assertValid(); } @@ -816,7 +563,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase ] ] ])); - $schema->assertValid(); + $this->assertEquals([], $schema->validate()); } /** @@ -837,207 +584,59 @@ class ValidationTest extends \PHPUnit_Framework_TestCase ]); $schema = new Schema(['query' => $QueryType]); - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject.badField(bad-name-with-dashes:) Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but "bad-name-with-dashes" does not.' + $this->assertContainsValidationMessage( + $schema->validate(), + [['message' => 'Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but "bad-name-with-dashes" does not.']] ); - - $schema->assertValid(); } - // DESCRIBE: Type System: Fields args must be objects + // DESCRIBE: Type System: Union types must be valid /** - * @it accepts an Object type with field args - */ - public function testAcceptsAnObjectTypeWithFieldArgs() - { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => [ - 'goodField' => [ - 'type' => Type::string(), - 'args' => [ - 'goodArg' => ['type' => Type::string()] - ] - ] - ] - ])); - $schema->assertValid(); - } - - /** - * @it rejects an Object type with incorrectly typed field args - */ - public function testRejectsAnObjectTypeWithIncorrectlyTypedFieldArgs() - { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'fields' => [ - 'badField' => [ - 'type' => Type::string(), - 'args' => [ - ['badArg' => Type::string()] - ] - ] - ] - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject.badField(0:) Must be named. Unexpected name: 0' - ); - - $schema->assertValid(); - } - - // DESCRIBE: Type System: Object interfaces must be array - - /** - * @it accepts an Object type with array interfaces - */ - public function testAcceptsAnObjectTypeWithArrayInterfaces() - { - $AnotherInterfaceType = new InterfaceType([ - 'name' => 'AnotherInterface', - 'fields' => ['f' => ['type' => Type::string()]] - ]); - - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'interfaces' => [$AnotherInterfaceType], - 'fields' => ['f' => ['type' => Type::string()]] - ])); - $schema->assertValid(); - } - - /** - * @it accepts an Object type with interfaces as a function returning an array - */ - public function testAcceptsAnObjectTypeWithInterfacesAsAFunctionReturningAnArray() - { - $AnotherInterfaceType = new InterfaceType([ - 'name' => 'AnotherInterface', - 'fields' => ['f' => ['type' => Type::string()]] - ]); - - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'interfaces' => function () use ($AnotherInterfaceType) { - return [$AnotherInterfaceType]; - }, - 'fields' => ['f' => ['type' => Type::string()]] - ])); - $schema->assertValid(); - } - - /** - * @it rejects an Object type with incorrectly typed interfaces - */ - public function testRejectsAnObjectTypeWithIncorrectlyTypedInterfaces() - { - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject interfaces must be an Array or a callable which returns an Array.' - ); - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'interfaces' => new \stdClass(), - 'fields' => ['f' => ['type' => Type::string()]] - ])); - $schema->assertValid(); - } - - /** - * @it rejects an Object type with interfaces as a function returning an incorrect type - */ - public function testRejectsAnObjectTypeWithInterfacesAsAFunctionReturningAnIncorrectType() - { - $this->setExpectedException( - InvariantViolation::class, - 'SomeObject interfaces must be an Array or a callable which returns an Array.' - ); - $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'interfaces' => function () { - return new \stdClass(); - }, - 'fields' => ['f' => ['type' => Type::string()]] - ])); - } - - // DESCRIBE: Type System: Union types must be array - - /** - * @it accepts a Union type with array types + * @it accepts a Union type with member types */ public function testAcceptsAUnionTypeWithArrayTypes() { - $schema = $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - 'types' => [$this->SomeObjectType], - ])); - $schema->assertValid(); - } + $schema = BuildSchema::build(' + type Query { + test: GoodUnion + } - /** - * @it accepts a Union type with function returning an array of types - */ - public function testAcceptsAUnionTypeWithFunctionReturningAnArrayOfTypes() - { - $schema = $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - 'types' => function () { - return [$this->SomeObjectType]; - }, - ])); - $schema->assertValid(); - } + type TypeA { + field: String + } - /** - * @it rejects a Union type without types - */ - public function testRejectsAUnionTypeWithoutTypes() - { - $this->setExpectedException( - InvariantViolation::class, - 'SomeUnion types must be an Array or a callable which returns an Array.' - ); - $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - ])); + type TypeB { + field: String + } + + union GoodUnion = + | TypeA + | TypeB + '); + + $this->assertEquals([], $schema->validate()); } /** * @it rejects a Union type with empty types */ - public function testRejectsAUnionTypeWithemptyTypes() + public function testRejectsAUnionTypeWithEmptyTypes() { - $schema = $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - 'types' => [] - ])); + $schema = BuildSchema::build(' + type Query { + test: BadUnion + } - $this->setExpectedException( - InvariantViolation::class, - 'SomeUnion types must not be empty' + union BadUnion + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Union type BadUnion must define one or more member types.', + 'locations' => [['line' => 6, 'column' => 7]], + ]] ); - $schema->assertValid(); - } - - /** - * @it rejects a Union type with incorrectly typed types - */ - public function testRejectsAUnionTypeWithIncorrectlyTypedTypes() - { - $this->setExpectedException( - InvariantViolation::class, - 'SomeUnion types must be an Array or a callable which returns an Array.' - ); - $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - 'types' => $this->SomeObjectType - ])); } /** @@ -1045,19 +644,88 @@ class ValidationTest extends \PHPUnit_Framework_TestCase */ public function testRejectsAUnionTypeWithDuplicatedMemberType() { - $schema = $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - 'types' => [ - $this->SomeObjectType, - $this->SomeObjectType, - ], - ])); + $schema = BuildSchema::build(' + type Query { + test: BadUnion + } - $this->setExpectedException( - InvariantViolation::class, - 'SomeUnion can include SomeObject type only once.' + type TypeA { + field: String + } + + type TypeB { + field: String + } + + union BadUnion = + | TypeA + | TypeB + | TypeA + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Union type BadUnion can only include type TypeA once.', + 'locations' => [['line' => 15, 'column' => 11], ['line' => 17, 'column' => 11]], + ]] ); - $schema->assertValid(); + } + + /** + * @it rejects a Union type with non-Object members types + */ + public function testRejectsAUnionTypeWithNonObjectMembersType() + { + $schema = BuildSchema::build(' + type Query { + test: BadUnion + } + + type TypeA { + field: String + } + + type TypeB { + field: String + } + + union BadUnion = + | TypeA + | String + | TypeB + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Union type BadUnion can only include Object types, ' . + 'it cannot include String.', + 'locations' => [['line' => 16, 'column' => 11]], + ]] + + ); + + $badUnionMemberTypes = [ + Type::string(), + Type::nonNull($this->SomeObjectType), + Type::listOf($this->SomeObjectType), + $this->SomeInterfaceType, + $this->SomeUnionType, + $this->SomeEnumType, + $this->SomeInputObjectType, + ]; + + foreach($badUnionMemberTypes as $memberType) { + $badSchema = $this->schemaWithFieldType( + new UnionType(['name' => 'BadUnion', 'types' => [$memberType]]) + ); + $this->assertContainsValidationMessage( + $badSchema->validate(), + [[ + 'message' => 'Union type BadUnion can only include Object types, ' . + "it cannot include ". Utils::printSafe($memberType) . ".", + ]] + ); + } } // DESCRIBE: Type System: Input Objects must have fields @@ -1067,31 +735,16 @@ class ValidationTest extends \PHPUnit_Framework_TestCase */ public function testAcceptsAnInputObjectTypeWithFields() { - $schema = $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - 'fields' => [ - 'f' => ['type' => Type::string()] - ] - ])); + $schema = BuildSchema::build(' + type Query { + field(arg: SomeInputObject): String + } - $schema->assertValid(); - } - - /** - * @it accepts an Input Object type with a field function - */ - public function testAcceptsAnInputObjectTypeWithAFieldFunction() - { - $schema = $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - 'fields' => function () { - return [ - 'f' => ['type' => Type::string()] - ]; - } - ])); - - $schema->assertValid(); + input SomeInputObject { + field: String + } + '); + $this->assertEquals([], $schema->validate()); } /** @@ -1099,15 +752,20 @@ class ValidationTest extends \PHPUnit_Framework_TestCase */ public function testRejectsAnInputObjectTypeWithMissingFields() { - $schema = $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - ])); + $schema = BuildSchema::build(' + type Query { + field(arg: SomeInputObject): String + } - $this->setExpectedException( - InvariantViolation::class, - 'SomeInputObject fields must not be empty' + input SomeInputObject + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Input Object type SomeInputObject must define one or more fields.', + 'locations' => [['line' => 6, 'column' => 7]], + ]] ); - $schema->assertValid(); } /** @@ -1115,558 +773,80 @@ class ValidationTest extends \PHPUnit_Framework_TestCase */ public function testRejectsAnInputObjectTypeWithIncorrectlyTypedFields() { - $schema = $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - 'fields' => [ - ['field' => Type::string()] - ] - ])); + $schema = BuildSchema::build(' + type Query { + field(arg: SomeInputObject): String + } + + type SomeObject { + field: String + } - $this->setExpectedException( - InvariantViolation::class, - 'SomeInputObject.0: Must be named. Unexpected name: 0' + union SomeUnion = SomeObject + + input SomeInputObject { + badObject: SomeObject + badUnion: SomeUnion + goodInputObject: SomeInputObject + } + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of SomeInputObject.badObject must be Input Type but got: SomeObject.', + 'locations' => [['line' => 13, 'column' => 20]], + ],[ + 'message' => 'The type of SomeInputObject.badUnion must be Input Type but got: SomeUnion.', + 'locations' => [['line' => 14, 'column' => 19]], + ]] ); - $schema->assertValid(); } - /** - * @it rejects an Input Object type with empty fields - */ - public function testRejectsAnInputObjectTypeWithEmptyFields() - { - $this->setExpectedException( - InvariantViolation::class, - 'SomeInputObject fields must be an array or a callable which returns such an array.' - ); - $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - 'fields' => new \stdClass() - ])); - } - - /** - * @it rejects an Input Object type with a field function that returns nothing - */ - public function testRejectsAnInputObjectTypeWithAFieldFunctionThatReturnsNothing() - { - $this->setExpectedException( - InvariantViolation::class, - 'SomeInputObject fields must be an array or a callable which returns such an array.' - ); - $this->schemaWithInputObject(new ObjectType([ - 'name' => 'SomeInputObject', - 'fields' => function () { - } - ])); - } - - /** - * @it rejects an Input Object type with a field function that returns empty - */ - public function testRejectsAnInputObjectTypeWithAFieldFunctionThatReturnsEmpty() - { - $this->setExpectedException( - InvariantViolation::class, - 'SomeInputObject fields must be an array or a callable which returns such an array.' - ); - $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - 'fields' => function () { - return new \stdClass(); - } - ])); - } - - // DESCRIBE: Type System: Input Object fields must not have resolvers - - /** - * @it accepts an Input Object type with no resolver - */ - public function testAcceptsAnInputObjectTypeWithNoResolver() - { - $schema = $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - 'fields' => [ - 'f' => [ - 'type' => Type::string(), - ] - ] - ])); - - $schema->assertValid(); - } - - /** - * @it accepts an Input Object type with null resolver - */ - public function testAcceptsAnInputObjectTypeWithNullResolver() - { - $schema = $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - 'fields' => [ - 'f' => [ - 'type' => Type::string(), - 'resolve' => null, - ] - ] - ])); - $schema->assertValid(); - } - - /** - * @it rejects an Input Object type with resolver function - */ - public function testRejectsAnInputObjectTypeWithResolverFunction() - { - $schema = $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - 'fields' => [ - 'f' => [ - 'type' => Type::string(), - 'resolve' => function () { - return 0; - }, - ] - ] - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeInputObject.f field type has a resolve property, but Input Types cannot define resolvers.' - ); - $schema->assertValid(); - } - - /** - * @it rejects an Input Object type with resolver constant - */ - public function testRejectsAnInputObjectTypeWithResolverConstant() - { - $schema = $this->schemaWithInputObject(new InputObjectType([ - 'name' => 'SomeInputObject', - 'fields' => [ - 'f' => [ - 'type' => Type::string(), - 'resolve' => new \stdClass(), - ] - ] - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeInputObject.f field type has a resolve property, but Input Types cannot define resolvers.' - ); - $schema->assertValid(); - } - - - // DESCRIBE: Type System: Object types must be assertable - - /** - * @it accepts an Object type with an isTypeOf function - */ - public function testAcceptsAnObjectTypeWithAnIsTypeOfFunction() - { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'AnotherObject', - 'isTypeOf' => function () { - return true; - }, - 'fields' => ['f' => ['type' => Type::string()]] - ])); - $schema->assertValid(); - } - - /** - * @it rejects an Object type with an incorrect type for isTypeOf - */ - public function testRejectsAnObjectTypeWithAnIncorrectTypeForIsTypeOf() - { - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'AnotherObject', - 'isTypeOf' => new \stdClass(), - 'fields' => ['f' => ['type' => Type::string()]] - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'AnotherObject must provide \'isTypeOf\' as a function' - ); - - $schema->assertValid(); - } - - // DESCRIBE: Type System: Interface types must be resolvable - - /** - * @it accepts an Interface type defining resolveType - */ - public function testAcceptsAnInterfaceTypeDefiningResolveType() - { - $AnotherInterfaceType = new InterfaceType([ - 'name' => 'AnotherInterface', - 'fields' => ['f' => ['type' => Type::string()]] - ]); - - $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'interfaces' => [$AnotherInterfaceType], - 'fields' => ['f' => ['type' => Type::string()]] - ])); - } - - /** - * @it accepts an Interface with implementing type defining isTypeOf - */ - public function testAcceptsAnInterfaceWithImplementingTypeDefiningIsTypeOf() - { - $InterfaceTypeWithoutResolveType = new InterfaceType([ - 'name' => 'InterfaceTypeWithoutResolveType', - 'fields' => ['f' => ['type' => Type::string()]] - ]); - - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'isTypeOf' => function () { - return true; - }, - 'interfaces' => [$InterfaceTypeWithoutResolveType], - 'fields' => ['f' => ['type' => Type::string()]] - ])); - - $schema->assertValid(); - } - - /** - * @it accepts an Interface type defining resolveType with implementing type defining isTypeOf - */ - public function testAcceptsAnInterfaceTypeDefiningResolveTypeWithImplementingTypeDefiningIsTypeOf() - { - $AnotherInterfaceType = new InterfaceType([ - 'name' => 'AnotherInterface', - 'fields' => ['f' => ['type' => Type::string()]] - ]); - - $schema = $this->schemaWithFieldType(new ObjectType([ - 'name' => 'SomeObject', - 'isTypeOf' => function () { - return true; - }, - 'interfaces' => [$AnotherInterfaceType], - 'fields' => ['f' => ['type' => Type::string()]] - ])); - - $schema->assertValid(); - } - - /** - * @it rejects an Interface type with an incorrect type for resolveType - */ - public function testRejectsAnInterfaceTypeWithAnIncorrectTypeForResolveType() - { - $type = new InterfaceType([ - 'name' => 'AnotherInterface', - 'resolveType' => new \stdClass(), - 'fields' => ['f' => ['type' => Type::string()]] - ]); - - $this->setExpectedException( - InvariantViolation::class, - 'AnotherInterface must provide "resolveType" as a function.' - ); - - $type->assertValid(); - } - - // DESCRIBE: Type System: Union types must be resolvable - - /** - * @it accepts a Union type defining resolveType - */ - public function testAcceptsAUnionTypeDefiningResolveType() - { - $schema = $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - 'types' => [$this->SomeObjectType], - ])); - $schema->assertValid(); - } - - /** - * @it accepts a Union of Object types defining isTypeOf - */ - public function testAcceptsAUnionOfObjectTypesDefiningIsTypeOf() - { - $schema = $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - 'types' => [$this->ObjectWithIsTypeOf], - ])); - - $schema->assertValid(); - } - - /** - * @it accepts a Union type defining resolveType of Object types defining isTypeOf - */ - public function testAcceptsAUnionTypeDefiningResolveTypeOfObjectTypesDefiningIsTypeOf() - { - $schema = $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - 'types' => [$this->ObjectWithIsTypeOf], - ])); - $schema->assertValid(); - } - - /** - * @it rejects a Union type with an incorrect type for resolveType - */ - public function testRejectsAUnionTypeWithAnIncorrectTypeForResolveType() - { - $schema = $this->schemaWithFieldType(new UnionType([ - 'name' => 'SomeUnion', - 'resolveType' => new \stdClass(), - 'types' => [$this->ObjectWithIsTypeOf], - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeUnion must provide "resolveType" as a function.' - ); - - $schema->assertValid(); - } - - // DESCRIBE: Type System: Scalar types must be serializable - - /** - * @it accepts a Scalar type defining serialize - */ - public function testAcceptsAScalarTypeDefiningSerialize() - { - $schema = $this->schemaWithFieldType(new CustomScalarType([ - 'name' => 'SomeScalar', - 'serialize' => function () { - }, - ])); - $schema->assertValid(); - } - - /** - * @it rejects a Scalar type not defining serialize - */ - public function testRejectsAScalarTypeNotDefiningSerialize() - { - $schema = $this->schemaWithFieldType(new CustomScalarType([ - 'name' => 'SomeScalar', - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeScalar must provide "serialize" function. If this custom Scalar is also used as an input type, '. - 'ensure "parseValue" and "parseLiteral" functions are also provided.' - ); - - $schema->assertValid(); - } - - /** - * @it rejects a Scalar type defining serialize with an incorrect type - */ - public function testRejectsAScalarTypeDefiningSerializeWithAnIncorrectType() - { - $schema = $this->schemaWithFieldType(new CustomScalarType([ - 'name' => 'SomeScalar', - 'serialize' => new \stdClass() - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeScalar must provide "serialize" function. If this custom Scalar ' . - 'is also used as an input type, ensure "parseValue" and "parseLiteral" ' . - 'functions are also provided.' - ); - - $schema->assertValid(); - } - - /** - * @it accepts a Scalar type defining parseValue and parseLiteral - */ - public function testAcceptsAScalarTypeDefiningParseValueAndParseLiteral() - { - $schema = $this->schemaWithFieldType(new CustomScalarType([ - 'name' => 'SomeScalar', - 'serialize' => function () { - }, - 'parseValue' => function () { - }, - 'parseLiteral' => function () { - }, - ])); - - $schema->assertValid(); - } - - /** - * @it rejects a Scalar type defining parseValue but not parseLiteral - */ - public function testRejectsAScalarTypeDefiningParseValueButNotParseLiteral() - { - $schema = $this->schemaWithFieldType(new CustomScalarType([ - 'name' => 'SomeScalar', - 'serialize' => function () { - }, - 'parseValue' => function () { - }, - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeScalar must provide both "parseValue" and "parseLiteral" functions.' - ); - $schema->assertValid(); - } - - /** - * @it rejects a Scalar type defining parseLiteral but not parseValue - */ - public function testRejectsAScalarTypeDefiningParseLiteralButNotParseValue() - { - $schema = $this->schemaWithFieldType(new CustomScalarType([ - 'name' => 'SomeScalar', - 'serialize' => function () { - }, - 'parseLiteral' => function () { - }, - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeScalar must provide both "parseValue" and "parseLiteral" functions.' - ); - - $schema->assertValid(); - } - - /** - * @it rejects a Scalar type defining parseValue and parseLiteral with an incorrect type - */ - public function testRejectsAScalarTypeDefiningParseValueAndParseLiteralWithAnIncorrectType() - { - $schema = $this->schemaWithFieldType(new CustomScalarType([ - 'name' => 'SomeScalar', - 'serialize' => function () { - }, - 'parseValue' => new \stdClass(), - 'parseLiteral' => new \stdClass(), - ])); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeScalar must provide both "parseValue" and "parseLiteral" functions.' - ); - - $schema->assertValid(); - } - - // DESCRIBE: Type System: Enum types must be well defined - /** - * @it accepts a well defined Enum type with empty value definition - */ - public function testAcceptsAWellDefinedEnumTypeWithEmptyValueDefinition() - { - $type = new EnumType([ - 'name' => 'SomeEnum', - 'values' => [ - 'FOO' => [], - 'BAR' => [], - ] - ]); - - $type->assertValid(); - } - - // TODO: accepts a well defined Enum type with internal value definition - - /** - * @it accepts a well defined Enum type with internal value definition - */ - public function testAcceptsAWellDefinedEnumTypeWithInternalValueDefinition() - { - $type = new EnumType([ - 'name' => 'SomeEnum', - 'values' => [ - 'FOO' => ['value' => 10], - 'BAR' => ['value' => 20], - ] - ]); - $type->assertValid(); - } - /** * @it rejects an Enum type without values */ public function testRejectsAnEnumTypeWithoutValues() { - $type = new EnumType([ - 'name' => 'SomeEnum', - ]); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeEnum values must be an array.' + $schema = BuildSchema::build(' + type Query { + field: SomeEnum + } + + enum SomeEnum + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Enum type SomeEnum must define one or more values.', + 'locations' => [['line' => 6, 'column' => 7]], + ]] ); - - $type->assertValid(); } /** - * @it rejects an Enum type with empty values + * @it rejects an Enum type with duplicate values */ - public function testRejectsAnEnumTypeWithEmptyValues() + public function testRejectsAnEnumTypeWithDuplicateValues() { - $type = new EnumType([ - 'name' => 'SomeEnum', - 'values' => [] - ]); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeEnum values must be not empty.' + $schema = BuildSchema::build(' + type Query { + field: SomeEnum + } + + enum SomeEnum { + SOME_VALUE + SOME_VALUE + } + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Enum type SomeEnum can include value SOME_VALUE only once.', + 'locations' => [['line' => 7, 'column' => 9], ['line' => 8, 'column' => 9]], + ]] ); - - $type->assertValid(); - } - - /** - * @it rejects an Enum type with incorrectly typed values - */ - public function testRejectsAnEnumTypeWithIncorrectlyTypedValues() - { - $type = new EnumType([ - 'name' => 'SomeEnum', - 'values' => [ - ['FOO' => 10] - ] - ]); - - $this->setExpectedException( - InvariantViolation::class, - 'SomeEnum values must be an array with value names as keys.' - ); - $type->assertValid(); - } - - public function invalidEnumValueName() - { - return [ - ['#value', 'SomeEnum has value with invalid name: #value (Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but "#value" does not.)'], - ['true', 'SomeEnum: "true" can not be used as an Enum value.'], - ['false', 'SomeEnum: "false" can not be used as an Enum value.'], - ['null', 'SomeEnum: "null" can not be used as an Enum value.'], - ]; } public function testDoesNotAllowIsDeprecatedWithoutDeprecationReasonOnEnum() @@ -1684,14 +864,28 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $enum->assertValid(); } - private function enumValue($name) + private function schemaWithEnum($name) { - return new EnumType([ - 'name' => 'SomeEnum', - 'values' => [ - $name => [] - ] - ]); + return $this->schemaWithFieldType( + new EnumType([ + 'name' => 'SomeEnum', + 'values' => [ + $name => [] + ] + ]) + ); + } + + public function invalidEnumValueName() + { + return [ + ['#value', 'Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but "#value" does not.'], + ['1value', 'Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but "1value" does not.'], + ['KEBAB-CASE', 'Names must match /^[_a-zA-Z][_a-zA-Z0-9]*$/ but "KEBAB-CASE" does not.'], + ['false', 'Enum type SomeEnum cannot include value: false.'], + ['true', 'Enum type SomeEnum cannot include value: true.'], + ['null', 'Enum type SomeEnum cannot include value: null.'], + ]; } /** @@ -1700,10 +894,14 @@ class ValidationTest extends \PHPUnit_Framework_TestCase */ public function testRejectsAnEnumTypeWithIncorrectlyNamedValues($name, $expectedMessage) { - $enum = $this->enumValue($name); + $schema = $this->schemaWithEnum($name); - $this->setExpectedException(InvariantViolation::class, $expectedMessage); - $enum->assertValid(); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => $expectedMessage, + ]] + ); } // DESCRIBE: Type System: Object fields must have output types @@ -1715,12 +913,10 @@ class ValidationTest extends \PHPUnit_Framework_TestCase { foreach ($this->outputTypes as $type) { $schema = $this->schemaWithObjectFieldOfType($type); - $schema->assertValid(); + $this->assertEquals([], $schema->validate()); } } - // TODO: rejects an empty Object field type - /** * @it rejects an empty Object field type */ @@ -1728,12 +924,12 @@ class ValidationTest extends \PHPUnit_Framework_TestCase { $schema = $this->schemaWithObjectFieldOfType(null); - $this->setExpectedException( - InvariantViolation::class, - 'BadObject.badField field type must be Output Type but got: null' + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of BadObject.badField must be Output Type but got: null.', + ]] ); - - $schema->assertValid(); } /** @@ -1744,96 +940,123 @@ class ValidationTest extends \PHPUnit_Framework_TestCase foreach ($this->notOutputTypes as $type) { $schema = $this->schemaWithObjectFieldOfType($type); - try { - $schema->assertValid(); - $this->fail('Expected exception not thrown for ' . Utils::printSafe($type)); - } catch (InvariantViolation $e) { - $this->assertEquals( - 'BadObject.badField field type must be Output Type but got: ' . Utils::printSafe($type), - $e->getMessage() - ); - } + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of BadObject.badField must be Output Type but got: ' . Utils::printSafe($type) . '.', + ]] + ); } } - // DESCRIBE: Type System: Object fields must have valid resolve values - /** - * @it accepts a lambda as an Object field resolver + * @it rejects with relevant locations for a non-output type as an Object field type */ - public function testAcceptsALambdaAsAnObjectFieldResolver() + public function testRejectsWithReleventLocationsForANonOutputTypeAsAnObjectFieldType() { - $schema = $this->schemaWithObjectWithFieldResolver(function() {return [];}); - $schema->assertValid(); - } - - /** - * @it rejects an empty Object field resolver - */ - public function testRejectsAnEmptyObjectFieldResolver() - { - $schema = $this->schemaWithObjectWithFieldResolver([]); - - $this->setExpectedException( - InvariantViolation::class, - 'BadResolver.badField field resolver must be a function if provided, but got: []' + $schema = BuildSchema::build(' + type Query { + field: [SomeInputObject] + } + + input SomeInputObject { + field: String + } + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of Query.field must be Output Type but got: [SomeInputObject].', + 'locations' => [['line' => 3, 'column' => 16]], + ]] ); - - $schema->assertValid(); } + // DESCRIBE: Type System: Objects can only implement unique interfaces + /** - * @it rejects a constant scalar value resolver + * @it rejects an Object implementing a non-Interface type */ - public function testRejectsAConstantScalarValueResolver() + public function testRejectsAnObjectImplementingANonInterfaceType() { - $schema = $this->schemaWithObjectWithFieldResolver(0); - $this->setExpectedException( - InvariantViolation::class, - 'BadResolver.badField field resolver must be a function if provided, but got: 0' + $schema = BuildSchema::build(' + type Query { + field: BadObject + } + + input SomeInputObject { + field: String + } + + type BadObject implements SomeInputObject { + field: String + } + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Type BadObject must only implement Interface types, it cannot implement SomeInputObject.', + 'locations' => [['line' => 10, 'column' => 33]], + ]] ); - $schema->assertValid(); - } - - // DESCRIBE: Type System: Unions must represent Object types - - /** - * @it accepts a Union of an Object Type - */ - public function testAcceptsAUnionOfAnObjectType() - { - $schema = $this->schemaWithUnionOfType($this->SomeObjectType); - $schema->assertValid(); } /** - * @it rejects a Union of a non-Object type + * @it rejects an Object implementing the same interface twice */ - public function testRejectsAUnionOfANonObjectType() + public function testRejectsAnObjectImplementingTheSameInterfaceTwice() { - $notObjectTypes = $this->withModifiers([ - $this->SomeScalarType, - $this->SomeEnumType, - $this->SomeInterfaceType, - $this->SomeUnionType, - $this->SomeInputObjectType, - ]); - foreach ($notObjectTypes as $type) { - $schema = $this->schemaWithUnionOfType($type); - try { - $schema->assertValid(); - $this->fail('Expected exception not thrown for type: ' . $type); - } catch (InvariantViolation $e) { - $this->assertEquals( - 'BadUnion may only contain Object types, it cannot contain: ' . $type . '.', - $e->getMessage() - ); - } - } - - // "BadUnion may only contain Object types, it cannot contain: $type." + $schema = BuildSchema::build(' + type Query { + field: AnotherObject + } + + interface AnotherInterface { + field: String + } + + type AnotherObject implements AnotherInterface, AnotherInterface { + field: String + } + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Type AnotherObject can only implement AnotherInterface once.', + 'locations' => [['line' => 10, 'column' => 37], ['line' => 10, 'column' => 55]], + ]] + ); } + /** + * @it rejects an Object implementing the same interface twice due to extension + */ + public function testRejectsAnObjectImplementingTheSameInterfaceTwiceDueToExtension() + { + $this->markTestIncomplete('extend does not work this way (yet).'); + $schema = BuildSchema::build(' + type Query { + field: AnotherObject + } + + interface AnotherInterface { + field: String + } + + type AnotherObject implements AnotherInterface { + field: String + } + + extend type AnotherObject implements AnotherInterface + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'Type AnotherObject can only implement AnotherInterface once.', + 'locations' => [['line' => 10, 'column' => 37], ['line' => 14, 'column' => 38]], + ]] + ); + } // DESCRIBE: Type System: Interface fields must have output types @@ -1844,7 +1067,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase { foreach ($this->outputTypes as $type) { $schema = $this->schemaWithInterfaceFieldOfType($type); - $schema->assertValid(); + $this->assertEquals([], $schema->validate()); } } @@ -1854,13 +1077,12 @@ class ValidationTest extends \PHPUnit_Framework_TestCase public function testRejectsAnEmptyInterfaceFieldType() { $schema = $this->schemaWithInterfaceFieldOfType(null); - - $this->setExpectedException( - InvariantViolation::class, - 'BadInterface.badField field type must be Output Type but got: null' + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of BadInterface.badField must be Output Type but got: null.', + ]] ); - - $schema->assertValid(); } /** @@ -1871,18 +1093,41 @@ class ValidationTest extends \PHPUnit_Framework_TestCase foreach ($this->notOutputTypes as $type) { $schema = $this->schemaWithInterfaceFieldOfType($type); - try { - $schema->assertValid(); - $this->fail('Expected exception not thrown for type ' . $type); - } catch (InvariantViolation $e) { - $this->assertEquals( - 'BadInterface.badField field type must be Output Type but got: ' . Utils::printSafe($type), - $e->getMessage() - ); - } + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of BadInterface.badField must be Output Type but got: ' . Utils::printSafe($type) . '.', + ]] + ); } } + /** + * @it rejects a non-output type as an Interface field type with locations + */ + public function testRejectsANonOutputTypeAsAnInterfaceFieldTypeWithLocations() + { + $schema = BuildSchema::build(' + type Query { + field: SomeInterface + } + + interface SomeInterface { + field: SomeInputObject + } + + input SomeInputObject { + foo: String + } + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of SomeInterface.field must be Output Type but got: SomeInputObject.', + 'locations' => [['line' => 7, 'column' => 16]], + ]] + ); + } // DESCRIBE: Type System: Field arguments must have input types @@ -1893,7 +1138,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase { foreach ($this->inputTypes as $type) { $schema = $this->schemaWithArgOfType($type); - $schema->assertValid(); + $this->assertEquals([], $schema->validate()); } } @@ -1903,9 +1148,12 @@ class ValidationTest extends \PHPUnit_Framework_TestCase public function testRejectsAnEmptyFieldArgType() { $schema = $this->schemaWithArgOfType(null); - - $this->setExpectedException(InvariantViolation::class, 'BadObject.badField(badArg): argument type must be Input Type but got: null'); - $schema->assertValid(); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of BadObject.badField(badArg:) must be Input Type but got: null.', + ]] + ); } /** @@ -1915,18 +1163,37 @@ class ValidationTest extends \PHPUnit_Framework_TestCase { foreach ($this->notInputTypes as $type) { $schema = $this->schemaWithArgOfType($type); - try { - $schema->assertValid(); - $this->fail('Expected exception not thrown for type ' . $type); - } catch (InvariantViolation $e) { - $this->assertEquals( - 'BadObject.badField(badArg): argument type must be Input Type but got: ' . Utils::printSafe($type), - $e->getMessage() - ); - } + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of BadObject.badField(badArg:) must be Input Type but got: ' . Utils::printSafe($type) . '.', + ]] + ); } } + /** + * @it rejects a non-input type as a field arg with locations + */ + public function testANonInputTypeAsAFieldArgWithLocations() + { + $schema = BuildSchema::build(' + type Query { + test(arg: SomeObject): String + } + + type SomeObject { + foo: String + } + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of Query.test(arg:) must be Input Type but got: SomeObject.', + 'locations' => [['line' => 3, 'column' => 19]], + ]] + ); + } // DESCRIBE: Type System: Input Object fields must have input types @@ -1937,7 +1204,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase { foreach ($this->inputTypes as $type) { $schema = $this->schemaWithInputFieldOfType($type); - $schema->assertValid(); + $this->assertEquals([], $schema->validate()); } } @@ -1947,12 +1214,12 @@ class ValidationTest extends \PHPUnit_Framework_TestCase public function testRejectsAnEmptyInputFieldType() { $schema = $this->schemaWithInputFieldOfType(null); - - $this->setExpectedException( - InvariantViolation::class, - 'BadInputObject.badField field type must be Input Type but got: null.' + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of BadInputObject.badField must be Input Type but got: null.', + ]] ); - $schema->assertValid(); } /** @@ -1962,126 +1229,40 @@ class ValidationTest extends \PHPUnit_Framework_TestCase { foreach ($this->notInputTypes as $type) { $schema = $this->schemaWithInputFieldOfType($type); - try { - $schema->assertValid(); - $this->fail('Expected exception not thrown for type ' . $type); - } catch (InvariantViolation $e) { - $this->assertEquals( - "BadInputObject.badField field type must be Input Type but got: " . Utils::printSafe($type) . ".", - $e->getMessage() - ); - } - } - } - - - // DESCRIBE: Type System: List must accept only types - - /** - * @it accepts an type as item type of list - */ - public function testAcceptsAnTypeAsItemTypeOfList() - { - $types = $this->withModifiers([ - Type::string(), - $this->SomeScalarType, - $this->SomeObjectType, - $this->SomeUnionType, - $this->SomeInterfaceType, - $this->SomeEnumType, - $this->SomeInputObjectType, - ]); - - foreach ($types as $type) { - try { - Type::listOf($type); - } catch (\Exception $e) { - throw new \Exception("Expection thrown for type $type: {$e->getMessage()}", null, $e); - } + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of BadInputObject.badField must be Input Type but got: ' . Utils::printSafe($type) . '.', + ]] + ); } } /** - * @it rejects a non-type as item type of list + * @it rejects a non-input type as an input object field with locations */ - public function testRejectsANonTypeAsItemTypeOfList() + public function testRejectsANonInputTypeAsAnInputObjectFieldWithLocations() { - $notTypes = [ - [], - new \stdClass(), - 'String', - 10, - null, - true, - false, - // TODO: function() {} - ]; - foreach ($notTypes as $type) { - try { - Type::listOf($type); - $this->fail("Expected exception not thrown for: " . Utils::printSafe($type)); - } catch (InvariantViolation $e) { - $this->assertEquals( - 'Expected '. Utils::printSafe($type) . ' to be a GraphQL type.', - $e->getMessage() - ); - } - } - } - - - // DESCRIBE: Type System: NonNull must only accept non-nullable types - - /** - * @it accepts an type as nullable type of non-null - */ - public function testAcceptsAnTypeAsNullableTypeOfNonNull() - { - $nullableTypes = [ - Type::string(), - $this->SomeScalarType, - $this->SomeObjectType, - $this->SomeUnionType, - $this->SomeInterfaceType, - $this->SomeEnumType, - $this->SomeInputObjectType, - Type::listOf(Type::string()), - Type::listOf(Type::nonNull(Type::string())), - ]; - foreach ($nullableTypes as $type) { - try { - Type::nonNull($type); - } catch (\Exception $e) { - throw new \Exception("Exception thrown for type $type: " . $e->getMessage(), null, $e); - } - } - } - - /** - * @it rejects a non-type as nullable type of non-null - */ - public function testRejectsANonTypeAsNullableTypeOfNonNull() - { - $notNullableTypes = [ - Type::nonNull(Type::string()), - [], - new \stdClass(), - 'String', - null, - true, - false - ]; - foreach ($notNullableTypes as $type) { - try { - Type::nonNull($type); - $this->fail("Expected exception not thrown for: " . Utils::printSafe($type)); - } catch (InvariantViolation $e) { - $this->assertEquals( - 'Expected ' . Utils::printSafe($type) . ' to be a GraphQL nullable type.', - $e->getMessage() - ); - } - } + $schema = BuildSchema::build(' + type Query { + test(arg: SomeInputObject): String + } + + input SomeInputObject { + foo: SomeObject + } + + type SomeObject { + bar: String + } + '); + $this->assertContainsValidationMessage( + $schema->validate(), + [[ + 'message' => 'The type of SomeInputObject.foo must be Input Type but got: SomeObject.', + 'locations' => [['line' => 7, 'column' => 14]], + ]] + ); } // DESCRIBE: Objects must adhere to Interface they implement @@ -2183,9 +1364,11 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - '"AnotherInterface" expects field "field" but ' . - '"AnotherObject" does not provide it.', - [['line' => 7, 'column' => 9], ['line' => 10, 'column' => 7]] + [[ + 'message' => 'Interface field AnotherInterface.field expected but ' . + 'AnotherObject does not provide it.', + 'locations' => [['line' => 7, 'column' => 9], ['line' => 10, 'column' => 7]], + ]] ); } @@ -2210,9 +1393,11 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'AnotherInterface.field expects type "String" but ' . - 'AnotherObject.field is type "Int".', - [['line' => 7, 'column' => 31], ['line' => 11, 'column' => 31]] + [[ + 'message' => 'Interface field AnotherInterface.field expects type String but ' . + 'AnotherObject.field is type Int.', + 'locations' => [['line' => 7, 'column' => 31], ['line' => 11, 'column' => 31]], + ]] ); } @@ -2240,9 +1425,11 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'AnotherInterface.field expects type "A" but ' . - 'AnotherObject.field is type "B".', - [['line' => 10, 'column' => 16], ['line' => 14, 'column' => 16]] + [[ + 'message' => 'Interface field AnotherInterface.field expects type A but ' . + 'AnotherObject.field is type B.', + 'locations' => [['line' => 10, 'column' => 16], ['line' => 14, 'column' => 16]], + ]] ); } @@ -2265,10 +1452,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase } '); - $this->assertEquals( - [], - $schema->validate() - ); + $this->assertEquals([], $schema->validate()); } /** @@ -2296,10 +1480,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase } '); - $this->assertEquals( - [], - $schema->validate() - ); + $this->assertEquals([],$schema->validate()); } /** @@ -2323,9 +1504,11 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'AnotherInterface.field expects argument "input" but ' . - 'AnotherObject.field does not provide it.', - [['line' => 7, 'column' => 15], ['line' => 11, 'column' => 9]] + [[ + 'message' => 'Interface field argument AnotherInterface.field(input:) expected ' . + 'but AnotherObject.field does not provide it.', + 'locations' => [['line' => 7, 'column' => 15], ['line' => 11, 'column' => 9]], + ]] ); } @@ -2350,9 +1533,11 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'AnotherInterface.field(input:) expects type "String" but ' . - 'AnotherObject.field(input:) is type "Int".', - [['line' => 7, 'column' => 22], ['line' => 11, 'column' => 22]] + [[ + 'message' => 'Interface field argument AnotherInterface.field(input:) expects ' . + 'type String but AnotherObject.field(input:) is type Int.', + 'locations' => [['line' => 7, 'column' => 22], ['line' => 11, 'column' => 22]], + ]] ); } @@ -2377,15 +1562,15 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'AnotherInterface.field expects type "String" but ' . - 'AnotherObject.field is type "Int".', - [['line' => 7, 'column' => 31], ['line' => 11, 'column' => 28]] - ); - $this->assertContainsValidationMessage( - $schema->validate(), - 'AnotherInterface.field(input:) expects type "String" but ' . - 'AnotherObject.field(input:) is type "Int".', - [['line' => 7, 'column' => 22], ['line' => 11, 'column' => 22]] + [[ + 'message' => 'Interface field AnotherInterface.field expects type String but ' . + 'AnotherObject.field is type Int.', + 'locations' => [['line' => 7, 'column' => 31], ['line' => 11, 'column' => 28]], + ], [ + 'message' => 'Interface field argument AnotherInterface.field(input:) expects ' . + 'type String but AnotherObject.field(input:) is type Int.', + 'locations' => [['line' => 7, 'column' => 22], ['line' => 11, 'column' => 22]], + ]] ); } @@ -2410,17 +1595,19 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'AnotherObject.field(anotherInput:) is of required type ' . - '"String!" but is not also provided by the interface ' . - 'AnotherInterface.field.', - [['line' => 11, 'column' => 44], ['line' => 7, 'column' => 9]] + [[ + 'message' => 'Object field argument AnotherObject.field(anotherInput:) is of ' . + 'required type String! but is not also provided by the Interface ' . + 'field AnotherInterface.field.', + 'locations' => [['line' => 11, 'column' => 44], ['line' => 7, 'column' => 9]], + ]] ); } /** - * @it accepts an Object with an equivalently modified Interface field type + * @it accepts an Object with an equivalently wrapped Interface field type */ - public function testAcceptsAnObjectWithAnEquivalentlyModifiedInterfaceFieldType() + public function testAcceptsAnObjectWithAnEquivalentlyWrappedInterfaceFieldType() { $schema = BuildSchema::build(' type Query { @@ -2436,10 +1623,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase } '); - $this->assertEquals( - [], - $schema->validate() - ); + $this->assertEquals([], $schema->validate()); } /** @@ -2463,9 +1647,11 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'AnotherInterface.field expects type "[String]" but ' . - 'AnotherObject.field is type "String".', - [['line' => 7, 'column' => 16], ['line' => 11, 'column' => 16]] + [[ + 'message' => 'Interface field AnotherInterface.field expects type [String] ' . + 'but AnotherObject.field is type String.', + 'locations' => [['line' => 7, 'column' => 16], ['line' => 11, 'column' => 16]], + ]] ); } @@ -2490,9 +1676,11 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'AnotherInterface.field expects type "String" but ' . - 'AnotherObject.field is type "[String]".', - [['line' => 7, 'column' => 16], ['line' => 11, 'column' => 16]] + [[ + 'message' => 'Interface field AnotherInterface.field expects type String but ' . + 'AnotherObject.field is type [String].', + 'locations' => [['line' => 7, 'column' => 16], ['line' => 11, 'column' => 16]], + ]] ); } @@ -2515,10 +1703,7 @@ class ValidationTest extends \PHPUnit_Framework_TestCase } '); - $this->assertEquals( - [], - $schema->validate() - ); + $this->assertEquals([], $schema->validate()); } /** @@ -2542,35 +1727,14 @@ class ValidationTest extends \PHPUnit_Framework_TestCase $this->assertContainsValidationMessage( $schema->validate(), - 'AnotherInterface.field expects type "String!" but ' . - 'AnotherObject.field is type "String".', - [['line' => 7, 'column' => 16], ['line' => 11, 'column' => 16]] + [[ + 'message' => 'Interface field AnotherInterface.field expects type String! ' . + 'but AnotherObject.field is type String.', + 'locations' => [['line' => 7, 'column' => 16], ['line' => 11, 'column' => 16]], + ]] ); } - /** - * @it does not allow isDeprecated without deprecationReason on field - */ - public function testDoesNotAllowIsDeprecatedWithoutDeprecationReasonOnField() - { - $OldObject = new ObjectType([ - 'name' => 'OldObject', - 'fields' => [ - 'field' => [ - 'type' => Type::string(), - 'isDeprecated' => true - ] - ] - ]); - - $schema = $this->schemaWithFieldType($OldObject); - $this->setExpectedException( - InvariantViolation::class, - 'OldObject.field should provide "deprecationReason" instead of "isDeprecated".' - ); - $schema->assertValid(); - } - public function testRejectsDifferentInstancesOfTheSameType() { // Invalid: always creates new instance vs returning one from registry @@ -2613,26 +1777,6 @@ class ValidationTest extends \PHPUnit_Framework_TestCase } } - private function assertWarnsOnce($closures, $expectedError) - { - $warned = false; - - foreach ($closures as $index => $factory) { - if (!$warned) { - try { - $factory(); - $this->fail('Expected exception not thrown for entry ' . $index); - } catch (\PHPUnit_Framework_Error_Warning $e) { - $warned = true; - $this->assertEquals($expectedError, $e->getMessage(), 'Error in callable #' . $index); - } - } else { - // Should not throw - $factory(); - } - } - } - private function schemaWithFieldType($type) { return new Schema([ @@ -2644,23 +1788,6 @@ class ValidationTest extends \PHPUnit_Framework_TestCase ]); } - private function schemaWithInputObject($inputObjectType) - { - return new Schema([ - 'query' => new ObjectType([ - 'name' => 'Query', - 'fields' => [ - 'f' => [ - 'type' => Type::string(), - 'args' => [ - 'input' => ['type' => $inputObjectType] - ] - ] - ] - ]) - ]); - } - private function schemaWithObjectFieldOfType($fieldType) { $BadObjectType = new ObjectType([ @@ -2681,28 +1808,6 @@ class ValidationTest extends \PHPUnit_Framework_TestCase ]); } - private function schemaWithObjectWithFieldResolver($resolveValue) - { - $BadResolverType = new ObjectType([ - 'name' => 'BadResolver', - 'fields' => [ - 'badField' => [ - 'type' => Type::string(), - 'resolve' => $resolveValue - ] - ] - ]); - - return new Schema([ - 'query' => new ObjectType([ - 'name' => 'Query', - 'fields' => [ - 'f' => ['type' => $BadResolverType] - ] - ]) - ]); - } - private function withModifiers($types) { return array_merge( @@ -2719,22 +1824,6 @@ class ValidationTest extends \PHPUnit_Framework_TestCase ); } - private function schemaWithUnionOfType($type) - { - $BadUnionType = new UnionType([ - 'name' => 'BadUnion', - 'types' => [$type], - ]); - return new Schema([ - 'query' => new ObjectType([ - 'name' => 'Query', - 'fields' => [ - 'f' => ['type' => $BadUnionType] - ] - ]) - ]); - } - private function schemaWithInterfaceFieldOfType($fieldType) { $BadInterfaceType = new InterfaceType([ @@ -2751,18 +1840,6 @@ class ValidationTest extends \PHPUnit_Framework_TestCase 'f' => ['type' => $BadInterfaceType] ] ]), - // Have to add types implementing interfaces to bypass the "could not find implementers" exception - 'types' => [ - new ObjectType([ - 'name' => 'BadInterfaceImplementer', - 'fields' => [ - 'badField' => ['type' => $fieldType] - ], - 'interfaces' => [$BadInterfaceType], - 'isTypeOf' => function() {} - ]), - $this->SomeObjectType - ] ]); } diff --git a/tests/Utils/AssertValidNameTest.php b/tests/Utils/AssertValidNameTest.php new file mode 100644 index 0000000..50899f7 --- /dev/null +++ b/tests/Utils/AssertValidNameTest.php @@ -0,0 +1,47 @@ +setExpectedException( + Error::class, + '"__bad" must not begin with "__", which is reserved by GraphQL introspection.' + ); + Utils::assertValidName('__bad'); + } + + /** + * @it throws for non-strings + */ + public function testThrowsForNonStrings() + { + $this->setExpectedException( + InvariantViolation::class, + 'Expected string' + ); + Utils::assertValidName([]); + } + + /** + * @it throws for names with invalid characters + */ + public function testThrowsForNamesWithInvalidCharacters() + { + $this->setExpectedException( + Error::class, + 'Names must match' + ); + Utils::assertValidName('>--()-->'); + } +}