'Pet', 'fields' => [ 'name' => ['type' => Type::string()] ] ]); // Added to interface type when defined $dogType = new ObjectType([ 'name' => 'Dog', 'interfaces' => [$petType], 'isTypeOf' => function($obj) { return $obj instanceof Dog; }, 'fields' => [ 'name' => ['type' => Type::string()], 'woofs' => ['type' => Type::boolean()] ] ]); $catType = new ObjectType([ 'name' => 'Cat', 'interfaces' => [$petType], 'isTypeOf' => function ($obj) { return $obj instanceof Cat; }, 'fields' => [ 'name' => ['type' => Type::string()], 'meows' => ['type' => Type::boolean()], ] ]); $schema = new Schema([ 'query' => new ObjectType([ 'name' => 'Query', 'fields' => [ 'pets' => [ 'type' => Type::listOf($petType), 'resolve' => function () { return [new Dog('Odie', true), new Cat('Garfield', false)]; } ] ] ]), 'types' => [$catType, $dogType] ]); $query = '{ pets { name ... on Dog { woofs } ... on Cat { meows } } }'; $expected = new ExecutionResult([ 'pets' => [ ['name' => 'Odie', 'woofs' => true], ['name' => 'Garfield', 'meows' => false] ] ]); Warning::suppress(Warning::RESOLVE_TYPE_WARNING); $result = Executor::execute($schema, Parser::parse($query)); $this->assertEquals($expected, $result); Warning::enable(Warning::RESOLVE_TYPE_WARNING); $result = Executor::execute($schema, Parser::parse($query)); $this->assertEquals(1, count($result->errors)); $this->assertInstanceOf('PHPUnit_Framework_Error_Warning', $result->errors[0]->getPrevious()); $this->assertEquals( 'GraphQL Interface Type `Pet` returned `null` from it`s `resolveType` function for value: '. 'instance of GraphQL\Tests\Executor\Dog. Switching to slow resolution method using `isTypeOf` of '. 'all possible implementations. It degrades query performance significantly. '. 'Make sure your `resolveType` always returns valid implementation or throws.', $result->errors[0]->getMessage()); } /** * @it isTypeOf used to resolve runtime type for Union */ public function testIsTypeOfUsedToResolveRuntimeTypeForUnion() { $dogType = new ObjectType([ 'name' => 'Dog', 'isTypeOf' => function($obj) { return $obj instanceof Dog; }, 'fields' => [ 'name' => ['type' => Type::string()], 'woofs' => ['type' => Type::boolean()] ] ]); $catType = new ObjectType([ 'name' => 'Cat', 'isTypeOf' => function ($obj) { return $obj instanceof Cat; }, 'fields' => [ 'name' => ['type' => Type::string()], 'meows' => ['type' => Type::boolean()], ] ]); $petType = new UnionType([ 'name' => 'Pet', 'types' => [$dogType, $catType] ]); $schema = new Schema([ 'query' => new ObjectType([ 'name' => 'Query', 'fields' => [ 'pets' => [ 'type' => Type::listOf($petType), 'resolve' => function() { return [ new Dog('Odie', true), new Cat('Garfield', false) ]; } ] ] ]) ]); $query = '{ pets { name ... on Dog { woofs } ... on Cat { meows } } }'; $expected = new ExecutionResult([ 'pets' => [ ['name' => 'Odie', 'woofs' => true], ['name' => 'Garfield', 'meows' => false] ] ]); $this->assertEquals($expected, Executor::execute($schema, Parser::parse($query))); } /** * @it resolveType on Interface yields useful error */ function testResolveTypeOnInterfaceYieldsUsefulError() { $DogType = null; $CatType = null; $HumanType = null; $PetType = new InterfaceType([ 'name' => 'Pet', 'resolveType' => function ($obj) use (&$DogType, &$CatType, &$HumanType) { if ($obj instanceof Dog) { return $DogType; } if ($obj instanceof Cat) { return $CatType; } if ($obj instanceof Human) { return $HumanType; } return null; }, 'fields' => [ 'name' => ['type' => Type::string()] ] ]); $HumanType = new ObjectType([ 'name' => 'Human', 'fields' => [ 'name' => ['type' => Type::string()], ] ]); $DogType = new ObjectType([ 'name' => 'Dog', 'interfaces' => [$PetType], 'fields' => [ 'name' => ['type' => Type::string()], 'woofs' => ['type' => Type::boolean()], ] ]); $CatType = new ObjectType([ 'name' => 'Cat', 'interfaces' => [$PetType], 'fields' => [ 'name' => ['type' => Type::string()], 'meows' => ['type' => Type::boolean()], ] ]); $schema = new Schema([ 'query' => new ObjectType([ 'name' => 'Query', 'fields' => [ 'pets' => [ 'type' => Type::listOf($PetType), 'resolve' => function () { return [ new Dog('Odie', true), new Cat('Garfield', false), new Human('Jon') ]; } ] ], ]), 'types' => [$DogType, $CatType] ]); $query = '{ pets { name ... on Dog { woofs } ... on Cat { meows } } }'; $expected = [ 'data' => [ 'pets' => [ ['name' => 'Odie', 'woofs' => true], ['name' => 'Garfield', 'meows' => false], null ] ], 'errors' => [[ 'message' => 'Runtime Object type "Human" is not a possible type for "Pet".', 'locations' => [['line' => 2, 'column' => 11]], 'path' => ['pets', 2] ]] ]; $actual = GraphQL::execute($schema, $query); $this->assertEquals($expected, $actual); } /** * @it resolveType on Union yields useful error */ public function testResolveTypeOnUnionYieldsUsefulError() { $HumanType = new ObjectType([ 'name' => 'Human', 'fields' => [ 'name' => ['type' => Type::string()], ] ]); $DogType = new ObjectType([ 'name' => 'Dog', 'fields' => [ 'name' => ['type' => Type::string()], 'woofs' => ['type' => Type::boolean()], ] ]); $CatType = new ObjectType([ 'name' => 'Cat', 'fields' => [ 'name' => ['type' => Type::string()], 'meows' => ['type' => Type::boolean()], ] ]); $PetType = new UnionType([ 'name' => 'Pet', 'resolveType' => function ($obj) use ($DogType, $CatType, $HumanType) { if ($obj instanceof Dog) { return $DogType; } if ($obj instanceof Cat) { return $CatType; } if ($obj instanceof Human) { return $HumanType; } }, 'types' => [$DogType, $CatType] ]); $schema = new Schema([ 'query' => new ObjectType([ 'name' => 'Query', 'fields' => [ 'pets' => [ 'type' => Type::listOf($PetType), 'resolve' => function () { return [ new Dog('Odie', true), new Cat('Garfield', false), new Human('Jon') ]; } ] ] ]) ]); $query = '{ pets { ... on Dog { name woofs } ... on Cat { name meows } } }'; $result = GraphQL::execute($schema, $query); $expected = [ 'data' => [ 'pets' => [ ['name' => 'Odie', 'woofs' => true], ['name' => 'Garfield', 'meows' => false], null ] ], 'errors' => [[ 'message' => 'Runtime Object type "Human" is not a possible type for "Pet".', 'locations' => [['line' => 2, 'column' => 11]], 'path' => ['pets', 2] ]] ]; $this->assertEquals($expected, $result); } /** * @it resolveType allows resolving with type name */ public function testResolveTypeAllowsResolvingWithTypeName() { $PetType = new InterfaceType([ 'name' => 'Pet', 'resolveType' => function($obj) { if ($obj instanceof Dog) return 'Dog'; if ($obj instanceof Cat) return 'Cat'; return null; }, 'fields' => [ 'name' => [ 'type' => Type::string() ] ] ]); $DogType = new ObjectType([ 'name' => 'Dog', 'interfaces' => [ $PetType ], 'fields' => [ 'name' => [ 'type' => Type::string() ], 'woofs' => [ 'type' => Type::boolean() ], ] ]); $CatType = new ObjectType([ 'name' => 'Cat', 'interfaces' => [ $PetType ], 'fields' => [ 'name' => [ 'type' => Type::string() ], 'meows' => [ 'type' => Type::boolean() ], ] ]); $schema = new Schema([ 'query' => new ObjectType([ 'name' => 'Query', 'fields' => [ 'pets' => [ 'type' => Type::listOf($PetType), 'resolve' => function() { return [ new Dog('Odie', true), new Cat('Garfield', false) ]; } ] ] ]), 'types' => [ $CatType, $DogType ] ]); $query = '{ pets { name ... on Dog { woofs } ... on Cat { meows } } }'; $result = GraphQL::execute($schema, $query); $this->assertEquals([ 'data' => [ 'pets' => [ ['name' => 'Odie', 'woofs' => true], ['name' => 'Garfield', 'meows' => false] ] ] ], $result); } }