Compare commits

...

34 commits

Author SHA1 Message Date
Vladimir Razuvaev
4c545e5ec4 v0.12.6 2018-09-02 21:59:54 +07:00
Vladimir Razuvaev
f2622fe7fb
Merge pull request #336 from martiis/0.12.x
from master to 0.12.x - Call to a member function getLocation() on null
2018-09-02 21:46:02 +07:00
Jayme Rotsaert
b36caceade Fix: fatal Call to a member function getLocation(), before normal error is thrown 2018-08-31 14:18:46 +03:00
Vladimir Razuvaev
f1fc5d66c9
Merge pull request #329 from mcg-web/code_quality
Code quality
2018-08-28 14:41:55 +07:00
Jeremiah VALERIE
761bcfb6f7
Fix ReactPromiseInterface 2018-08-23 09:21:44 +02:00
Jeremiah VALERIE
6996e2da2b
Remove unused variables 2018-08-23 09:16:08 +02:00
Jeremiah VALERIE
4209a2f60e
Fix typehint 2018-08-23 09:15:56 +02:00
Jeremiah VALERIE
19235f5a2d
Add undefined variable complexityFn and fix typehint 2018-08-23 09:15:08 +02:00
Jeremiah VALERIE
639f493e47
Fix Utils fqcn 2018-08-23 09:14:23 +02:00
Jeremiah VALERIE
297bac0dea
Extract VisitorOperation to be more psr-4 compliant 2018-08-23 09:13:20 +02:00
Jeremiah VALERIE
68f0be08cb
Fix variable $editValue might not be defined. 2018-08-23 09:12:19 +02:00
Jeremiah VALERIE
788581ad24
Fix typehint 2018-08-23 09:11:17 +02:00
Jeremiah VALERIE
f7132eb34f
Add undefined property $promises 2018-08-23 09:10:10 +02:00
Jeremiah VALERIE
3da424dacf
Fix dev deps 2018-08-23 09:07:05 +02:00
Jeremiah VALERIE
fce0e4dd22
Use continue 2 instead of continue 2018-08-23 08:43:25 +02:00
Jeremiah VALERIE
0d63d74cbb
Add phpstan PHP Static Analysis Tool 2018-08-23 08:34:49 +02:00
Vladimir Razuvaev
d5b4d446ce v0.12.5 2018-08-03 21:04:43 +07:00
Vladimir Razuvaev
5c9b5576e6 Perf: memoize collectSubfields 2018-08-03 21:03:48 +07:00
Vladimir Razuvaev
b705ee797f v0.12.4 2018-07-08 04:43:07 +07:00
Vladimir Razuvaev
17392e05dd
Merge pull request #303 from theofidry/bugfix/stringeable-object
Allow stringeable objects to be serialized by StringType
2018-07-08 04:40:16 +07:00
Théo FIDRY
c258109844
Allow stringeable objects to be serialized by StringType
Closes #302
2018-07-07 22:55:53 +02:00
Vladimir Razuvaev
50dfd3fac2 v0.12.3 2018-07-07 22:46:58 +07:00
Iain Mckay
c5b78c66e9 Adds support for the multipart/form-data content type
(cherry picked from commit 750ce38)
2018-07-07 22:44:18 +07:00
Vasily Kartashov
47bcabfd7b Update complementary-tools.md
Adding reference to graphql-batch-processing library

(cherry picked from commit ec77f43)
2018-07-07 22:43:49 +07:00
Daniel Tschinder
4ef7920961 fix: Correct namespace and link for DirectiveLocation in docs
(cherry picked from commit 1b6fb4c)
2018-07-07 22:43:28 +07:00
Vladimir Razuvaev
167c3e7354 v0.12.2 2018-06-25 23:56:32 +07:00
Daniel Tschinder
a8cd87acff Use multi-line block for trailing quote
ref: fdc10bb918 (diff-ebaed8492e8d884ee4f2255e39909568)

(cherry picked from commit 6e64983)
2018-06-25 23:50:19 +07:00
Vladimir Razuvaev
e9cd1daedb v0.12.1 2018-06-23 11:38:13 +07:00
Daniel Tschinder
2540436a2c Fix wrong length being used in validator.
(cherry picked from commit 8ba1460)
2018-06-23 11:30:09 +07:00
Daniel Tschinder
d53f7f041e Improve example
(cherry picked from commit 3a4f520)
2018-06-23 11:30:00 +07:00
Daniel Tschinder
91daa23c5f Add one more breaking change in 0.12
(cherry picked from commit 300b580)
2018-06-23 11:29:52 +07:00
Ben Zhu
57c77623ee fix port bug
(cherry picked from commit 72e8607)
2018-06-23 11:29:27 +07:00
Ilya Shaydullin
78e2862e3b Fix is null condition
(cherry picked from commit 06490ca)
2018-06-23 11:29:03 +07:00
Ilya Shaydullin
63b04df6ef Removing data elements from response if the error throwing
(cherry picked from commit c7f114d)
2018-06-23 11:28:53 +07:00
26 changed files with 315 additions and 123 deletions

1
.gitignore vendored
View file

@ -3,3 +3,4 @@ composer.phar
composer.lock
vendor/
bin/
phpstan.phar

View file

@ -9,7 +9,13 @@ php:
- 7.2
- nightly
matrix:
jobs:
include:
- stage: Code Quality
php: 7.2
env: STATIC_ANALYSIS
install: travis_retry composer install --prefer-dist
script: composer static-analysis
allow_failures:
- php: nightly
@ -22,10 +28,7 @@ before_install:
- phpenv config-rm xdebug.ini || true
- composer selfupdate
install:
- composer install --dev --prefer-dist
- composer require react/promise:2.*
- composer require psr/http-message:1.*
install: composer install --dev --prefer-dist
script: if [ "$TRAVIS_PHP_VERSION" == "5.6" ]; then bin/phpunit --coverage-clover build/logs/clover.xml --group default,ReactPromise; else bin/phpunit --group default,ReactPromise; fi

View file

@ -1,4 +1,25 @@
# Changelog
#### v0.12.6
- Bugfix: Call to a member function getLocation() on null (#336)
- Fixed several errors discovered by static analysis (#329)
#### v0.12.5
- Execution performance optimization for lists
#### v0.12.4
- Allow stringeable objects to be serialized by StringType (#303)
#### v0.12.3
- StandardServer: add support for the multipart/form-data content type (#300)
#### v0.12.2
- SchemaPrinter: Use multi-line block for trailing quote (#294)
#### v0.12.1
- Fixed bug in validation rule OverlappingFieldsCanBeMerged (#292)
- Added one more breaking change note in UPGRADE.md (#291)
- Spec compliance: remove `data` entry from response on top-level error (#281)
## v0.12.0
- RFC: Block String (multi-line strings via triple-quote """string""")
- GraphQL Schema SDL: Descriptions as strings (including multi-line)

View file

@ -10,6 +10,33 @@ Exception inside `parseLiteral()`, `parseValue()` and `serialize()`.
Returning null from any of these methods will now be treated as valid result.
### Breaking: Custom scalar types parseLiteral() declaration changed
A new parameter was added to `parseLiteral()`, which also needs to be added to any custom scalar type extending from `ScalarType`
Before:
```php
class MyType extends ScalarType {
...
public function parseLiteral($valueNode) {
//custom implementation
}
}
```
After:
```php
class MyType extends ScalarType {
...
public function parseLiteral($valueNode, array $variables = null) {
//custom implementation
}
}
```
### Breaking: Descriptions in comments are not used as descriptions by default anymore
Descriptions now need to be inside Strings or BlockStrings in order to be picked up as
description. If you want to keep the old behaviour you can supply the option `commentDescriptions`

View file

@ -14,16 +14,14 @@
},
"require-dev": {
"phpunit/phpunit": "^4.8",
"psr/http-message": "^1.0"
"psr/http-message": "^1.0",
"react/promise": "2.*"
},
"config": {
"bin-dir": "bin",
"platform": {
"php": "5.6.0"
},
"preferred-install": "dist",
"sort-packages": true
},
},
"autoload": {
"psr-4": {
"GraphQL\\": "src/"
@ -36,6 +34,15 @@
"GraphQL\\Examples\\Blog\\": "examples/01-blog/Blog/"
}
},
"scripts": {
"static-analysis": [
"rm phpstan.phar || true",
"@composer req --ansi --no-interaction --dev phpstan/phpstan-shim",
"cp -f vendor/phpstan/phpstan-shim/phpstan.phar .",
"@composer rem --ansi --dev phpstan/phpstan-shim",
"@php phpstan.phar analyse --ansi -l 1 -c phpstan.neon src"
]
},
"suggest": {
"react/promise": "To leverage async resolving on React PHP platform",
"psr/http-message": "To use standard GraphQL server"

View file

@ -11,6 +11,7 @@
* [DataLoader PHP](https://github.com/overblog/dataloader-php) - as a ready implementation for [deferred resolvers](data-fetching.md#solving-n1-problem)
* [PSR 15 compliant middleware](https://github.com/phps-cans/psr7-middleware-graphql) for the Standard Server (experimental)
* [GraphQL Uploads](https://github.com/Ecodev/graphql-upload) for the Standard Server
* [GraphQL Batch Processor](https://github.com/vasily-kartashov/graphql-batch-processing) - Simple library that provides a builder interface for defining collection, querying, filtering, and post-processing logic of batched data fetches.
# General GraphQL Tools

View file

@ -106,7 +106,7 @@ echo json_encode($output);
Our example is finished. Try it by running:
```sh
php -S localhost:8000 graphql.php
php -S localhost:8080 graphql.php
curl http://localhost:8080 -d '{"query": "query { echo(message: \"Hello World\") }" }'
```

View file

@ -58,4 +58,4 @@ $trackDirective = new Directive([
```
See possible directive locations in
[`GraphQL\Type\Definition\DirectiveLocation`](../reference.md#graphqltypedefinitiondirectivelocation).
[`GraphQL\Language\DirectiveLocation`](../reference.md#graphqllanguagedirectivelocation).

0
phpstan.neon Normal file
View file

View file

@ -265,7 +265,7 @@ class Error extends \Exception implements \JsonSerializable, ClientAware
}, $positions);
} else if ($nodes) {
$this->locations = array_filter(array_map(function ($node) {
if ($node->loc) {
if ($node->loc && $node->loc->source) {
return $node->loc->source->getLocation($node->loc->start);
}
}, $nodes));

View file

@ -2,6 +2,7 @@
namespace GraphQL\Executor;
use GraphQL\Error\Error;
use GraphQL\Executor\Promise\PromiseAdapter;
use GraphQL\Language\AST\FragmentDefinitionNode;
use GraphQL\Language\AST\OperationDefinitionNode;
use GraphQL\Type\Schema;
@ -56,6 +57,11 @@ class ExecutionContext
*/
public $errors;
/**
* @var PromiseAdapter
*/
public $promises;
public function __construct(
$schema,
$fragments,

View file

@ -273,6 +273,11 @@ class Executor
*/
private $promises;
/**
* @var \SplObjectStorage
*/
private $subFieldCache;
/**
* Executor constructor.
*
@ -285,6 +290,7 @@ class Executor
}
$this->exeContext = $context;
$this->subFieldCache = new \SplObjectStorage();
}
/**
@ -311,7 +317,10 @@ class Executor
return null;
})
->then(function ($data) {
return new ExecutionResult((array) $data, $this->exeContext->errors);
if ($data !== null){
$data = (array) $data;
}
return new ExecutionResult($data, $this->exeContext->errors);
});
}
@ -551,7 +560,7 @@ class Executor
switch ($selection->kind) {
case NodeKind::FIELD:
if (!$this->shouldIncludeNode($selection)) {
continue;
continue 2;
}
$name = self::getFieldEntryKey($selection);
if (!isset($fields[$name])) {
@ -563,7 +572,7 @@ class Executor
if (!$this->shouldIncludeNode($selection) ||
!$this->doesFragmentConditionMatch($selection, $runtimeType)
) {
continue;
continue 2;
}
$this->collectFields(
$runtimeType,
@ -575,14 +584,14 @@ class Executor
case NodeKind::FRAGMENT_SPREAD:
$fragName = $selection->name->value;
if (!empty($visitedFragmentNames[$fragName]) || !$this->shouldIncludeNode($selection)) {
continue;
continue 2;
}
$visitedFragmentNames[$fragName] = true;
/** @var FragmentDefinitionNode|null $fragment */
$fragment = isset($exeContext->fragments[$fragName]) ? $exeContext->fragments[$fragName] : null;
if (!$fragment || !$this->doesFragmentConditionMatch($fragment, $runtimeType)) {
continue;
continue 2;
}
$this->collectFields(
$runtimeType,
@ -1308,12 +1317,28 @@ class Executor
&$result
)
{
// Collect sub-fields to execute to complete this value.
$subFieldNodes = new \ArrayObject();
$visitedFragmentNames = new \ArrayObject();
$subFieldNodes = $this->collectSubFields($returnType, $fieldNodes);
return $this->executeFields($returnType, $result, $path, $subFieldNodes);
}
foreach ($fieldNodes as $fieldNode) {
if (isset($fieldNode->selectionSet)) {
/**
* @param ObjectType $returnType
* @param $fieldNodes
* @return \ArrayObject
*/
private function collectSubFields(ObjectType $returnType, $fieldNodes)
{
if (!isset($this->subFieldCache[$returnType])) {
$this->subFieldCache[$returnType] = new \SplObjectStorage();
}
if (!isset($this->subFieldCache[$returnType][$fieldNodes])) {
// Collect sub-fields to execute to complete this value.
$subFieldNodes = new \ArrayObject();
$visitedFragmentNames = new \ArrayObject();
foreach ($fieldNodes as $fieldNode) {
if (!isset($fieldNode->selectionSet)) {
continue;
}
$subFieldNodes = $this->collectFields(
$returnType,
$fieldNode->selectionSet,
@ -1321,9 +1346,9 @@ class Executor
$visitedFragmentNames
);
}
$this->subFieldCache[$returnType][$fieldNodes] = $subFieldNodes;
}
return $this->executeFields($returnType, $result, $path, $subFieldNodes);
return $this->subFieldCache[$returnType][$fieldNodes];
}
/**

View file

@ -6,15 +6,6 @@ use GraphQL\Language\AST\NodeKind;
use GraphQL\Language\AST\NodeList;
use GraphQL\Utils\TypeInfo;
class VisitorOperation
{
public $doBreak;
public $doContinue;
public $removeNode;
}
/**
* Utility for efficient AST traversal and modification.
*
@ -258,6 +249,7 @@ class Visitor
if ($visitFn) {
$result = call_user_func($visitFn, $node, $key, $parent, $path, $ancestors);
$editValue = null;
if ($result !== null) {
if ($result instanceof VisitorOperation) {

View file

@ -0,0 +1,11 @@
<?php
namespace GraphQL\Language;
class VisitorOperation
{
public $doBreak;
public $doContinue;
public $removeNode;
}

View file

@ -71,6 +71,8 @@ class Helper
}
} else if (stripos($contentType, 'application/x-www-form-urlencoded') !== false) {
$bodyParams = $_POST;
} else if (stripos($contentType, 'multipart/form-data') !== false) {
$bodyParams = $_POST;
} else if (null === $contentType) {
throw new RequestError('Missing "Content-Type" header');
} else {

View file

@ -5,7 +5,7 @@ use GraphQL\Error\FormattedError;
use GraphQL\Error\InvariantViolation;
use GraphQL\Executor\ExecutionResult;
use GraphQL\Executor\Promise\Promise;
use GraphQL\Utils;
use GraphQL\Utils\Utils;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Message\StreamInterface;

View file

@ -62,6 +62,11 @@ class FieldDefinition
*/
public $config;
/**
* @var null|callable
*/
public $complexityFn;
/**
* @var OutputType
*/
@ -108,7 +113,7 @@ class FieldDefinition
}
/**
* @param array|Config $field
* @param array $field
* @param string $typeName
* @return FieldDefinition
*/

View file

@ -24,7 +24,7 @@ class InputObjectField
public $description;
/**
* @var callback|InputType
* @var callable|InputType
*/
public $type;

View file

@ -40,6 +40,9 @@ represent free-form human-readable text.';
if ($value === null) {
return 'null';
}
if (is_object($value) && method_exists($value, '__toString')) {
return (string) $value;
}
if (!is_scalar($value)) {
throw new Error("String cannot represent non scalar value: " . Utils::printSafe($value));
}

View file

@ -275,15 +275,35 @@ class SchemaPrinter
return self::printDescriptionWithComments($lines, $indentation, $firstInBlock);
}
$description = ($indentation && !$firstInBlock) ? "\n" : '';
if (count($lines) === 1 && mb_strlen($lines[0]) < 70) {
$description .= $indentation . '"""' . self::escapeQuote($lines[0]) . "\"\"\"\n";
return $description;
$description = ($indentation && !$firstInBlock)
? "\n" . $indentation . '"""'
: $indentation . '"""';
// In some circumstances, a single line can be used for the description.
if (
count($lines) === 1 &&
mb_strlen($lines[0]) < 70 &&
substr($lines[0], -1) !== '"'
) {
return $description . self::escapeQuote($lines[0]) . "\"\"\"\n";
}
$description .= $indentation . "\"\"\"\n";
foreach ($lines as $line) {
$description .= $indentation . self::escapeQuote($line) . "\n";
// Format a multi-line block quote to account for leading space.
$hasLeadingSpace = isset($lines[0]) &&
(
substr($lines[0], 0, 1) === ' ' ||
substr($lines[0], 0, 1) === '\t'
);
if (!$hasLeadingSpace) {
$description .= "\n";
}
$lineLength = count($lines);
for ($i = 0; $i < $lineLength; $i++) {
if ($i !== 0 || !$hasLeadingSpace) {
$description .= $indentation;
}
$description .= self::escapeQuote($lines[$i]) . "\n";
}
$description .= $indentation . "\"\"\"\n";

View file

@ -426,7 +426,7 @@ class OverlappingFieldsCanBeMerged extends AbstractValidationRule
$fragmentNames1Length = count($fragmentNames1);
if ($fragmentNames1Length !== 0) {
$comparedFragments = [];
for ($i = 0; $i < $fragmentNames2Length; $i++) {
for ($i = 0; $i < $fragmentNames1Length; $i++) {
$this->collectConflictsBetweenFieldsAndFragment(
$context,
$conflicts,

View file

@ -49,10 +49,10 @@ class VariablesDefaultValueAllowed extends AbstractValidationRule
return Visitor::skipNode();
},
NodeKind::SELECTION_SET => function(SelectionSetNode $node) use ($context) {
NodeKind::SELECTION_SET => function(SelectionSetNode $node) {
return Visitor::skipNode();
},
NodeKind::FRAGMENT_DEFINITION => function(FragmentDefinitionNode $node) use ($context) {
NodeKind::FRAGMENT_DEFINITION => function(FragmentDefinitionNode $node) {
return Visitor::skipNode();
},
];

View file

@ -732,7 +732,6 @@ class NonNullTest extends \PHPUnit_Framework_TestCase
';
$expected = [
'data' => null,
'errors' => [
FormattedError::create($this->nonNullSyncError->getMessage(), [new SourceLocation(2, 17)])
]
@ -750,7 +749,6 @@ class NonNullTest extends \PHPUnit_Framework_TestCase
$ast = Parser::parse($doc);
$expected = [
'data' => null,
'errors' => [
FormattedError::create($this->nonNullPromiseError->getMessage(), [new SourceLocation(2, 17)]),
]
@ -789,7 +787,6 @@ class NonNullTest extends \PHPUnit_Framework_TestCase
$ast = Parser::parse($doc);
$expected = [
'data' => null,
'errors' => [
[
'debugMessage' => 'Cannot return null for non-nullable field DataType.nonNullPromise.',

View file

@ -69,6 +69,28 @@ class RequestParsingTest extends \PHPUnit_Framework_TestCase
}
}
public function testParsesMultipartFormdataRequest()
{
$query = '{my query}';
$variables = ['test' => 1, 'test2' => 2];
$operation = 'op';
$post = [
'query' => $query,
'variables' => $variables,
'operationName' => $operation
];
$parsed = [
'raw' => $this->parseRawMultipartFormdataRequest($post),
'psr' => $this->parsePsrMultipartFormdataRequest($post)
];
foreach ($parsed as $method => $parsedBody) {
$this->assertValidOperationParams($parsedBody, $query, null, $variables, $operation, $method);
$this->assertFalse($parsedBody->isReadOnly(), $method);
}
}
public function testParsesJSONRequest()
{
$query = '{my query}';
@ -327,6 +349,37 @@ class RequestParsingTest extends \PHPUnit_Framework_TestCase
return $helper->parsePsrRequest($psrRequest);
}
/**
* @param array $postValue
* @return OperationParams|OperationParams[]
*/
private function parseRawMultipartFormDataRequest($postValue)
{
$_SERVER['CONTENT_TYPE'] = 'multipart/form-data; boundary=----FormBoundary';
$_SERVER['REQUEST_METHOD'] = 'POST';
$_POST = $postValue;
$helper = new Helper();
return $helper->parseHttpRequest(function() {
throw new InvariantViolation("Shouldn't read from php://input for multipart/form-data request");
});
}
/**
* @param $postValue
* @return array|Helper
*/
private function parsePsrMultipartFormDataRequest($postValue)
{
$psrRequest = new PsrRequestStub();
$psrRequest->headers['content-type'] = ['multipart/form-data; boundary=----FormBoundary'];
$psrRequest->method = 'POST';
$psrRequest->parsedBody = $postValue;
$helper = new Helper();
return $helper->parsePsrRequest($psrRequest);
}
/**
* @param $getValue
* @return OperationParams

View file

@ -150,6 +150,7 @@ class ScalarSerializationTest extends \PHPUnit_Framework_TestCase
$this->assertSame('true', $stringType->serialize(true));
$this->assertSame('false', $stringType->serialize(false));
$this->assertSame('null', $stringType->serialize(null));
$this->assertSame('2', $stringType->serialize(new ObjectIdStub(2)));
}
public function testSerializesOutputStringsCannotRepresentArray()

View file

@ -11,6 +11,7 @@ use GraphQL\Type\Definition\ObjectType;
use GraphQL\Type\Definition\Type;
use GraphQL\Type\Definition\EnumType;
use GraphQL\Type\Definition\UnionType;
use GraphQL\Utils\BuildSchema;
use GraphQL\Utils\SchemaPrinter;
class SchemaPrinterTest extends \PHPUnit_Framework_TestCase
@ -24,13 +25,13 @@ class SchemaPrinterTest extends \PHPUnit_Framework_TestCase
private function printSingleFieldSchema($fieldConfig)
{
$root = new ObjectType([
'name' => 'Root',
$query = new ObjectType([
'name' => 'Query',
'fields' => [
'singleField' => $fieldConfig
]
]);
return $this->printForTest(new Schema(['query' => $root]));
return $this->printForTest(new Schema(['query' => $query]));
}
/**
@ -42,11 +43,7 @@ class SchemaPrinterTest extends \PHPUnit_Framework_TestCase
'type' => Type::string()
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField: String
}
', $output);
@ -61,11 +58,7 @@ type Root {
'type' => Type::listOf(Type::string())
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField: [String]
}
', $output);
@ -80,11 +73,7 @@ type Root {
'type' => Type::nonNull(Type::string())
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField: String!
}
', $output);
@ -99,11 +88,7 @@ type Root {
'type' => Type::nonNull(Type::listOf(Type::string()))
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField: [String]!
}
', $output);
@ -118,11 +103,7 @@ type Root {
'type' => Type::listOf(Type::nonNull(Type::string()))
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField: [String!]
}
', $output);
@ -137,11 +118,7 @@ type Root {
'type' => Type::nonNull(Type::listOf(Type::nonNull(Type::string())))
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField: [String!]!
}
', $output);
@ -189,11 +166,7 @@ type Root {
'args' => ['argOne' => ['type' => Type::int()]]
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField(argOne: Int): String
}
', $output);
@ -209,11 +182,7 @@ type Root {
'args' => ['argOne' => ['type' => Type::int(), 'defaultValue' => 2]]
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField(argOne: Int = 2): String
}
', $output);
@ -229,11 +198,7 @@ type Root {
'args' => ['argOne' => ['type' => Type::int(), 'defaultValue' => null]]
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField(argOne: Int = null): String
}
', $output);
@ -249,11 +214,7 @@ type Root {
'args' => ['argOne' => ['type' => Type::nonNull(Type::int())]]
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField(argOne: Int!): String
}
', $output);
@ -272,11 +233,7 @@ type Root {
]
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField(argOne: Int, argTwo: String): String
}
', $output);
@ -296,11 +253,7 @@ type Root {
]
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField(argOne: Int = 1, argTwo: String, argThree: Boolean): String
}
', $output);
@ -320,11 +273,7 @@ type Root {
]
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField(argOne: Int, argTwo: String = "foo", argThree: Boolean): String
}
', $output);
@ -344,11 +293,7 @@ type Root {
]
]);
$this->assertEquals('
schema {
query: Root
}
type Root {
type Query {
singleField(argOne: Int, argTwo: String, argThree: Boolean = false): String
}
', $output);
@ -661,6 +606,78 @@ type Query {
', $output);
}
/**
* @it One-line prints a short description
*/
public function testOneLinePrintsAShortDescription()
{
$description = 'This field is awesome';
$output = $this->printSingleFieldSchema([
"type" => Type::string(),
"description" => $description
]);
$this->assertEquals('
type Query {
"""This field is awesome"""
singleField: String
}
', $output);
$recreatedRoot = BuildSchema::build($output)->getTypeMap()['Query'];
$recreatedField = $recreatedRoot->getFields()['singleField'];
$this->assertEquals($description, $recreatedField->description);
}
/**
* @it Does not one-line print a description that ends with a quote
*/
public function testDoesNotOneLinePrintADescriptionThatEndsWithAQuote()
{
$description = 'This field is "awesome"';
$output = $this->printSingleFieldSchema([
"type" => Type::string(),
"description" => $description
]);
$this->assertEquals('
type Query {
"""
This field is "awesome"
"""
singleField: String
}
', $output);
$recreatedRoot = BuildSchema::build($output)->getTypeMap()['Query'];
$recreatedField = $recreatedRoot->getFields()['singleField'];
$this->assertEquals($description, $recreatedField->description);
}
/**
* @it Preserves leading spaces when printing a description
*/
public function testPReservesLeadingSpacesWhenPrintingADescription()
{
$description = ' This field is "awesome"';
$output = $this->printSingleFieldSchema([
"type" => Type::string(),
"description" => $description
]);
$this->assertEquals('
type Query {
""" This field is "awesome"
"""
singleField: String
}
', $output);
$recreatedRoot = BuildSchema::build($output)->getTypeMap()['Query'];
$recreatedField = $recreatedRoot->getFields()['singleField'];
$this->assertEquals($description, $recreatedField->description);
}
/**
* @it Print Introspection Schema
*/