syncError = new \Exception('sync'); $this->nonNullSyncError = new \Exception('nonNullSync'); $this->throwingData = [ 'sync' => function () { throw $this->syncError; }, 'nonNullSync' => function () { throw $this->nonNullSyncError; }, 'nest' => function () { return $this->throwingData; }, 'nonNullNest' => function () { return $this->throwingData; }, ]; $this->nullingData = [ 'sync' => function () { return null; }, 'nonNullSync' => function () { return null; }, 'nest' => function () { return $this->nullingData; }, 'nonNullNest' => function () { return $this->nullingData; }, ]; $dataType = new ObjectType([ 'name' => 'DataType', 'fields' => function() use (&$dataType) { return [ 'sync' => ['type' => Type::string()], 'nonNullSync' => ['type' => Type::nonNull(Type::string())], 'nest' => $dataType, 'nonNullNest' => Type::nonNull($dataType) ]; } ]); $this->schema = new Schema(['query' => $dataType]); } // Execute: handles non-nullable types /** * @it nulls a nullable field that throws synchronously */ public function testNullsANullableFieldThatThrowsSynchronously() { $doc = ' query Q { sync } '; $ast = Parser::parse($doc); $expected = [ 'data' => [ 'sync' => null, ], 'errors' => [ FormattedError::create( $this->syncError->getMessage(), [new SourceLocation(3, 9)] ) ] ]; $this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->throwingData, null, [], 'Q')->toArray()); } public function testNullsASynchronouslyReturnedObjectThatContainsANonNullableFieldThatThrowsSynchronously() { // nulls a synchronously returned object that contains a non-nullable field that throws synchronously $doc = ' query Q { nest { nonNullSync, } } '; $ast = Parser::parse($doc); $expected = [ 'data' => [ 'nest' => null ], 'errors' => [ FormattedError::create($this->nonNullSyncError->getMessage(), [new SourceLocation(4, 11)]) ] ]; $this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->throwingData, null, [], 'Q')->toArray()); } public function testNullsAComplexTreeOfNullableFieldsThatThrow() { $doc = ' query Q { nest { sync nest { sync } } } '; $ast = Parser::parse($doc); $expected = [ 'data' => [ 'nest' => [ 'sync' => null, 'nest' => [ 'sync' => null, ] ] ], 'errors' => [ FormattedError::create($this->syncError->getMessage(), [new SourceLocation(4, 11)]), FormattedError::create($this->syncError->getMessage(), [new SourceLocation(6, 13)]), ] ]; $this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->throwingData, null, [], 'Q')->toArray()); } public function testNullsANullableFieldThatSynchronouslyReturnsNull() { $doc = ' query Q { sync } '; $ast = Parser::parse($doc); $expected = [ 'data' => [ 'sync' => null, ] ]; $this->assertEquals($expected, Executor::execute($this->schema, $ast, $this->nullingData, null, [], 'Q')->toArray()); } public function test4() { // nulls a synchronously returned object that contains a non-nullable field that returns null synchronously $doc = ' query Q { nest { nonNullSync, } } '; $ast = Parser::parse($doc); $expected = [ 'data' => [ 'nest' => null ], 'errors' => [ FormattedError::create('Cannot return null for non-nullable field DataType.nonNullSync.', [new SourceLocation(4, 11)]) ] ]; $this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->nullingData, null, [], 'Q')->toArray()); } public function test5() { // nulls a complex tree of nullable fields that return null $doc = ' query Q { nest { sync nest { sync nest { sync } } } } '; $ast = Parser::parse($doc); $expected = [ 'data' => [ 'nest' => [ 'sync' => null, 'nest' => [ 'sync' => null, 'nest' => [ 'sync' => null ] ], ], ] ]; $this->assertEquals($expected, Executor::execute($this->schema, $ast, $this->nullingData, null, [], 'Q')->toArray()); } public function testNullsTheTopLevelIfSyncNonNullableFieldThrows() { $doc = ' query Q { nonNullSync } '; $expected = [ 'errors' => [ FormattedError::create($this->nonNullSyncError->getMessage(), [new SourceLocation(2, 17)]) ] ]; $this->assertArraySubset($expected, Executor::execute($this->schema, Parser::parse($doc), $this->throwingData)->toArray()); } public function testNullsTheTopLevelIfSyncNonNullableFieldReturnsNull() { // nulls the top level if sync non-nullable field returns null $doc = ' query Q { nonNullSync } '; $expected = [ 'errors' => [ FormattedError::create('Cannot return null for non-nullable field DataType.nonNullSync.', [new SourceLocation(2, 17)]), ] ]; $this->assertArraySubset($expected, Executor::execute($this->schema, Parser::parse($doc), $this->nullingData)->toArray()); } }