Improving Latency with @defer and @stream Directives
Rob Richard and Liliana Matos are front-end engineers at 1stDibs.com. They have been working with the GraphQL Working Group as champions of the
@defer
and@stream
directives.
The @defer
and @stream
directives have been a much anticipated set of features ever since Lee Byron first talked about it at GraphQL Europe 2016. For most of 2020, we have been working with the GraphQL Working Group to standardize this feature. It is now a Stage 2 proposal, but to advance further, we are looking to the GraphQL community to try using these directives and provide feedback. We have released experimental versions of GraphQL.js
and express-graphql
. They are published to npm under graphql@experimental-stream-defer
and express-graphql@experimental-stream-defer
. We encourage everyone interested in this feature to try out these releases and let us know how it goes in the issue created for feedback. Read on to find out more about what this proposal offers.
One of the disadvantages of GraphQL’s request/response model is that the GraphQL response is not returned to clients until the entire request has finished processing. However, not all requested data may be of equal importance, and in some use cases it may be possible for applications to act on a subset of the requested data. An application can speed up its time-to-interactive if the GraphQL server can send the most important data as soon as it’s ready. The new @defer
and @stream
directives allow GraphQL servers to do exactly that by returning multiple payloads from a single GraphQL response.
The @defer
directive can be applied to fragment spreads and inline fragments. It is a declarative way for developers to mark parts of a query as non-essential for immediate return.
Here’s an example of the @defer
directive:
@defer
Request
query {
person(id: "cGVvcGxlOjE=") {
name
...HomeworldFragment @defer(label: "homeworldDefer")
}
}
fragment HomeworldFragment on Person {
homeworld {
name
}
}
Response
Payload 1
{
"data": {
"person": {
"name": "Luke Skywalker"
}
},
"hasNext": true
}
Payload 2
{
"label": "homeworldDefer",
"path": ["person"],
"data": {
"homeworld": {
"name": "Tatooine"
}
},
"hasNext": false
}
When the GraphQL execution engine encounters the @defer
directive, it will fork execution and begin to resolve those fields asynchronously. While the deferred payload is still being prepared, the client can receive and act on the initial payload. This is most useful when the deferred data is large, expensive to load, or not on the critical path for interactivity.
Similar to @defer
, the @stream
directive also allows the client to receive data before the entire result is ready. @stream
can be used on list fields. Here’s an example of the @stream
directive:
@stream
Request
query {
person(id: "cGVvcGxlOjE=") {
name
films @stream(initialCount: 2, label: "filmsStream") {
title
}
}
}
Response
Payload 1
{
"data": {
"person": {
"name": "Luke Skywalker",
"films": [
{ "title": "A New Hope" },
{ "title": "The Empire Strikes Back" }
]
}
},
"hasNext": true
}
Payload 2
{
"label": "filmsStream",
"path": ["person", "films", 2],
"data": {
"title": "Return of the Jedi"
},
"hasNext": true
}
Payload 3
{
"label": "filmsStream",
"path": ["person", "films", 3],
"data": {
"title": "Revenge of the Sith"
},
"hasNext": false
}
When the GraphQL execution engine encounters the @stream
directive, it will resolve as many list items specified by the initialCount
argument. The rest will be resolved asynchronously. This is especially useful for an interface where only a few elements can be rendered above the fold. The client can render these elements as soon as possible while the server is still resolving the rest of the data.
While the GraphQL specification doesn’t specify transport protocols, we expect the most common transport for queries with @defer
/@stream
to be HTTP with chunked transfer encoding. This allows a GraphQL server to keep a standard HTTP connection open, while streaming each payload to the client as soon as it’s ready. It has low overhead, has been supported by browsers for decades, and can work easily with most infrastructure.
You can learn more about these directives at:
- The @defer/@stream RFC
- The proposed spec edits
- The GraphQL Incremental Delivery over HTTP RFC
- Our talk at GraphQL Summit 2020
– Rob Richard, Liliana Matos, Front-End Engineering, 1stDibs.com