lastdragon-ru / lara-asp-graphql
The Awesome Set of Packages for Laravel - The GraphQL Extensions.
Requires
- php: ^8.2|^8.3
- ext-filter: *
- ext-mbstring: *
- composer/semver: ^3.2
- illuminate/collections: ^10.34.0|^11.0.0
- illuminate/contracts: ^10.34.0|^11.0.0
- illuminate/database: ^10.34.0|^11.0.0
- illuminate/support: ^10.34.0|^11.0.0
- lastdragon-ru/lara-asp-core: 7.1.0
- lastdragon-ru/lara-asp-eloquent: 7.1.0
- lastdragon-ru/lara-asp-graphql-printer: 7.1.0
- lastdragon-ru/lara-asp-serializer: 7.1.0
- nuwave/lighthouse: ^6.5.0
- symfony/deprecation-contracts: ^3.0.0
- symfony/polyfill-php83: ^1.28
- webonyx/graphql-php: ^15.4.0
Requires (Dev)
- ext-pdo_sqlite: *
- illuminate/cache: ^10.34.0|^11.0.0
- laravel/scout: ^9.8.0|^10.0.0
- lastdragon-ru/lara-asp-testing: 7.1.0
- mockery/mockery: ^1.6.5
- orchestra/testbench: ^8.0.0|^9.0.0
- phpunit/phpunit: ^10.1.0|^11.0.0
Suggests
- laravel/scout: Can be used by @searchBy/@sortBy/etc directives to work with Scout Builder.
- dev-main
- 7.x-dev
- 7.1.0
- 7.0.1
- 7.0.0
- 6.x-dev
- 6.4.2
- 6.4.1
- 6.4.0
- 6.3.0
- 6.2.0
- 6.1.0
- 6.0.0
- 5.x-dev
- 5.6.0
- 5.5.0
- 5.4.0
- 5.3.1
- 5.3.0
- 5.2.0
- 5.1.0
- 5.0.0
- 5.0.0-beta.1
- 5.0.0-beta.0
- 4.x-dev
- 4.6.0
- 4.5.2
- 4.5.1
- 4.5.0
- 4.4.0
- 4.3.0
- 4.2.1
- 4.2.0
- 4.1.0
- 4.0.0
- 3.0.0
- 2.x-dev
- 2.1.0
- 2.0.3
- 2.0.2
- 2.0.1
- 2.0.0
- 1.x-dev
- 1.1.2
- 1.1.1
- 1.1.0
- 1.0.4
- 1.0.3
- 1.0.2
- 1.0.1
- 1.0.0
- 0.15.0
- 0.14.1
- 0.14.0
- 0.13.0
- 0.12.0
- 0.11.0
- 0.10.0
- 0.9.0
- 0.8.1
- 0.8.0
- 0.7.0
- 0.6.1
- 0.6.0
- 0.5.0
This package is auto-updated.
Last update: 2025-01-13 11:18:24 UTC
README
This package provides highly powerful @searchBy
, @sortBy
, @stream
directives for lighthouse-php. The @searchBy
directive provides basic conditions like =
, >
, <
, etc, relations, not (<condition>)
, enums, and custom operators support. All are strictly typed so you no need to use Mixed
type anymore. The @sortBy
is not only about standard sorting by columns but also allows use relations. ๐
Requirements
Installation
composer require lastdragon-ru/lara-asp-graphql
Configuration
Config can be used, for example, to customize supported operators for each type. Before this, you need to publish it via the following command, and then you can edit config/lara-asp-graphql.php
.
php artisan vendor:publish --provider=LastDragon_ru\\LaraASP\\GraphQL\\PackageProvider --tag=config
Directives
@searchBy
Probably the most powerful directive to provide search (where
conditions) for your GraphQL queries.
@sortBy
Probably the most powerful directive to provide sort (order by
conditions) for your GraphQL queries.
@stream
๐งช
Unlike the @paginate
(and similar) directive, the @stream
provides a uniform way to perform Offset/Limit and Cursor pagination of Eloquent/Query/Scout builders. Filtering and sorting enabled by default via @searchBy
and @sortBy
directives.
@type
Converts scalar into GraphQL Type. Similar to Lighthouse's @scalar
directive, but uses Laravel Container to resolve instance and also supports PHP enums.
Scalars
Important
You should register the Scalar before use, it can be done via AstManipulator
(useful while AST manipulation), TypeRegistry
, or as a custom scalar inside the Schema:
scalar JsonString @type( class: "LastDragon_ru\\LaraASP\\GraphQL\\Scalars\\JsonStringType" )
JsonString
Represents JSON string.
Scout
Scout is also supported ๐คฉ. You just need to add @search
directive to an argument. Please note that available operators depend on Scout itself.
Please note that if the @search
directive added, the generated query will expect the Scout builder only. So recommended using non-nullable String!
type to avoid using the Eloquent builder (it will happen if the search argument missed or null
; see also lighthouse#2465.
Input type auto-generation
The type used with the Builder directives like @searchBy
/@sortBy
may be Explicit (when you specify the input
name field(where: InputTypeName @searchBy): [Object!]!
) or Implicit (when the _
used, field(where: _ @searchBy): [Object!]!
). They are processing a bit differently.
For Explicit type, all fields except unions and marked as ignored (if supported by the directive) will be included.
For Implicit type, the following rules are applied (in this order; concrete directive may have differences, please check its docs):
- Union? - exclude
- Has
Operator
of the concrete directive? - include - Has
Nuwave\Lighthouse\Support\Contracts\FieldResolver
?- Yes
- Is
Nuwave\Lighthouse\Schema\Directives\RelationDirective
? - Include if is theObject
or list ofObject
- Is
Nuwave\Lighthouse\Schema\Directives\RenameDirective
? - Include if allowed, isscalar
/enum
(notObject
), and no arguments - Otherwise - exclude
- Is
- No
- Is
Object
or has arguments - exclude - Otherwise - include
- Is
- Yes
- Ignored (if supported)? - exclude
When converting the field, some of the original directives will be copied into the newly generated field. For the Explicit type, all directives except operators of other directives will be copied. For Implicit type, you can use Config::$allowedDirectives
setting to control. Be aware of directive locations - the package doesn't perform any checks to ensure that the copied directive allowed on INPUT_FIELD_DEFINITION
, it just copies it as is.
Builder field/column name
By default @searchBy
/@sortBy
will convert nested/related fields into dot string: eg {user: {name: asc}}
will be converted into user.name
. You can redefine this behavior by BuilderFieldResolver
:
// AppProvider $this->app->bind( LastDragon_ru\LaraASP\GraphQL\Builder\Contracts\BuilderFieldResolver::class, MyBuilderFieldResolver::class, );
Builder type detection
Directives like @searchBy
/@sortBy
have a unique set of operators and other features for each type of Builder (Eloquent/Scout/etc). Detection of the current Builder works fine for standard Lighthouse directives like @all
, @paginated
, @search
, etc and relies on proper type hints of Relations/Queries/Resolvers. You may get BuilderUnknown
error if the type hint is missed or the union type is used.
<?php declare(strict_types = 1); namespace App\Models; use Illuminate\Database\Eloquent\Relations\BelongsTo; class Comment extends Model { protected $table = 'comments'; /** * Will NOT work */ public function user() { return $this->belongsTo(User::class); } /** * Must be */ public function user(): BelongsTo { return $this->belongsTo(User::class); } }
If you implement custom directives which internally enhance the Builder (like standard directives do), you may get BuilderUnknown
error because the proper/expected builder type was not detected. In this case, your directive should implement BuilderInfoProvider
interface and to specify the builder type explicitly.
<?php declare(strict_types = 1); namespace App\GraphQL\Directives; use Illuminate\Database\Eloquent\Builder; use LastDragon_ru\LaraASP\GraphQL\Builder\BuilderInfo; use LastDragon_ru\LaraASP\GraphQL\Builder\Contracts\BuilderInfoProvider; use LastDragon_ru\LaraASP\GraphQL\Builder\Contracts\TypeSource; use Nuwave\Lighthouse\Support\Contracts\Directive; use Override; class CustomDirective implements Directive, BuilderInfoProvider { #[Override] public static function definition(): string { return 'directive @custom'; } #[Override] public function getBuilderInfo(TypeSource $source): ?BuilderInfo { return BuilderInfo::create(Builder::class); } public function __invoke(): mixed { return null; } }
Printer
The package provides bindings for Printer
so you can simply use:
<?php declare(strict_types = 1); use LastDragon_ru\LaraASP\Dev\App\Example; use LastDragon_ru\LaraASP\GraphQLPrinter\Contracts\DirectiveFilter; use LastDragon_ru\LaraASP\GraphQLPrinter\Contracts\Printer; use LastDragon_ru\LaraASP\GraphQLPrinter\Settings\DefaultSettings; use Nuwave\Lighthouse\Schema\SchemaBuilder; $schema = app()->make(SchemaBuilder::class)->schema(); $printer = app()->make(Printer::class); $settings = new DefaultSettings(); $printer->setSettings( $settings->setDirectiveDefinitionFilter( new class() implements DirectiveFilter { #[Override] public function isAllowedDirective(string $directive, bool $isStandard): bool { return !in_array($directive, ['eq', 'all', 'find'], true); } }, ), ); Example::raw($printer->print($schema), 'graphql');
Example output
The $printer->print($schema)
is:
""" Use Input as Search Conditions for the current Builder. """ directive @searchBy on | ARGUMENT_DEFINITION directive @searchByOperatorAllOf on | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorAnyOf on | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorCondition on | INPUT_FIELD_DEFINITION directive @searchByOperatorContains on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorEndsWith on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorEqual on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorField on | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorIn on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorLike on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorNot on | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorNotContains on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorNotEndsWith on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorNotEqual on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorNotIn on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorNotLike on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorNotStartsWith on | ENUM | INPUT_FIELD_DEFINITION | SCALAR directive @searchByOperatorStartsWith on | ENUM | INPUT_FIELD_DEFINITION | SCALAR """ Available conditions for `type User` (only one field allowed at a time). """ input SearchByConditionUser { """ Field condition. """ id: SearchByScalarID @searchByOperatorCondition """ Field condition. """ name: SearchByScalarString @searchByOperatorCondition } """ Available conditions for `type User` (only one field allowed at a time). """ input SearchByRootUser { """ All of the conditions must be true. """ allOf: [SearchByRootUser!] @searchByOperatorAllOf """ Any of the conditions must be true. """ anyOf: [SearchByRootUser!] @searchByOperatorAnyOf """ Field. """ field: SearchByConditionUser @searchByOperatorField """ Not. """ not: SearchByRootUser @searchByOperatorNot } """ Available operators for `scalar ID` (only one operator allowed at a time). """ input SearchByScalarID { """ Equal (`=`). """ equal: ID @searchByOperatorEqual """ Within a set of values. """ in: [ID!] @searchByOperatorIn """ Not Equal (`!=`). """ notEqual: ID @searchByOperatorNotEqual """ Outside a set of values. """ notIn: [ID!] @searchByOperatorNotIn } """ Available operators for `scalar String` (only one operator allowed at a time). """ input SearchByScalarString { """ Contains. """ contains: String @searchByOperatorContains """ Ends with a string. """ endsWith: String @searchByOperatorEndsWith """ Equal (`=`). """ equal: String @searchByOperatorEqual """ Within a set of values. """ in: [String!] @searchByOperatorIn """ Like. """ like: String @searchByOperatorLike """ Not contains. """ notContains: String @searchByOperatorNotContains """ Not ends with a string. """ notEndsWith: String @searchByOperatorNotEndsWith """ Not Equal (`!=`). """ notEqual: String @searchByOperatorNotEqual """ Outside a set of values. """ notIn: [String!] @searchByOperatorNotIn """ Not like. """ notLike: String @searchByOperatorNotLike """ Not starts with a string. """ notStartsWith: String @searchByOperatorNotStartsWith """ Starts with a string. """ startsWith: String @searchByOperatorStartsWith } type Query { """ Find a single user by an identifying attribute. """ user( """ Search by primary key. """ id: ID @eq ): User @find """ List multiple users. """ users( where: SearchByRootUser @searchBy ): [User!]! @all } """ Account of a person who utilizes this application. """ type User { """ Unique primary key. """ id: ID! """ Non-unique name. """ name: String! }
Testing Assertions
assertGraphQLIntrospectionEquals
Compares default public schema (as the client sees it through introspection).
assertGraphQLSchemaEquals
Compares default internal schema (with all directives).
assertGraphQLSchemaNoBreakingChanges
Checks that no breaking changes in the default internal schema (with all directives).
assertGraphQLSchemaNoDangerousChanges
Checks that no dangerous changes in the default internal schema (with all directives).
assertGraphQLSchemaValid
Validates default internal schema (with all directives). Faster than lighthouse:validate-schema
command because loads only used directives.
Upgrading
Please follow Upgrade Guide.
Contributing
This package is the part of Awesome Set of Packages for Laravel. Please use the main repository to report issues, send pull requests, or ask questions.