-
-
Notifications
You must be signed in to change notification settings - Fork 266
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
add multiple types of parameter resolution #419
Changes from all commits
377b94a
67ab0a5
5111514
c82fb19
3ad9592
b8ebf74
9047073
07835f3
265dc9c
2bec80e
0fa10db
329b4af
50cfe57
acaf93f
6b33458
b5f4964
5994077
7631c5d
a5b4458
6b2626e
272f2b7
edc0bd2
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -4,10 +4,20 @@ | |
|
||
namespace Rebing\GraphQL\Support; | ||
|
||
use RuntimeException; | ||
use GraphQL\Language\AST\FieldNode; | ||
use GraphQL\Language\AST\ArgumentNode; | ||
use GraphQL\Language\AST\ValueNode; | ||
use GraphQL\Language\AST\IntValueNode; | ||
use GraphQL\Language\AST\VariableNode; | ||
use GraphQL\Language\AST\EnumValueNode; | ||
use GraphQL\Language\AST\ListValueNode; | ||
use GraphQL\Language\AST\NullValueNode; | ||
use GraphQL\Language\AST\FloatValueNode; | ||
use GraphQL\Type\Definition\ResolveInfo; | ||
use GraphQL\Language\AST\ObjectFieldNode; | ||
use GraphQL\Language\AST\ObjectValueNode; | ||
use GraphQL\Language\AST\StringValueNode; | ||
use GraphQL\Language\AST\BooleanValueNode; | ||
use GraphQL\Language\AST\SelectionSetNode; | ||
use GraphQL\Language\AST\FragmentSpreadNode; | ||
use GraphQL\Language\AST\InlineFragmentNode; | ||
|
@@ -114,7 +124,7 @@ private function foldSelectionSet(SelectionSetNode $selectionSet, int $descend): | |
]; | ||
|
||
foreach ($selectionNode->arguments ?? [] as $argumentNode) { | ||
$fields[$name]['args'][$argumentNode->name->value] = $this->getValue($argumentNode); | ||
$fields[$name]['args'][$argumentNode->name->value] = $this->getValue($argumentNode->value); | ||
} | ||
} elseif ($selectionNode instanceof FragmentSpreadNode) { | ||
$spreadName = $selectionNode->name->value; | ||
|
@@ -132,15 +142,68 @@ private function foldSelectionSet(SelectionSetNode $selectionSet, int $descend): | |
return $fields; | ||
} | ||
|
||
private function getValue(ArgumentNode $argumentNode) | ||
private function getValue(ValueNode $value) | ||
{ | ||
$value = $argumentNode->value; | ||
if ($value instanceof VariableNode) { | ||
$variableName = $value->name->value; | ||
|
||
return $this->info->variableValues[$variableName] ?? null; | ||
} | ||
|
||
return $argumentNode->value->value; | ||
if ($value instanceof IntValueNode) { | ||
return (int) $value->value; | ||
} | ||
|
||
if ($value instanceof FloatValueNode) { | ||
return (float) $value->value; | ||
} | ||
|
||
if ($value instanceof StringValueNode) { | ||
return (string) $value->value; | ||
} | ||
|
||
if ($value instanceof BooleanValueNode) { | ||
return (bool) $value->value; | ||
} | ||
|
||
if ($value instanceof EnumValueNode) { | ||
return (string) $value->value; | ||
} | ||
|
||
if ($value instanceof NullValueNode) { | ||
return; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Shouldn't this be more explicit and actually return There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Originally I returned null, but There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. How strange, thanks! |
||
} | ||
|
||
if ($value instanceof ObjectValueNode) { | ||
return $this->getInputObjectValue($value); | ||
} | ||
|
||
if ($value instanceof ListValueNode) { | ||
return $this->getInputListObjectValue($value); | ||
} | ||
|
||
throw new RuntimeException('Failed to resolve unknown ValueNode type'); | ||
} | ||
|
||
private function getInputObjectValue(ObjectValueNode $objectValueNode): array | ||
{ | ||
$value = []; | ||
foreach ($objectValueNode->fields->getIterator() as $item) { | ||
if ($item instanceof ObjectFieldNode) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Not sure about this. The code makes it look like How can we improve this? Should we remove the check? Or add an Because as it is right now, if that case ever would happen, we silently would carry on. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
|
||
$value[$item->name->value] = $this->getValue($item->value); | ||
} | ||
} | ||
|
||
return $value; | ||
} | ||
|
||
private function getInputListObjectValue(ListValueNode $listValueNode): array | ||
{ | ||
$value = []; | ||
foreach ($listValueNode->values as $valueNode) { | ||
$value[] = $this->getValue($valueNode); | ||
} | ||
|
||
return $value; | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Rebing\GraphQL\Tests\Database\SelectFields\ValidateDiffNodeTests; | ||
|
||
use Rebing\GraphQL\Support\EnumType; | ||
|
||
class EpisodeEnum extends EnumType | ||
{ | ||
protected $attributes = [ | ||
'name' => 'Episode', | ||
'description' => 'The types of demographic elements', | ||
'values' => [ | ||
'NEWHOPE' => 'NEWHOPE', | ||
'EMPIRE' => 'EMPIRE', | ||
'JEDI' => 'JEDI', | ||
], | ||
]; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Rebing\GraphQL\Tests\Database\SelectFields\ValidateDiffNodeTests; | ||
|
||
use GraphQL\Type\Definition\Type; | ||
use Rebing\GraphQL\Support\InputType; | ||
|
||
class FilterInput extends InputType | ||
{ | ||
protected $attributes = [ | ||
'name' => 'Filter', | ||
'description' => 'filter object', | ||
]; | ||
|
||
public function fields(): array | ||
{ | ||
return [ | ||
'body' => [ | ||
'type' => Type::string(), | ||
], | ||
'id' => [ | ||
'type' => Type::int(), | ||
], | ||
'title' => [ | ||
'type' => Type::string(), | ||
], | ||
]; | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Rebing\GraphQL\Tests\Database\SelectFields\ValidateDiffNodeTests; | ||
|
||
use GraphQL\Type\Definition\Type; | ||
use Rebing\GraphQL\Tests\Support\Models\Post; | ||
use Rebing\GraphQL\Support\Type as GraphQLType; | ||
|
||
class PostType extends GraphQLType | ||
{ | ||
protected $attributes = [ | ||
'name' => 'Post', | ||
'model' => Post::class, | ||
]; | ||
|
||
public function fields(): array | ||
{ | ||
return [ | ||
'body' => [ | ||
'type' => Type::string(), | ||
], | ||
'id' => [ | ||
'type' => Type::ID(), | ||
], | ||
]; | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,86 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Rebing\GraphQL\Tests\Database\SelectFields\ValidateDiffNodeTests; | ||
|
||
use PHPUnit\Framework\Assert; | ||
use GraphQL\Type\Definition\Type; | ||
use Rebing\GraphQL\Support\Facades\GraphQL; | ||
use Rebing\GraphQL\Tests\Support\Models\User; | ||
use Rebing\GraphQL\Support\Type as GraphQLType; | ||
|
||
class UserType extends GraphQLType | ||
{ | ||
protected $attributes = [ | ||
'name' => 'User', | ||
'model' => User::class, | ||
]; | ||
|
||
public function fields(): array | ||
{ | ||
return [ | ||
'id' => [ | ||
'type' => Type::int(), | ||
], | ||
'name' => [ | ||
'type' => Type::string(), | ||
], | ||
'posts' => [ | ||
'type' => Type::listOf(Type::nonNull(GraphQL::type('Post'))), | ||
'args' => [ | ||
'id' => [ | ||
'type' => Type::int(), | ||
], | ||
'name' => [ | ||
'type' => Type::string(), | ||
], | ||
'price' => [ | ||
'type' => Type::float(), | ||
], | ||
'status' => [ | ||
'type' => Type::boolean(), | ||
], | ||
'flag' => [ | ||
'type' => Type::string(), | ||
], | ||
'author' => [ | ||
'type' => GraphQL::type('Episode'), | ||
], | ||
'post' => [ | ||
'type' => GraphQL::type('Filter'), | ||
], | ||
'keywords' => [ | ||
'type' => Type::listOf(Type::string()), | ||
], | ||
'customType' => [ | ||
'type' => GraphQL::type('MyCustomScalarString'), | ||
], | ||
], | ||
'query' => function (array $args, $query) { | ||
$expectedQueryArgs = [ | ||
'id' => 2, | ||
'name' => 'tom', | ||
'price' => 1.3, | ||
'status' => false, | ||
'flag' => null, | ||
'author' => 'EMPIRE', | ||
'post' => [ | ||
'id' => 2, | ||
'body' => 'body2', | ||
], | ||
'keywords' => [ | ||
'key4', | ||
'key5', | ||
'key6', | ||
], | ||
'customType' => 'custom string', | ||
]; | ||
Assert::assertSame($expectedQueryArgs, $args); | ||
|
||
return $query; | ||
}, | ||
], | ||
]; | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,87 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace Rebing\GraphQL\Tests\Database\SelectFields\ValidateDiffNodeTests; | ||
|
||
use PHPUnit\Framework\Assert; | ||
use GraphQL\Type\Definition\Type; | ||
use Rebing\GraphQL\Support\Query; | ||
use GraphQL\Type\Definition\ResolveInfo; | ||
use Rebing\GraphQL\Support\SelectFields; | ||
use Rebing\GraphQL\Support\Facades\GraphQL; | ||
use Rebing\GraphQL\Tests\Support\Models\User; | ||
|
||
class UsersQuery extends Query | ||
{ | ||
protected $attributes = [ | ||
'name' => 'users', | ||
]; | ||
|
||
public function args(): array | ||
{ | ||
return [ | ||
'id' => [ | ||
'type' => Type::int(), | ||
], | ||
'name' => [ | ||
'type' => Type::string(), | ||
], | ||
'price' => [ | ||
'type' => Type::float(), | ||
], | ||
'status' => [ | ||
'type' => Type::boolean(), | ||
], | ||
'flag' => [ | ||
'type' => Type::string(), | ||
], | ||
'author' => [ | ||
'type' => GraphQL::type('Episode'), | ||
], | ||
'post' => [ | ||
'type' => GraphQL::type('Filter'), | ||
], | ||
'keywords' => [ | ||
'type' => Type::listOf(Type::string()), | ||
], | ||
'customType' => [ | ||
'type' => GraphQL::type('MyCustomScalarString'), | ||
], | ||
]; | ||
} | ||
|
||
public function type(): Type | ||
{ | ||
return Type::nonNull(Type::listOf(Type::nonNull(GraphQL::type('User')))); | ||
} | ||
|
||
public function resolve($root, $args, $context, ResolveInfo $resolveInfo, \Closure $getSelectFields) | ||
{ | ||
/** @var SelectFields $fields */ | ||
$fields = $getSelectFields(); | ||
$expectedQueryArgs = [ | ||
'id' => 1, | ||
'name' => 'john', | ||
'price' => 1.2, | ||
'status' => true, | ||
'flag' => null, | ||
'author' => 'NEWHOPE', | ||
'post' => [ | ||
'body' => 'body', | ||
'id' => 1, | ||
], | ||
'keywords' => [ | ||
'key1', | ||
'key2', | ||
'key3', | ||
], | ||
'customType' => 'hello world', | ||
]; | ||
Assert::assertSame($expectedQueryArgs, $args); | ||
|
||
return User::select($fields->getSelect()) | ||
->with($fields->getRelations()) | ||
->get(); | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This pretty cool and a very clear indicator that the current code was broken :/ Great 👍