mirror of
https://github.com/retailcrm/graphql-php.git
synced 2024-11-25 14:26:08 +03:00
770 lines
22 KiB
PHP
770 lines
22 KiB
PHP
<?php
|
|
|
|
namespace GraphQL\Tests\Executor;
|
|
|
|
use GraphQL\Deferred;
|
|
use GraphQL\Executor\Executor;
|
|
use GraphQL\Error\FormattedError;
|
|
use GraphQL\Language\Parser;
|
|
use GraphQL\Language\SourceLocation;
|
|
use GraphQL\Schema;
|
|
use GraphQL\Type\Definition\ObjectType;
|
|
use GraphQL\Type\Definition\Type;
|
|
|
|
class NonNullTest extends \PHPUnit_Framework_TestCase
|
|
{
|
|
/** @var \Exception */
|
|
public $syncError;
|
|
|
|
/** @var \Exception */
|
|
public $nonNullSyncError;
|
|
|
|
/** @var \Exception */
|
|
public $promiseError;
|
|
|
|
/** @var \Exception */
|
|
public $nonNullPromiseError;
|
|
|
|
public $throwingData;
|
|
public $nullingData;
|
|
public $schema;
|
|
|
|
public function setUp()
|
|
{
|
|
$this->syncError = new \Exception('sync');
|
|
$this->nonNullSyncError = new \Exception('nonNullSync');
|
|
$this->promiseError = new \Exception('promise');
|
|
$this->nonNullPromiseError = new \Exception('nonNullPromise');
|
|
|
|
$this->throwingData = [
|
|
'sync' => function () {
|
|
throw $this->syncError;
|
|
},
|
|
'nonNullSync' => function () {
|
|
throw $this->nonNullSyncError;
|
|
},
|
|
'promise' => function () {
|
|
return new Deferred(function () {
|
|
throw $this->promiseError;
|
|
});
|
|
},
|
|
'nonNullPromise' => function () {
|
|
return new Deferred(function () {
|
|
throw $this->nonNullPromiseError;
|
|
});
|
|
},
|
|
'nest' => function () {
|
|
return $this->throwingData;
|
|
},
|
|
'nonNullNest' => function () {
|
|
return $this->throwingData;
|
|
},
|
|
'promiseNest' => function () {
|
|
return new Deferred(function () {
|
|
return $this->throwingData;
|
|
});
|
|
},
|
|
'nonNullPromiseNest' => function () {
|
|
return new Deferred(function () {
|
|
return $this->throwingData;
|
|
});
|
|
},
|
|
];
|
|
|
|
$this->nullingData = [
|
|
'sync' => function () {
|
|
return null;
|
|
},
|
|
'nonNullSync' => function () {
|
|
return null;
|
|
},
|
|
'promise' => function () {
|
|
return new Deferred(function () {
|
|
return null;
|
|
});
|
|
},
|
|
'nonNullPromise' => function () {
|
|
return new Deferred(function () {
|
|
return null;
|
|
});
|
|
},
|
|
'nest' => function () {
|
|
return $this->nullingData;
|
|
},
|
|
'nonNullNest' => function () {
|
|
return $this->nullingData;
|
|
},
|
|
'promiseNest' => function () {
|
|
return new Deferred(function () {
|
|
return $this->nullingData;
|
|
});
|
|
},
|
|
'nonNullPromiseNest' => function () {
|
|
return new Deferred(function () {
|
|
return $this->nullingData;
|
|
});
|
|
},
|
|
];
|
|
|
|
$dataType = new ObjectType([
|
|
'name' => 'DataType',
|
|
'fields' => function() use (&$dataType) {
|
|
return [
|
|
'sync' => ['type' => Type::string()],
|
|
'nonNullSync' => ['type' => Type::nonNull(Type::string())],
|
|
'promise' => Type::string(),
|
|
'nonNullPromise' => Type::nonNull(Type::string()),
|
|
'nest' => $dataType,
|
|
'nonNullNest' => Type::nonNull($dataType),
|
|
'promiseNest' => $dataType,
|
|
'nonNullPromiseNest' => 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 testNullsANullableFieldThatThrowsInAPromise()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
promise
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'promise' => null,
|
|
],
|
|
'errors' => [
|
|
FormattedError::create(
|
|
$this->promiseError->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 testNullsAsynchronouslyReturnedObjectThatContainsANonNullableFieldThatThrowsInAPromise()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
nest {
|
|
nonNullPromise,
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'nest' => null
|
|
],
|
|
'errors' => [
|
|
FormattedError::create($this->nonNullPromiseError->getMessage(), [new SourceLocation(4, 11)])
|
|
]
|
|
];
|
|
|
|
$this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->throwingData, null, [], 'Q')->toArray());
|
|
}
|
|
|
|
public function testNullsAnObjectReturnedInAPromiseThatContainsANonNullableFieldThatThrowsSynchronously()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
promiseNest {
|
|
nonNullSync,
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'promiseNest' => 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 testNullsAnObjectReturnedInAPromiseThatContainsANonNullableFieldThatThrowsInAPromise()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
promiseNest {
|
|
nonNullPromise,
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'promiseNest' => null
|
|
],
|
|
'errors' => [
|
|
FormattedError::create($this->nonNullPromiseError->getMessage(), [new SourceLocation(4, 11)])
|
|
]
|
|
];
|
|
|
|
$this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->throwingData, null, [], 'Q')->toArray());
|
|
}
|
|
|
|
/**
|
|
* @it nulls a complex tree of nullable fields that throw
|
|
*/
|
|
public function testNullsAComplexTreeOfNullableFieldsThatThrow()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
nest {
|
|
sync
|
|
promise
|
|
nest {
|
|
sync
|
|
promise
|
|
}
|
|
promiseNest {
|
|
sync
|
|
promise
|
|
}
|
|
}
|
|
promiseNest {
|
|
sync
|
|
promise
|
|
nest {
|
|
sync
|
|
promise
|
|
}
|
|
promiseNest {
|
|
sync
|
|
promise
|
|
}
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'nest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
'nest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
],
|
|
'promiseNest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
],
|
|
],
|
|
'promiseNest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
'nest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
],
|
|
'promiseNest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
],
|
|
],
|
|
],
|
|
'errors' => [
|
|
FormattedError::create($this->syncError->getMessage(), [new SourceLocation(4, 11)]),
|
|
FormattedError::create($this->syncError->getMessage(), [new SourceLocation(7, 13)]),
|
|
FormattedError::create($this->syncError->getMessage(), [new SourceLocation(11, 13)]),
|
|
FormattedError::create($this->syncError->getMessage(), [new SourceLocation(16, 11)]),
|
|
FormattedError::create($this->syncError->getMessage(), [new SourceLocation(19, 13)]),
|
|
FormattedError::create($this->promiseError->getMessage(), [new SourceLocation(5, 11)]),
|
|
FormattedError::create($this->promiseError->getMessage(), [new SourceLocation(8, 13)]),
|
|
FormattedError::create($this->syncError->getMessage(), [new SourceLocation(23, 13)]),
|
|
FormattedError::create($this->promiseError->getMessage(), [new SourceLocation(12, 13)]),
|
|
FormattedError::create($this->promiseError->getMessage(), [new SourceLocation(17, 11)]),
|
|
FormattedError::create($this->promiseError->getMessage(), [new SourceLocation(20, 13)]),
|
|
FormattedError::create($this->promiseError->getMessage(), [new SourceLocation(24, 13)]),
|
|
]
|
|
];
|
|
|
|
$this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->throwingData, null, [], 'Q')->toArray());
|
|
}
|
|
|
|
public function testNullsTheFirstNullableObjectAfterAFieldThrowsInALongChainOfFieldsThatAreNonNull()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
nest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullSync
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
promiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullSync
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
anotherNest: nest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullPromise
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
anotherPromiseNest: promiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullPromise
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'nest' => null,
|
|
'promiseNest' => null,
|
|
'anotherNest' => null,
|
|
'anotherPromiseNest' => null,
|
|
],
|
|
'errors' => [
|
|
FormattedError::create($this->nonNullSyncError->getMessage(), [new SourceLocation(8, 19)]),
|
|
FormattedError::create($this->nonNullSyncError->getMessage(), [new SourceLocation(19, 19)]),
|
|
FormattedError::create($this->nonNullPromiseError->getMessage(), [new SourceLocation(30, 19)]),
|
|
FormattedError::create($this->nonNullPromiseError->getMessage(), [new SourceLocation(41, 19)]),
|
|
]
|
|
];
|
|
|
|
$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 testNullsANullableFieldThatReturnsNullInAPromise()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
promise
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'promise' => null,
|
|
]
|
|
];
|
|
|
|
$this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->nullingData, null, [], 'Q')->toArray());
|
|
}
|
|
|
|
public function testNullsASynchronouslyReturnedObjectThatContainsANonNullableFieldThatReturnsNullSynchronously()
|
|
{
|
|
$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 testNullsASynchronouslyReturnedObjectThatContainsANonNullableFieldThatReturnsNullInAPromise()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
nest {
|
|
nonNullPromise,
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'nest' => null,
|
|
],
|
|
'errors' => [
|
|
FormattedError::create('Cannot return null for non-nullable field DataType.nonNullPromise.', [new SourceLocation(4, 11)]),
|
|
]
|
|
];
|
|
|
|
$this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->nullingData, null, [], 'Q')->toArray());
|
|
}
|
|
|
|
public function testNullsAnObjectReturnedInAPromiseThatContainsANonNullableFieldThatReturnsNullSynchronously()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
promiseNest {
|
|
nonNullSync,
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'promiseNest' => 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 testNullsAnObjectReturnedInAPromiseThatContainsANonNullableFieldThatReturnsNullInaAPromise()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
promiseNest {
|
|
nonNullPromise,
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'promiseNest' => null,
|
|
],
|
|
'errors' => [
|
|
FormattedError::create('Cannot return null for non-nullable field DataType.nonNullPromise.', [new SourceLocation(4, 11)]),
|
|
]
|
|
];
|
|
|
|
$this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->nullingData, null, [], 'Q')->toArray());
|
|
}
|
|
|
|
public function testNullsAComplexTreeOfNullableFieldsThatReturnNull()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
nest {
|
|
sync
|
|
promise
|
|
nest {
|
|
sync
|
|
promise
|
|
}
|
|
promiseNest {
|
|
sync
|
|
promise
|
|
}
|
|
}
|
|
promiseNest {
|
|
sync
|
|
promise
|
|
nest {
|
|
sync
|
|
promise
|
|
}
|
|
promiseNest {
|
|
sync
|
|
promise
|
|
}
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'nest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
'nest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
],
|
|
'promiseNest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
]
|
|
],
|
|
'promiseNest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
'nest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
],
|
|
'promiseNest' => [
|
|
'sync' => null,
|
|
'promise' => null,
|
|
]
|
|
]
|
|
]
|
|
];
|
|
|
|
$actual = Executor::execute($this->schema, $ast, $this->nullingData, null, [], 'Q')->toArray();
|
|
$this->assertEquals($expected, $actual);
|
|
}
|
|
|
|
public function testNullsTheFirstNullableObjectAfterAFieldReturnsNullInALongChainOfFieldsThatAreNonNull()
|
|
{
|
|
$doc = '
|
|
query Q {
|
|
nest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullSync
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
promiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullSync
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
anotherNest: nest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullPromise
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
anotherPromiseNest: promiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullNest {
|
|
nonNullPromiseNest {
|
|
nonNullPromise
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => [
|
|
'nest' => null,
|
|
'promiseNest' => null,
|
|
'anotherNest' => null,
|
|
'anotherPromiseNest' => null,
|
|
],
|
|
'errors' => [
|
|
FormattedError::create('Cannot return null for non-nullable field DataType.nonNullSync.', [new SourceLocation(8, 19)]),
|
|
FormattedError::create('Cannot return null for non-nullable field DataType.nonNullSync.', [new SourceLocation(19, 19)]),
|
|
FormattedError::create('Cannot return null for non-nullable field DataType.nonNullPromise.', [new SourceLocation(30, 19)]),
|
|
FormattedError::create('Cannot return null for non-nullable field DataType.nonNullPromise.', [new SourceLocation(41, 19)]),
|
|
]
|
|
];
|
|
|
|
$this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->nullingData, null, [], 'Q')->toArray());
|
|
}
|
|
|
|
/**
|
|
* @it nulls the top level if sync non-nullable field throws
|
|
*/
|
|
public function testNullsTheTopLevelIfSyncNonNullableFieldThrows()
|
|
{
|
|
$doc = '
|
|
query Q { nonNullSync }
|
|
';
|
|
|
|
$expected = [
|
|
'data' => null,
|
|
'errors' => [
|
|
FormattedError::create($this->nonNullSyncError->getMessage(), [new SourceLocation(2, 17)])
|
|
]
|
|
];
|
|
$actual = Executor::execute($this->schema, Parser::parse($doc), $this->throwingData)->toArray();
|
|
$this->assertArraySubset($expected, $actual);
|
|
}
|
|
|
|
public function testNullsTheTopLevelIfAsyncNonNullableFieldErrors()
|
|
{
|
|
$doc = '
|
|
query Q { nonNullPromise }
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => null,
|
|
'errors' => [
|
|
FormattedError::create($this->nonNullPromiseError->getMessage(), [new SourceLocation(2, 17)]),
|
|
]
|
|
];
|
|
|
|
$this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->throwingData, null, [], 'Q')->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());
|
|
}
|
|
|
|
public function testNullsTheTopLevelIfAsyncNonNullableFieldResolvesNull()
|
|
{
|
|
$doc = '
|
|
query Q { nonNullPromise }
|
|
';
|
|
|
|
$ast = Parser::parse($doc);
|
|
|
|
$expected = [
|
|
'data' => null,
|
|
'errors' => [
|
|
FormattedError::create('Cannot return null for non-nullable field DataType.nonNullPromise.', [new SourceLocation(2, 17)]),
|
|
]
|
|
];
|
|
|
|
$this->assertArraySubset($expected, Executor::execute($this->schema, $ast, $this->nullingData, null, [], 'Q')->toArray());
|
|
}
|
|
}
|