GraphQL is a query language made to communicate with an API and therefore is an alternative to REST.
It has some advantages compared to REST: it solves the over-fetching or under-fetching of data, is strongly typed, and is capable of retrieving multiple and nested data in one go, but it also comes with drawbacks. For example it creates overhead depending on the request.
API Platform creates a REST API by default. But you can choose to enable GraphQL as well.
Once enabled, you have nothing to do: your schema describing your API is automatically built and your GraphQL endpoint is ready to go!
To enable GraphQL and its IDE (GraphiQL and GraphQL Playground) in your API, simply require the graphql-php package using Composer and clear the cache one more time:
docker compose exec php sh -c '
composer require webonyx/graphql-php
bin/console cache:clear
'
You can now use GraphQL at the endpoint: https://localhost:8443/graphql
.
Note: If you used Symfony Flex to install API Platform, URLs will be prefixed with /api
by default. For example, the GraphQL endpoint will be: https://localhost:8443/api/graphql
.
Sometimes you may want to have the GraphQL endpoint at a different location. This can be done by manually configuring the GraphQL controller.
# api/config/routes.yaml
api_graphql_entrypoint:
path: /api/graphql
controller: api_platform.graphql.action.entrypoint
# ...
Change /api/graphql
to the URI you wish the GraphQL endpoint to be accessible on.
If Twig is installed in your project, go to the GraphQL endpoint with your browser. You will see a nice interface provided by GraphiQL to interact with your API.
The GraphiQL IDE can also be found at /graphql/graphiql
.
If you need to disable it, it can be done in the configuration:
# api/config/packages/api_platform.yaml
api_platform:
graphql:
graphiql:
enabled: false
# ...
If you want to add a different location besides /graphql/graphiql
, you can do it like this:
# app/config/routes.yaml
graphiql:
path: /docs/graphiql
controller: api_platform.graphql.action.graphiql
Another IDE is by default included in API Platform: GraphQL Playground.
It can be found at /graphql/graphql_playground
.
You can disable it if you want in the configuration:
# api/config/packages/api_platform.yaml
api_platform:
graphql:
graphql_playground:
enabled: false
# ...
You can add a different location besides /graphql/graphql_playground
:
# app/config/routes.yaml
graphql_playground:
path: /docs/graphql_playground
controller: api_platform.graphql.action.graphql_playground
When going to the GraphQL endpoint, you can choose to launch the IDE you want.
# api/config/packages/api_platform.yaml
api_platform:
graphql:
# Choose between graphiql or graphql-playground
default_ide: graphql-playground
# ...
You can also disable this feature by setting the configuration value to false
.
# api/config/packages/api_platform.yaml
api_platform:
graphql:
default_ide: false
# ...
application/graphql
Content-TypeIf you wish to send a POST request using the application/graphql
Content-Type,
you need to enable it in the allowed formats of API Platform:
# api/config/packages/api_platform.yaml
api_platform:
formats:
# ...
graphql: ['application/graphql']
To understand what an operation is, please refer to the operations documentation.
For GraphQL, the operations are defined by using the Query
, QueryCollection
, Mutation
, DeleteMutation
and Subscription
attributes.
By default, the following operations are enabled:
Query
QueryCollection
Mutation(name: 'create')
Mutation(name: 'update')
DeleteMutation(name: 'delete')
You can of course disable or configure these operations.
For instance, in the following example, only the query of an item and the create mutation are enabled:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\ApiResource;
#[ApiResource(graphQlOperations: [
new Query(),
new Mutation(name: 'create')
])]
class Book
{
// ...
}
If you don’t know what queries are yet, please read the documentation about them.
For each resource, two queries are available: one for retrieving an item and the other one for the collection.
For example, if you have a Book
resource, the queries book
and books
can be used.
When querying an item, you need to pass an identifier as argument. Following the GraphQL Global Object Identification Specification, the identifier needs to be globally unique. In API Platform, this argument is represented as an IRI (Internationalized Resource Identifier).
For example, to query a book having as identifier 89
, you have to run the following:
{
book(id: "/books/89") {
title
isbn
}
}
Note that in this example, we’re retrieving two fields: title
and isbn
.
To create a custom query, first of all you need to create its resolver.
If you want a custom query for a collection, create a class like this:
<?php
// api/src/Resolver/BookCollectionResolver.php
namespace App\Resolver;
use ApiPlatform\GraphQl\Resolver\QueryCollectionResolverInterface;
use App\Entity\Book;
final class BookCollectionResolver implements QueryCollectionResolverInterface
{
/**
* @param iterable<Book> $collection
*
* @return iterable<Book>
*/
public function __invoke(iterable $collection, array $context): iterable
{
// Query arguments are in $context['args'].
foreach ($collection as $book) {
// Do something with the book.
}
return $collection;
}
}
If you use autoconfiguration (the default Symfony configuration) in your application, then you are done!
Else, you need to tag your resolver like this:
# api/config/services.yaml
services:
# ...
App\Resolver\BookCollectionResolver:
tags:
- { name: api_platform.graphql.query_resolver }
The resolver for an item is very similar:
<?php
// api/src/Resolver/BookResolver.php
namespace App\Resolver;
use ApiPlatform\GraphQl\Resolver\QueryItemResolverInterface;
use App\Entity\Book;
final class BookResolver implements QueryItemResolverInterface
{
/**
* @param Book|null $item
*
* @return Book
*/
public function __invoke($item, array $context)
{
// Query arguments are in $context['args'].
// Do something with the book.
// Or fetch the book if it has not been retrieved.
return $item;
}
}
Note that you will receive the retrieved item or not in this resolver depending on how you configure your query in your resource.
Since the resolver is a service, you can inject some dependencies and fetch your item in the resolver if you want.
If you don’t use autoconfiguration, don’t forget to tag your resolver with api_platform.graphql.query_resolver
.
Now that your resolver is created and registered, you can configure your custom query and link its resolver.
In your resource, add the following:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\DeleteMutation;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
use App\Resolver\BookCollectionResolver;
use App\Resolver\BookResolver;
#[ApiResource(graphQlOperations: [
new Query(),
new QueryCollection(),
new Mutation(name: 'create'),
new Mutation(name: 'update'),
new DeleteMutation(name: 'delete'),
new Query(name: 'retrievedQuery', resolver: BookResolver::class),
new Query(
name: 'notRetrievedQuery',
resolver: BookResolver::class,
args: []
),
new Query(
name: 'withDefaultArgsNotRetrievedQuery',
resolver: BookResolver::class,
read: false
),
new Query(
name: 'withCustomArgsQuery',
resolver: BookResolver::class,
args: [
'id' => ['type' => 'ID!'],
'log' => ['type' => 'Boolean!', 'description' => 'Is logging activated?'],
'logDate' => ['type' => 'DateTime']
]
),
new QueryCollection(name: 'collectionQuery', resolver: BookCollectionResolver::class),
])]
class Book
{
// ...
}
Note that you need to explicitly add the auto-generated queries and mutations if they are needed when configuring custom queries, like it’s done for the operations.
As you can see, it’s possible to define your own arguments for your custom queries.
They are following the GraphQL type system.
If you don’t define the args
property, it will be the default ones (for example id
for an item).
If you don’t want API Platform to retrieve the item for you, disable the read
stage like in withDefaultArgsNotRetrievedQuery
.
Some other stages can be disabled.
Another option would be to make sure there is no id
argument.
This is the case for notRetrievedQuery
(empty args).
Conversely, if you need to add custom arguments, make sure id
is added among the arguments if you need the item to be retrieved automatically.
Note also that:
DateTime
).The arguments you have defined or the default ones and their value will be in $context['args']
of your resolvers.
You custom queries will be available like this:
{
retrievedQueryBook(id: "/books/56") {
title
}
notRetrievedQueryBook {
title
}
withDefaultArgsNotRetrievedQueryBook(id: "/books/56") {
title
}
withCustomArgsQueryBook(id: "/books/23", log: true, logDate: "2019-12-20") {
title
}
collectionQueryBooks {
edges {
node {
title
}
}
}
}
If you don’t know what mutations are yet, the documentation about them is here.
For each resource, three mutations are available:
Mutation(name: 'create')
for creating a new resourceMutation(name: 'update')
for updating an existing resourceDeleteMutation(name: 'delete')
for deleting an existing resourceWhen updating or deleting a resource, you need to pass the IRI of the resource as argument. See Global Object Identifier for more information.
Following the Relay Input Object Mutations Specification,
you can pass a clientMutationId
as argument and can ask its value as a field.
For example, if you delete a book:
mutation DeleteBook($id: ID!, $clientMutationId: String!) {
deleteBook(input: {id: $id, clientMutationId: $clientMutationId}) {
clientMutationId
}
}
Creating custom mutations is comparable to creating custom queries.
Create your resolver:
<?php
// api/src/Resolver/BookMutationResolver.php
namespace App\Resolver;
use ApiPlatform\GraphQl\Resolver\MutationResolverInterface;
use App\Entity\Book;
final class BookMutationResolver implements MutationResolverInterface
{
/**
* @param Book|null $item
*
* @return Book
*/
public function __invoke($item, array $context)
{
// Mutation input arguments are in $context['args']['input'].
// Do something with the book.
// Or fetch the book if it has not been retrieved.
// The returned item will pe persisted.
return $item;
}
}
As you can see, depending on how you configure your custom mutation in the resource, the item is retrieved or not.
For instance, if you don’t set an id
argument or if you disable the read
or the deserialize
stage (other stages can also be disabled),
the received item will be null
.
Likewise, if you don’t want your item to be persisted by API Platform,
you can return null
instead of the mutated item (be careful: the response will also be null
) or disable the write
stage.
Don’t forget the resolver is a service and you can inject the dependencies you want.
If you don’t use autoconfiguration, add the tag api_platform.graphql.mutation_resolver
to the resolver service.
Now in your resource:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\DeleteMutation;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
use App\Resolver\BookMutationResolver;
#[ApiResource(
graphQlOperations: [
new Query(),
new QueryCollection(),
new Mutation(name: 'create'),
new Mutation(name: 'update'),
new DeleteMutation(name: 'delete'),
new Mutation(name: 'mutation', resolver: BookMutationResolver::class),
new Mutation(
name: 'withCustomArgsMutation',
resolver: BookMutationResolver::class,
args: [
'sendMail' => [
'type' => 'Boolean!',
'description' => 'Send a mail?'
]
]
),
new Mutation(
name: 'disabledStagesMutation',
resolver: BookMutationResolver::class,
deserialize: false,
write: false
)
]
)]
class Book
{
// ...
}
Note that you need to explicitly add the auto-generated queries and mutations if they are needed when configuring custom mutations, like it’s done for the operations.
As the custom queries, you can define your own arguments if you don’t want to use the default ones (extracted from your resource).
The only difference with them is that, even if you define your own arguments, the clientMutationId
will always be set.
The arguments will be in $context['args']['input']
of your resolvers.
Your custom mutations will be available like this:
{
mutation {
mutationBook(input: {id: "/books/18", title: "The Fitz and the Fool"}) {
book {
title
}
}
}
mutation {
withCustomArgsMutationBook(input: {sendMail: true, clientMutationId: "myId"}) {
book {
title
}
clientMutationId
}
}
mutation {
disabledStagesMutationBook(input: {id: "/books/18", title: "The Fitz and the Fool"}) {
book {
title
}
clientMutationId
}
}
}
Subscriptions are an RFC to allow a client to receive pushed realtime data from the server.
In API Platform, the built-in subscription support is handled by using Mercure as its underlying protocol.
To enable update subscriptions for a resource, these conditions have to be met:
update
mutation needs to be enabled for the resource.For instance, your resource should look like this:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Subscription;
#[ApiResource(mercure: true, graphQlOperations: [
new Mutation(name: 'update'),
new Subscription()
])]
class Book
{
// ...
}
Doing a subscription is very similar to doing a query:
{
subscription {
updateBookSubscribe(input: {id: "/books/1", clientSubscriptionId: "myId"}) {
book {
title
isbn
}
mercureUrl
clientSubscriptionId
}
}
}
As you can see, you need to pass the IRI of the resource as argument. See Global Object Identifier for more information.
You can also pass clientSubscriptionId
as argument and can ask its value as a field.
In the payload of the subscription, the given fields of the resource will be the fields you subscribe to: if any of these fields is updated, you will be pushed their updated values.
The mercureUrl
field is the Mercure URL you need to use to subscribe to the updates on the client-side.
On the client-side, you will receive the pushed updated data like you would receive the updated data if you did an update
mutation.
For instance, you could receive a JSON payload like this:
{
"book": {
"title": "Updated title",
"isbn": "978-6-6344-4051-1"
}
}
Internally, API Platform stores the subscriptions in a cache, using the Symfony Cache.
The cache is named api_platform.graphql.cache.subscription
and the subscription keys are generated from the subscription payload by using a SHA-256 hash.
It’s recommended to use an adapter like Redis for this cache.
API Platform resolves the queries and mutations by using its own resolvers.
Even if you create your custom queries or your custom mutations, these resolvers will be used and yours will be called at the right time.
Each resolver follows a workflow composed of stages.
The schema below describes them:
Each stage corresponds to a service. It means you can take control of the workflow wherever you want by decorating them!
Here is an example of the decoration of the write stage, for instance if you want to persist your data as you want.
Create your WriteStage:
<?php
namespace App\Stage;
use ApiPlatform\GraphQl\Resolver\Stage\WriteStageInterface;
final class WriteStage implements WriteStageInterface
{
private $writeStage;
public function __construct(WriteStageInterface $writeStage)
{
$this->writeStage = $writeStage;
}
/**
* {@inheritdoc}
*/
public function __invoke($data, string $resourceClass, string $operationName, array $context)
{
// You can add pre-write code here.
// Call the decorated write stage (this syntax calls the __invoke method).
$writtenObject = ($this->writeStage)($data, $resourceClass, $operationName, $context);
// You can add post-write code here.
return $writtenObject;
}
}
Decorate the API Platform stage service:
# api/config/services.yaml
services:
# ...
'App\Stage\WriteStage':
decorates: api_platform.graphql.resolver.stage.write
If you need to, you can disable some stages done by the resolvers, for instance if you don’t want your data to be validated.
The following table lists the stages you can disable in your resource configuration.
Attribute | Type | Default | Description |
---|---|---|---|
read | bool | true | Enables or disables the reading of data |
deserialize | bool | true | Enables or disables the deserialization of data (mutation only) |
validate | bool | true | Enables or disables the validation of the denormalized data (mutation only) |
write | bool | true | Enables or disables the writing of data into the persistence system (mutation only) |
serialize | bool | true | Enables or disables the serialization of data |
A stage can be disabled at the operation level:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
#[ApiResource(graphQlOperations: [
new Query(),
new QueryCollection(),
new Mutation(name: 'create', write: false)
])]
class Book
{
// ...
}
Or at the resource attributes level (will be also applied in REST and for all operations):
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
#[ApiResource(write: false, graphQlOperations: [
new Query(),
new QueryCollection(),
new Mutation(name: 'create')
])]
class Book
{
// ...
}
No events are sent by the resolvers in API Platform. If you want to add your custom logic, decorating the stages is the recommended way to do it.
However, if you really want to use events, you can by installing a bundle dispatching events before and after the stages.
Filters are supported out-of-the-box. Follow the filters documentation and your filters will be available as arguments of queries.
However you don’t necessarily have the same needs for your GraphQL endpoint as for your REST one.
In the QueryCollection
attribute, you can choose to decorrelate the GraphQL filters.
In order to keep the default behavior (possibility to fetch, delete, update or create), define all the auto-generated operations (Query
,QueryCollection
, DeleteMutation
, and the update
and create
Mutation
).
For example, this entity will have a search filter for REST and a date filter for GraphQL:
<?php
// api/src/Entity/Offer.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\DeleteMutation;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
#[ApiResource(filters: ['offer.search_filter'], graphQlOperations: [
new Query(),
new QueryCollection(filters: ['offer.date_filter']),
new Mutation(name: 'create'),
new Mutation(name: 'update'),
new DeleteMutation(name: 'delete')
])]
class Offer
{
// ...
}
Some filters like the exists filter or the order filter take a list of key / value as arguments.
The first syntax coming to mind to use them is to write:
{
offers(order: {id: "ASC", name: "DESC"}) {
edges {
node {
id
name
}
}
}
}
However this syntax has a problematic issue: it doesn’t keep the order of the arguments. These filters usually need a proper order to give results as expected.
That’s why this syntax needs to be used instead:
{
offers(order: [{id: "ASC"}, {name: "DESC"}]) {
edges {
node {
id
name
}
}
}
}
Since a list is used for the arguments, the order is preserved.
Unlike for REST, all built-in filters support nested properties using the underscore (_
) syntax instead of the dot (.
) syntax, e.g.:
<?php
// api/src/Entity/Offer.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiFilter;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Doctrine\Orm\Filter\OrderFilter;
use ApiPlatform\Doctrine\Orm\Filter\SearchFilter;
#[ApiResource]
#[ApiFilter(OrderFilter::class, properties: ['product.releaseDate'])]
#[ApiFilter(SearchFilter::class, properties: ['product.color' => 'exact'])]
class Offer
{
// ...
}
The above allows you to find offers by their respective product’s color like for the REST Api. You can then filter using the following syntax:
{
offers(product_color: "red") {
edges {
node {
id
product {
name
color
}
}
}
}
}
Or order your results like:
{
offers(order: [{product_releaseDate: "DESC"}]) {
edges {
node {
id
product {
name
color
}
}
}
}
}
Another difference with the REST API filters is that the keyword _list
must be used instead of the traditional []
to filter over multiple values.
For example, if you want to search the offers with a green or a red product you can use the following syntax:
{
offers(product_color_list: ["red", "green"]) {
edges {
node {
id
product {
name
color
}
}
}
}
}
API Platform natively enables a cursor-based pagination for collections. It supports GraphQL’s Complete Connection Model and is compatible with GraphQL Cursor Connections Specification.
A page-based pagination can also be enabled per resource or per operation.
Here is an example query leveraging the pagination system:
{
offers(first: 10, after: "cursor") {
totalCount
pageInfo {
endCursor
hasNextPage
}
edges {
cursor
node {
id
}
}
}
}
Two pairs of parameters work with the query:
first
and after
;last
and before
.More precisely:
first
corresponds to the items per page starting from the beginning;
after
corresponds to the cursor
from which the items are returned.
last
corresponds to the items per page starting from the end;
before
corresponds to the cursor
from which the items are returned, from a backwards point of view.
The current page always has a startCursor
and an endCursor
, present in the pageInfo
field.
To get the next page, you would add the endCursor
from the current page as the after
parameter.
{
offers(first: 10, after: "endCursor") {
}
}
For the previous page, you would add the startCursor
from the current page as the before
parameter.
{
offers(last: 10, before: "startCursor") {
}
}
How do you know when you have reached the last page? It is the aim of the property hasNextPage
or hasPreviousPage
in pageInfo
.
When it is false, you know it is the last page and moving forward or backward will give you an empty result.
In order to use the page-based pagination, you need to enable it in the resource.
For instance at the operation level:
<?php
// api/src/Entity/Offer.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\DeleteMutation;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
#[ApiResource(graphQlOperations: [
new Query(),
new QueryCollection(paginationType: 'page'),
new Mutation(name: 'create'),
new Mutation(name: 'update'),
new DeleteMutation(name: 'delete')
])]
class Offer
{
// ...
}
Or if you want to do it at the resource level:
<?php
// api/src/Entity/Offer.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
#[ApiResource(paginationType: 'page')]
class Offer
{
// ...
}
Once enabled, a page
filter will be available in the collection query (its name can be changed in the configuration) and an itemsPerPage
filter will be available too if client-side-pagination is enabled.
A paginationInfo
field can be queried to obtain the following information:
itemsPerPage
: the number of items per page. To change it, follow the pagination documentation.lastPage
: the last page of the collection.totalCount
: the total number of items in the collection.The collection items data are available in the collection
field.
An example of a query:
{
offers(page: 3, itemsPerPage: 15) {
collection {
id
}
paginationInfo {
itemsPerPage
lastPage
totalCount
}
}
}
See also the pagination documentation.
The pagination can be disabled for all GraphQL resources using this configuration:
# api/config/packages/api_platform.yaml
api_platform:
graphql:
collection:
pagination:
enabled: false
It can also be disabled for a specific resource (REST and GraphQL):
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
#[ApiResource(paginationEnabled: false)]
class Book
{
// ...
}
You can also disable the pagination for a specific collection operation:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
#[ApiResource(graphQlOperations: [new QueryCollection(paginationEnabled: false)])]
class Book
{
// ...
}
Partial pagination is possible with GraphQL.
When enabled, backwards pagination will not be possible, and the hasNextPage
information will be always false
.
To add a security layer to your queries and mutations, follow the security documentation.
The REST security configuration and the GraphQL one are not correlated.
If you have only some parts differing between REST and GraphQL, you have to redefine the common parts anyway.
In the example below, we want the same security rules as we have in REST, but we also want to allow an admin to delete a book only in GraphQL.
Please note that, it’s not possible to update a book in GraphQL because the update
operation is not defined.
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\Get;
use ApiPlatform\Metadata\GraphQl\DeleteMutation;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
use ApiPlatform\Metadata\Post;
#[ApiResource(
security: "is_granted('ROLE_USER')",
operations: [
new Get(security: "is_granted('ROLE_USER') and object.owner == user", securityMessage: 'Sorry, but you are not the book owner.'),
new Post(security: "is_granted('ROLE_ADMIN')", securityMessage: 'Only admins can add books.')
],
graphQlOperations: [
new Query(security: "is_granted('ROLE_USER') and object.owner == user"),
new QueryCollection(security: "is_granted('ROLE_ADMIN')"),
new DeleteMutation(name: 'delete', security: "is_granted('ROLE_ADMIN')"),
new Mutation(name: 'create', security: "is_granted('ROLE_ADMIN')")
]
)]
class Book
{
// ...
}
You may want to limit access to certain resource properties with a security expression. This can be done with the ApiProperty
security
attribute.
Note: adding the ApiProperty
security
expression to a GraphQL property will automatically make the GraphQL property type nullable (if it wasn’t already).
This is because null
is returned as the property value if access is denied via the security
expression.
In GraphQL, it’s possible to expose associations - allowing nested querying.
For example, associations can be made with Doctrine ORM’s OneToMany
, ManyToOne
, ManyToMany
, etc.
It’s important to note that the security defined on resource operations applies only to the exposed query/mutation endpoints (e.g. Query.users
, Mutation.updateUser
, etc.).
Resource operation security is defined via the security
attribute for each operation defined on the resource.
This security is not applied to exposed associations.
Associations can instead be secured with the ApiProperty
security
attribute. This provides the flexibility to have different security depending on where an association is exposed.
To prevent traversal attacks, you should ensure that any exposed associations are secured appropriately.
A traversal attack is where a user can gain unintended access to a resource by querying nested associations, gaining access to a resource that prevents direct access (via the query endpoint).
For example, a user may be denied using Query.getUser
to get a user, but is able to access the user through an association on an object that they do have access to (e.g. document.createdBy
).
The following example shows how associations can be secured:
<?php
// api/src/Entity/User.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiProperty;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
use Doctrine\ORM\Mapping as ORM;
/**
* @ORM\Entity
*/
#[ApiResource(graphQlOperations: [
new Query(security: 'is_granted("VIEW", object)'),
new QueryCollection(security: 'is_granted("ROLE_ADMIN")')
])]
class User
{
// ...
/**
* @ORM\ManyToMany(targetEntity=Document::class, mappedBy="viewers")
*/
#[ApiProperty(security: 'is_granted("VIEW", object)')]
private Collection $viewableDocuments;
/**
* @ORM\Column(type="string", length=180, unique=true)
*/
#[ApiProperty(security: 'is_granted("ROLE_ADMIN")')]
private string $email;
}
<?php
// api/src/Entity/Document.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiProperty;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
use Doctrine\ORM\Mapping as ORM;
/**
* @ORM\Entity
*/
#[ApiResource(graphQlOperations: [
new Query(security: 'is_granted("VIEW", object)'),
new QueryCollection(security: 'is_granted("ROLE_ADMIN")')
])]
class Document
{
// ...
/**
* @ORM\ManyToMany(targetEntity=User::class, inversedBy="viewableDocuments")
*/
#[ApiProperty(security: 'is_granted("VIEW", object)')]
private Collection $viewers;
/**
* @ORM\ManyToOne(targetEntity=User::class)
*/
#[ApiProperty(security: 'is_granted("VIEW", object)')]
protected ?User $createdBy = null;
}
The above example only allows admins to see the full collection of each resource (QueryCollection
).
Users must be granted the VIEW
attribute on a resource to be able to query it directly (Query
) - which would use a Voter
to make this decision.
Similar to Query
, all associations are secured, requiring VIEW
access on the parent object (not on the association).
This means that a user with VIEW
access to a Document
is able to see all users who are in the viewers
collection, as well as the createdBy
association.
This may be a little too open, so you could instead do a role check here to only allow admins to access these fields, or check for a different attribute that could be implemented in the voter (e.g. VIEW_CREATED_BY
.)
Alternatively, you could still expose the users, but limit the visible fields by limiting access with ApiProperty
security
(such as the User::$email
property above) or with dynamic serializer groups.
You may want to restrict some resource’s attributes to your GraphQL clients.
As described in the serialization process documentation, you can use serialization groups to expose only the attributes you want in queries or in mutations.
If the (de)normalization context between GraphQL and REST is different, use the (de)normalizationContext
key to change it in each query and mutations.
Note that:
The following example shows you what can be done:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Mutation;
use ApiPlatform\Metadata\GraphQl\Query;
use ApiPlatform\Metadata\GraphQl\QueryCollection;
use Symfony\Component\Serializer\Annotation\Groups;
#[ApiResource(
normalizationContext: ['groups' => ['read']],
denormalizationContext: ['groups' => ['write']],
graphQlOperations: [
new Query(normalizationContext: ['groups' => ['query']]),
new QueryCollection(normalizationContext: ['groups' => ['query_collection']])
new Mutation(
name: 'create',
normalizationContext: ['groups' => ['query_collection']],
denormalizationContext: ['groups' => ['mutation']]
)
]
)]
class Book
{
// ...
#[Groups(['read', 'write', 'query', 'query_collection'])]
public $title;
#[Groups(['read', 'mutation', 'query'])]
public $author;
// ...
}
In this case, the REST endpoint will be able to get the two attributes of the book and to modify only its title.
The GraphQL endpoint will be able to query the title and author of an item. It will be able to query the title of the items in the collection. It will only be able to create a book with an author. When doing this mutation, the author of the created book will not be returned (the title will be instead).
When you use different serialization groups, it will create different types in your schema.
Make sure you understand the implications when doing this: having different types means breaking the cache features in some GraphQL clients (in Apollo Client for example).
For instance:
normalizationContext
for a mutation, a MyResourcePayloadData
type with the restricted fields will be generated and used instead of MyResource
(the query type).normalizationContext
for the query of an item (Query
attribute) and for the query of a collection (QueryCollection
attribute), two types MyResourceItem
and MyResourceCollection
with the restricted fields will be generated and used instead of MyResource
(the query type).By default, creating a relation when using a create
or update
mutation is not possible.
Indeed, the mutation expects an IRI for the relation in the input, so you need to use an existing relation.
For instance if you have the following resource:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Mutation;
#[ApiResource(graphQlOperations: [new Mutation(name: 'create')])]
class Book
{
// ...
public string $title;
public ?Author $author;
// ...
}
Creating a book with its author will be done like this, where /authors/32
is the IRI of an existing resource:
{
mutation {
createBook(input: {title: "The Name of the Wind", author: "/authors/32"}) {
book {
title
author {
name
}
}
}
}
}
In order to create an author as the same time as a book, you need to use the denormalization context and groups on the book and the author (see also the dedicated part in the serialization documentation:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Mutation;
use Symfony\Component\Serializer\Annotation\Groups;
#[ApiResource(graphQlOperations: [new Mutation(name: 'create', denormalizationContext: ['groups' => ['book:create']])])]
class Book
{
// ...
#[Groups(['book:create'])]
public string $title;
#[Groups(['book:create'])]
public ?Author $author;
// ...
}
And in the author resource:
<?php
// api/src/Entity/Author.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use Symfony\Component\Serializer\Annotation\Groups;
#[ApiResource]
class Author
{
// ...
#[Groups(['book:create'])]
public string $name;
// ...
}
In this case, creating a book with its author can now be done like this:
{
mutation {
createBook(input: {title: "The Name of the Wind", author: {name: "Patrick Rothfuss"}}) {
book {
title
author {
name
}
}
}
}
}
When there are errors (GraphQL ones, or if an exception is sent), a default error handler (api_platform.graphql.error_handler
) is called.
Its main responsibility is to apply a formatter to them.
If you need to log the errors, or if you want to filter them, you have to decorate this service.
For instance, create a class like this:
<?php
// api/src/Error/ErrorHandler.php
namespace App\Error;
use ApiPlatform\GraphQl\Error\ErrorHandlerInterface;
final class ErrorHandler implements ErrorHandlerInterface
{
private $defaultErrorHandler;
public function __construct(ErrorHandlerInterface $defaultErrorHandler)
{
$this->defaultErrorHandler = $defaultErrorHandler;
}
/**
* {@inheritdoc}
*/
public function __invoke(array $errors, callable $formatter): array
{
// Log or filter the errors.
return ($this->defaultErrorHandler)($errors, $formatter);
}
}
Then register the service:
# api/config/services.yaml
services:
# ...
App\Error\ErrorHandler:
decorates: api_platform.graphql.error_handler
<?xml version="1.0" encoding="UTF-8" ?>
<!-- api/config/services.xml -->
<container xmlns="http://symfony.com/schema/dic/services"
xmlns:xsd="http://www.w3.org/2001/XMLSchema-instance"
xsd:schemaLocation="http://symfony.com/schema/dic/services https://symfony.com/schema/dic/services/services-1.0.xsd">
<services>
<service id="App\Error\ErrorHandler"
decorates="api_platform.graphql.error_handler"
/>
</services>
</container>
<?php
// api/config/services.php
namespace Symfony\Component\DependencyInjection\Loader\Configurator;
use App\Error\ErrorHandler;
use App\Mailer;
return function(ContainerConfigurator $configurator) {
$services = $configurator->services();
$services->set(ErrorHandler::class)
->decorate('api_platform.graphql.error_handler');
};
By default, if an exception is sent when resolving a query or a mutation or if there are GraphQL errors, they are normalized following the GraphQL specification.
It means an errors
entry will be returned in the response, containing the following entries: message
, extensions
, locations
and path
.
For more information, please refer to the documentation in graphql-php.
In prod
mode, the displayed message will be a generic one, excepted for a RuntimeException
(and all exceptions inherited from it) for which it will be its actual message.
This behavior is different from what is described in the graphql-php documentation.
It’s because a built-in custom exception normalizer is used to normalize the RuntimeException
and change the default behavior.
If you are in dev
mode, more entries will be added in the response: debugMessage
(containing the actual exception message, for instance in the case of a LogicException
) and trace
(the formatted exception trace).
For some specific exceptions, built-in custom exception normalizers are also used to add more information.
It’s the case for a HttpException
for which the status
entry will be added under extensions
and for a ValidationException
for which status
(by default 422) and violations
entries will be added.
If you want to add more specific behaviors depending on the exception or if you want to change the behavior of the built-in ones, you can do so by creating your own normalizer.
Please follow the Symfony documentation to create a custom normalizer.
The code should look like this:
<?php
// api/src/Serializer/Exception/MyExceptionNormalizer.php
namespace App\Serializer\Exception;
use App\Exception\MyException;
use GraphQL\Error\Error;
use GraphQL\Error\FormattedError;
use Symfony\Component\Serializer\Normalizer\NormalizerInterface;
final class MyExceptionNormalizer implements NormalizerInterface
{
/**
* {@inheritdoc}
*/
public function normalize($object, $format = null, array $context = []): array
{
$exception = $object->getPrevious();
$error = FormattedError::createFromException($object);
// Add your logic here and add your specific data in the $error array (in the 'extensions' entry to follow the GraphQL specification).
// $error['extensions']['yourEntry'] = ...;
return $error;
}
/**
* {@inheritdoc}
*/
public function supportsNormalization($data, $format = null): bool
{
return $data instanceof Error && $data->getPrevious() instanceof MyException;
}
}
You can see that, in the normalize
method, you should add a call to FormattedError::createFromException
in order to have the same behavior as the other normalizers.
When registering your custom normalizer, you can add a priority to order your normalizers between themselves.
If you use a positive priority (or no priority), your normalizer will always be called before the built-in normalizers. For instance, you can register a custom normalizer like this:
# api/config/services.yaml
services:
App\Serializer\Exception\MyExceptionNormalizer:
tags:
- { name: 'serializer.normalizer', priority: 12 }
You can modify how the property names of your resources are converted into field and filter names of your GraphQL schema.
By default the property name will be used without conversion. If you want to apply a name converter, follow the Name Conversion documentation.
For instance, your resource can have properties in camelCase:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\ApiFilter;
use ApiPlatform\Doctrine\Orm\Filter\SearchFilter;
#[ApiResource]
#[ApiFilter(SearchFilter::class, properties: ['publicationDate' => 'partial'])]
class Book
{
// ...
public $publicationDate;
// ...
}
By default, with the search filter, the query to retrieve a collection will be:
{
books(publicationDate: "2010") {
edges {
node {
publicationDate
}
}
}
}
But if you use the CamelCaseToSnakeCaseNameConverter
, it will be:
{
books(publication_date: "2010") {
edges {
node {
publication_date
}
}
}
}
If you use snake_case, you can wonder how to make the difference between an underscore and the separator of the nested fields in the filter names, by default an underscore too.
For instance if you have this resource:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\ApiFilter;
use ApiPlatform\Doctrine\Orm\Filter\SearchFilter;
#[ApiResource]
#[ApiFilter(SearchFilter::class, properties: ['relatedBooks.title' => 'exact'])]
class Book
{
// ...
public $title;
#[ORM\OneToMany(targetEntity: Book::class)]
public $relatedBooks;
// ...
}
You would need to use the search filter like this:
{
books(related_books_title: "The Fitz and the Fool") {
edges {
node {
title
}
}
}
}
To avoid this issue, you can configure the nesting separator to use, for example, __
instead of _
:
# api/config/packages/api_platform.yaml
api_platform:
graphql:
nesting_separator: __
# ...
In this case, your query will be:
{
books(related_books__title: "The Fitz and the Fool") {
edges {
node {
title
}
}
}
}
Much better, isn’t it?
You might need to add your own types to your GraphQL application.
Create your type class by implementing the interface ApiPlatform\GraphQl\Type\Definition\TypeInterface
.
You should extend the GraphQL\Type\Definition\ScalarType
class too to take advantage of its useful methods.
For instance, to create a custom DateType
:
<?php
namespace App\Type\Definition;
use ApiPlatform\GraphQl\Type\Definition\TypeInterface;
use GraphQL\Error\Error;
use GraphQL\Language\AST\StringValueNode;
use GraphQL\Type\Definition\ScalarType;
use GraphQL\Utils\Utils;
final class DateTimeType extends ScalarType implements TypeInterface
{
public function __construct()
{
$this->name = 'DateTime';
$this->description = 'The `DateTime` scalar type represents time data.';
parent::__construct();
}
public function getName(): string
{
return $this->name;
}
/**
* {@inheritdoc}
*/
public function serialize($value)
{
// Already serialized.
if (\is_string($value)) {
return (new \DateTime($value))->format('Y-m-d');
}
if (!($value instanceof \DateTime)) {
throw new Error(sprintf('Value must be an instance of DateTime to be represented by DateTime: %s', Utils::printSafe($value)));
}
return $value->format(\DateTime::ATOM);
}
/**
* {@inheritdoc}
*/
public function parseValue($value)
{
if (!\is_string($value)) {
throw new Error(sprintf('DateTime cannot represent non string value: %s', Utils::printSafeJson($value)));
}
if (false === \DateTime::createFromFormat(\DateTime::ATOM, $value)) {
throw new Error(sprintf('DateTime cannot represent non date value: %s', Utils::printSafeJson($value)));
}
// Will be denormalized into a \DateTime.
return $value;
}
/**
* {@inheritdoc}
*/
public function parseLiteral($valueNode, ?array $variables = null)
{
if ($valueNode instanceof StringValueNode && false !== \DateTime::createFromFormat(\DateTime::ATOM, $valueNode->value)) {
return $valueNode->value;
}
// Intentionally without message, as all information already in wrapped Exception
throw new \Exception();
}
}
You can also check the documentation of graphql-php.
The big difference in API Platform is that the value is already serialized when it’s received in your type class. Similarly, you would not want to denormalize your parsed value since it will be done by API Platform later.
If you use autoconfiguration (the default Symfony configuration) in your application, then you are done!
Else, you need to tag your type class like this:
# api/config/services.yaml
services:
# ...
App\Type\Definition\DateTimeType:
tags:
- { name: api_platform.graphql.type }
Your custom type is now registered and is available in the TypesContainer
.
To use it please modify the extracted types or use it directly in custom queries or custom mutations.
The GraphQL schema and its types are extracted from your resources. In some cases, you would want to modify the extracted types for instance to use your custom ones.
To do so, you need to decorate the api_platform.graphql.type_converter
service:
# api/config/services.yaml
services:
# ...
'App\Type\TypeConverter':
decorates: api_platform.graphql.type_converter
Your class needs to look like this:
<?php
namespace App\Type;
use ApiPlatform\GraphQl\Type\TypeConverterInterface;
use ApiPlatform\Metadata\GraphQl\Operation;
use App\Entity\Book;
use GraphQL\Type\Definition\Type as GraphQLType;
use Symfony\Component\PropertyInfo\Type;
final class TypeConverter implements TypeConverterInterface
{
private $defaultTypeConverter;
public function __construct(TypeConverterInterface $defaultTypeConverter)
{
$this->defaultTypeConverter = $defaultTypeConverter;
}
/**
* {@inheritdoc}
*/
public function convertType(Type $type, bool $input, Operation $rootOperation, string $resourceClass, string $rootResource, ?string $property, int $depth)
{
if ('publicationDate' === $property
&& Book::class === $resourceClass
) {
return 'DateTime';
}
return $this->defaultTypeConverter->convertType($type, $input, $rootOperation, $resourceClass, $rootResource, $property, $depth);
}
/**
* {@inheritdoc}
*/
public function resolveType(string $type): ?GraphQLType
{
return $this->defaultTypeConverter->resolveType($type);
}
}
In this case, the publicationDate
property of the Book
class will have a custom DateTime
type.
You can even apply this logic for a kind of property. Replace the previous condition with something like this:
if (Type::BUILTIN_TYPE_OBJECT === $type->getBuiltinType()
&& is_a($type->getClassName(), \DateTimeInterface::class, true)
) {
return 'DateTime';
}
All DateTimeInterface
properties will have the DateTime
type in this example.
As REST, it’s possible to add dynamically a (de)serialization group when resolving a query or a mutation.
There are some differences though.
The service is api_platform.graphql.serializer.context_builder
and the method to override is create
.
The decorator could be like this:
<?php
namespace App\Serializer;
use ApiPlatform\GraphQl\Serializer\SerializerContextBuilderInterface;
use App\Entity\Book;
use Symfony\Component\Security\Core\Authorization\AuthorizationCheckerInterface;
final class BookContextBuilder implements SerializerContextBuilderInterface
{
private $decorated;
private $authorizationChecker;
public function __construct(SerializerContextBuilderInterface $decorated, AuthorizationCheckerInterface $authorizationChecker)
{
$this->decorated = $decorated;
$this->authorizationChecker = $authorizationChecker;
}
public function create(?string $resourceClass, string $operationName, array $resolverContext, bool $normalization): array
{
$context = $this->decorated->create($resourceClass, $operationName, $resolverContext, $normalization);
$resourceClass = $context['resource_class'] ?? null;
if ($resourceClass === Book::class && isset($context['groups']) && $this->authorizationChecker->isGranted('ROLE_ADMIN') && false === $normalization) {
$context['groups'][] = 'admin:input';
}
return $context;
}
}
You may need to export your schema in SDL (Schema Definition Language) to import it in some tools.
The api:graphql:export
command is provided to do so:
docker compose exec php \
bin/console api:graphql:export -o path/to/your/volume/schema.graphql
Since the command prints the schema to the output if you don’t use the -o
option, you can also use this command:
docker compose exec php \
bin/console api:graphql:export > path/in/host/schema.graphql
Please follow the file upload documentation, only the differences will be documented here.
The file upload with GraphQL follows the GraphQL multipart request specification.
You can also upload multiple files at the same time.
Configure the entity by adding a custom mutation resolver:
<?php
// api/src/Entity/MediaObject.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiProperty;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Mutation;
use App\Resolver\CreateMediaObjectResolver;
use Doctrine\ORM\Mapping as ORM;
use Symfony\Component\HttpFoundation\File\File;
use Symfony\Component\Serializer\Annotation\Groups;
use Symfony\Component\Validator\Constraints as Assert;
use Vich\UploaderBundle\Mapping\Annotation as Vich;
/**
* @Vich\Uploadable
*/
#[ORM\Entity]
#[ApiResource(
normalizationContext: ['groups' => ['media_object_read']],
types: ['https://schema.org/MediaObject'],
graphQlOperations: [
new Mutation(
name: 'upload',
resolver: CreateMediaObjectResolver::class,
deserialize: false,
args: [
'file' => [
'type' => 'Upload!',
'description' => 'The file to upload'
]
]
)
]
)]
class MediaObject
{
#[ORM\Id, ORM\Column, ORM\GeneratedValue]
protected ?int $id = null;
#[ApiProperty(types: ['https://schema.org/contentUrl'])]
#[Groups(['media_object_read'])]
public ?string $contentUrl = null;
/**
* @Vich\UploadableField(mapping="media_object", fileNameProperty="filePath")
*/
#[Assert\NotNull(groups: ['media_object_create'])]
public ?File $file = null;
#[ORM\Column(nullable: true)]
public ?string $filePath = null;
public function getId(): ?int
{
return $this->id;
}
}
As you can see, a dedicated type Upload
is used in the argument of the upload
mutation.
If you need to upload multiple files, replace 'file' => ['type' => 'Upload!', 'description' => 'The file to upload']
with 'files' => ['type' => '[Upload!]!', 'description' => 'Files to upload']
.
You don’t need to create it, it’s provided in API Platform.
The corresponding resolver you added in the resource configuration should be written like this:
<?php
// api/src/Resolver/CreateMediaObjectResolver.php
namespace App\Resolver;
use ApiPlatform\GraphQl\Resolver\MutationResolverInterface;
use App\Entity\MediaObject;
use Symfony\Component\HttpFoundation\File\UploadedFile;
final class CreateMediaObjectResolver implements MutationResolverInterface
{
/**
* @param null $item
*/
public function __invoke($item, array $context): MediaObject
{
$uploadedFile = $context['args']['input']['file'];
$mediaObject = new MediaObject();
$mediaObject->file = $uploadedFile;
return $mediaObject;
}
}
For handling the upload of multiple files, iterate over $context['args']['input']['files']
.
createMediaObject
MutationFollowing the specification, the upload must be done with a multipart/form-data
content type.
You need to enable it in the allowed formats of API Platform:
# api/config/packages/api_platform.yaml
api_platform:
formats:
# ...
multipart: ['multipart/form-data']
You can now upload files using the createMediaObject
mutation, for details check GraphQL multipart request specification
and for an example implementation for the Apollo client check out Apollo Upload Client.
mutation CreateMediaObject($file: Upload!) {
createMediaObject(input: {file: $file}) {
mediaObject {
id
contentUrl
}
}
}
By default, API Platform generates descriptions for mutations and subscriptions.
If you want to change them, or add some for queries, you can do it in the resource declaration, at the operation level, with the description
attribute.
For instance, if you want to change the description of the create
mutation:
<?php
// api/src/Entity/Book.php
namespace App\Entity;
use ApiPlatform\Metadata\ApiResource;
use ApiPlatform\Metadata\GraphQl\Mutation;
#[ApiResource(graphQlOperations: [
new Mutation(name: 'create', description: 'My custom description.')
])]
class Book
{
// ...
}
Made with love by
Les-Tilleuls.coop can help you design and develop your APIs and web projects, and train your teams in API Platform, Symfony, Next.js, Kubernetes and a wide range of other technologies.
Learn more