Document transforms
Make custom modifications to your GraphQL documents
This article assumes you're familiar with the anatomy of a GraphQL query
and the concept of an abstract syntax tree (AST). To explore a GraphQL AST, visit AST Explorer.
Have you noticed that Apollo Client modifies your queries—such as adding the __typename
field—before sending those queries to your GraphQL server? It does this through document transforms, functions that modify GraphQL documents before query execution.
Apollo Client provides an advanced capability that lets you define your own GraphQL document transforms to modify your GraphQL queries. This article explains how to make and use custom GraphQL document transforms.
Overview
Document transforms allow you to programmatically modify GraphQL documents used to query data in your application. A GraphQL document is an AST
that defines one or more operations and fragments, parsed from a raw GraphQL query string using thegql
function. You can create your own document transforms using the DocumentTransform
class. The created transform is then passed to the ApolloClient
constructor.1import { DocumentTransform } from '@apollo/client';
2
3const documentTransform = new DocumentTransform((document) => {
4 // modify the document
5 return transformedDocument;
6});
7
8const client = new ApolloClient({
9 documentTransform
10});
Lifecycle
Apollo Client runs document transforms before every GraphQL request for all operations. This extends to any API that performs a network request, such as the useQuery
hook or the refetch
function on ObservableQuery
.
Document transforms are run early in the request's lifecycle. This makes it possible for the cache to see modifications to GraphQL documents—an essential distinction from document modifications made to GraphQL documents in an Apollo Link. Since document transforms are run early in the request lifecycle, this makes it possible to add @client
directives to fields in your document transform to turn the field into a local-only field, or to add fragment selections for fragments defined in the fragment registry.
Interacting with built-in transforms
Apollo Client ships with built-in document transforms that are essential to the client's functionality.
The
__typename
field is added to every selection set in a query to identify the type of all objects returned by the GraphQL operation.GraphQL documents that use fragments defined in the fragment registry are added to the document before the request is sent over the network (requires Apollo Client 3.7 or later).
It's crucial for custom document transforms to interact with these built-in features. To make the most of your custom document transform, Apollo Client runs these built-in transforms twice: once before and once after your transform.
Running the built-in transforms before your custom transform allows your transform to both see the __typename
fields added to each field's selection set and modify fragment definitions defined in the fragment registry. Apollo Client understands that your transform may add new selection sets or new fragment selections to the GraphQL document. Because of this, Apollo Client reruns the built-in transforms after your custom transforms.
Running built-in transforms twice is a convenient capability because it means that you don't have to remember to include the __typename
field for any added selection sets. Nor do you need to look up fragment definitions in the fragment registry for fragment selections added to the GraphQL document.
Write your own document transform
As an example, let's write a document transform that ensures an id
field is selected anytime currentUser
is queried. We will use several helper functions and utilities provided by the graphql-js
First, we must create a new document transform using the DocumentTransform
class provided by Apollo Client. The DocumentTransform
constructor takes a callback function that runs for each GraphQL document transformed. The GraphQL document
is passed as the only argument to this callback function.
1import { DocumentTransform } from '@apollo/client';
2
3const documentTransform = new DocumentTransform((document) => {
4 // Modify the document
5});
To modify the document, we bring in the visit
graphql-js
that walks the AST and allows us to modify its nodes. The visit
function takes a GraphQL AST as the first argument and a visitor as the second argument. The visit
function returns our modified or unmodified GraphQL document, which we return from our document transform callback function.1import { DocumentTransform } from '@apollo/client';
2import { visit } from 'graphql';
3
4const documentTransform = new DocumentTransform((document) => {
5 const transformedDocument = visit(document, {
6 // visitor
7 });
8
9 return transformedDocument;
10});
Visitors allow you to visit many types of nodes in the AST, such as directives, fragments, and fields. In our example, we only care about visiting fields since we want to modify the currentUser
field in our queries. To visit a field, we need to define a Field
callback function that will be called whenever the traversal encounters one.
1const transformedDocument = visit(document, {
2 Field(field) {
3 // ...
4 }
5});
This example uses the shorthand visitor syntax, which defines the
enter
function on this node for us. This is equivalent to the following:TypeScript1visit(document, { 2 Field: { 3 enter(field) { 4 // ... 5 } 6 } 7});
Our document transform only needs to modify a field named currentUser
, so we need to check the field's name
property to determine if we are working with the currentUser
field. Let's add a conditional check and return early if we encounter any field not named currentUser
.
1const transformedDocument = visit(document, {
2 Field(field) {
3 if (field.name.value !== 'currentUser') {
4 return;
5 }
6 }
7});
Returning
undefined
from ourField
visitor tells thevisit
function to leave the node unchanged.
Now that we've determined we are working with the currentUser
field, we need to figure out if our id
field is already part of the currentUser
field's selection set. This ensures we don't accidentally select the field twice in our query.
To do so, let's get the field's selectionSet
property and loop over its selections
property to determine if the id
field is included.
It's important to note that a selectionSet
may contain selections
of both fields and fragments. Our implementation only needs to perform checks against fields, so we also check the selection's kind
property. If we find a match on a field named id
, we can stop traversal of the AST.
We will bring in both the Kind
graphql-js
, which allows us to compare against the selection's kind
property, and the BREAK
sentinel, which directs the visit
function to stop traversal of the AST.1import { visit, Kind, BREAK } from 'graphql';
2
3const transformedDocument = visit(document, {
4 Field(field) {
5 // ...
6 const selections = field.selectionSet?.selections ?? [];
7
8 for (const selection of selections) {
9 if (
10 selection.kind === Kind.FIELD &&
11 selection.name.value === 'id'
12 ) {
13 return BREAK;
14 }
15 }
16 }
17});
To keep our document transform simple, it does not traverse fragments within the
currentUser
field to determine if those fragments contain anid
field. A more complete version of this document transform might perform this check.
Now that we know the id
field is missing, we can add it to our currentUser
field's selection set. To do so, let's create a new field and give it a name of id
. This is represented as a plain object with the kind
property set to Kind.FIELD
and a name
node that defines the field's name.
1const idField = {
2 kind: Kind.FIELD,
3 name: {
4 kind: Kind.NAME,
5 value: 'id',
6 },
7};
We now return a modified field from our visitor that adds the id
field to the currentUser
field's selectionSet
. This updates our GraphQL document.
1const transformedDocument = visit(document, {
2 Field(field) {
3 // ...
4 const idField = {
5 // ...
6 };
7
8 return {
9 ...field,
10 selectionSet: {
11 ...field.selectionSet,
12 selections: [...selections, idField],
13 },
14 };
15 }
16});
This example adds the
id
field to the end of the selection set. Order doesn't matter—you may prefer to put the field elsewhere in theselections
array.
Hooray! We now have a working document transform that ensures the id
field is selected whenever a query containing the currentUser
field is sent to our server. For completeness, here is the full definition of our custom document transform after completing this example.
1import { DocumentTransform } from '@apollo/client';
2import { visit, Kind, BREAK } from 'graphql';
3
4const documentTransform = new DocumentTransform((document) => {
5 const transformedDocument = visit(document, {
6 Field(field) {
7 if (field.name.value !== 'currentUser') {
8 return;
9 }
10
11 const selections = field.selectionSet?.selections ?? [];
12
13 for (const selection of selections) {
14 if (
15 selection.kind === Kind.FIELD &&
16 selection.name.value === 'id'
17 ) {
18 return BREAK;
19 }
20 }
21
22 const idField = {
23 kind: Kind.FIELD,
24 name: {
25 kind: Kind.NAME,
26 value: 'id',
27 },
28 };
29
30 return {
31 ...field,
32 selectionSet: {
33 ...field.selectionSet,
34 selections: [...selections, idField],
35 },
36 };
37 },
38 });
39
40 return transformedDocument;
41});
Check our document transform
We can check our custom document transform by calling the transformDocument
function and passing a GraphQL query to it.
1import { print } from 'graphql';
2
3const query = gql`
4 query TestQuery {
5 currentUser {
6 name
7 }
8 }
9`;
10
11const documentTransform = new DocumentTransform((document) => {
12 // ...
13});
14
15const modifiedQuery = documentTransform.transformDocument(query);
16
17console.log(print(modifiedQuery));
18// query TestQuery {
19// currentUser {
20// name
21// id
22// }
23// }
We use the
function exported bygraphql-js
to make the query human-readable.
Similarly, we can verify that passing a query that doesn't query for currentUser
is unaffected by our transform.
1const query = gql`
2 query TestQuery {
3 user {
4 name
5 }
6 }
7`;
8
9const modifiedQuery = documentTransform.transformDocument(query);
10
11console.log(print(modifiedQuery));
12// query TestQuery {
13// user {
14// name
15// }
16// }
Query your server using the document transform
The transformDocument
function is useful to spot check your document transform. In practice, however, this will be done for you by Apollo Client.
Let's add our document transform to Apollo Client and send a query to the server. The network request will contain the updated GraphQL query and the data returned from the server will include the id
field.
1import { ApolloClient, DocumentTransform } from '@apollo/client';
2
3const query = gql`
4 query TestQuery {
5 currentUser {
6 name
7 }
8 }
9`;
10
11const documentTransform = new DocumentTransform((document) => {
12 // ...
13});
14
15const client = new ApolloClient({
16 // ...
17 documentTransform
18});
19
20const result = await client.query({ query });
21
22console.log(result.data);
23// {
24// currentUser: {
25// id: "...",
26// name: "..."
27// }
28// }
Composing document transforms
You may have noticed that the ApolloClient
constructor only takes a single documentTransform
option. As you add new capabilities to your document transforms, it may grow unwieldy. The DocumentTransform
class makes it easy to split and compose multiple transforms into a single one.
Combining multiple document transforms
You can combine multiple document transforms together using the concat()
function. This forms a "chain" of document transforms that are run one right after the other.
1const documentTransform1 = new DocumentTransform(transform1);
2const documentTransform2 = new DocumentTransform(transform2);
3
4const documentTransform = documentTransform1.concat(documentTransform2);
Here documentTransform1
is combined with documentTransform2
into a single document transform. Calling the transformDocument()
function on documentTransform
runs the GraphQL document through documentTransform1
and then through documentTransform2
. Changes made to the GraphQL document in documentTransform1
are seen by documentTransform2
.
A note about performance
Combining multiple transforms is a powerful feature that makes it easy to split up transform logic, which can boost maintainability. Depending on the implementation of your visitor, this can result in the traversal of the GraphQL document AST multiple times. Most of the time, this shouldn't be an issue. We recommend using the BREAK
graphql-js
to prevent unnecessary traversal.Suppose you are sending very large queries that require several traversals and have already optimized your visitors with the BREAK
sentinel. In that case, it's best to combine the transforms into a single visitor that traverses the AST once.
See the section on document caching to learn how Apollo Client applies optimizations to individual document transforms to mitigate the performance impact when transforming the same GraphQL document multiple times.
Conditionally running document transforms
At times, you may need to conditionally run a document transform depending on the GraphQL document. You can conditionally run a transform by calling the split()
static function on the DocumentTransform
constructor.
1import { isSubscriptionOperation } from '@apollo/client/utilities';
2
3const subscriptionTransform = new DocumentTransform(transform);
4
5const documentTransform = DocumentTransform.split(
6 (document) => isSubscriptionOperation(document),
7 subscriptionTransform
8);
This example uses the
isSubscriptionOperation
utility function added to Apollo Client in 3.8. Similarly,isQueryOperation
andisMutationOperation
utility functions are available for use.
Here the subscriptionTransform
is only run for subscription operations. For all other operations, no modifications are made to the GraphQL document. The resulting document transform will first check to see if the document
is a subscription operation, and if so, proceed to run subscriptionTransform
. If not, subscriptionTransform
is bypassed, and the GraphQL document is returned as-is.
The split
function also allows you to pass a second document transform to its function, allowing you to replicate an if/else condition.
1const subscriptionTransform = new DocumentTransform(transform1);
2const defaultTransform = new DocumentTransform(transform2)
3
4const documentTransform = DocumentTransform.split(
5 (document) => isSubscriptionOperation(document),
6 subscriptionTransform,
7 defaultTransform
8);
Here the subscriptionTransform
is only run for subscription operations. For all other operations, the GraphQL document is run through the defaultTransform
.
Why should I use the split()
function instead of a conditional check inside of the transform function?
Sometimes, using the split()
function is more efficient than running a conditional check inside the transform function.
For example, you can run a transform by adding a conditional check inside the transform function itself:
1const documentTransform = new DocumentTransform((document) => {
2 if (shouldTransform(document)) {
3 // ...
4 return transformedDocument
5 }
6
7 return document
8});
Consider the case where you've combined multiple document transforms using the concat()
function:
1const documentTransform1 = new DocumentTransform(transform1);
2const documentTransform2 = new DocumentTransform(transform2);
3const documentTransform3 = new DocumentTransform(transform3);
4
5const documentTransform = documentTransform1
6 .concat(documentTransform2)
7 .concat(documentTransform3);
The split()
function makes skipping the entire chain of document transforms easier.
1const documentTransform = DocumentTransform.split(
2 (document) => shouldTransform(document),
3 documentTransform1
4 .concat(documentTransform2)
5 .concat(documentTransform3)
6);
Document caching
You should strive to make your document transforms deterministic. This means the document transform should always output the same transformed GraphQL document when given the same input GraphQL document. The DocumentTransform
class optimizes for this case by caching the transformed result for each input GraphQL document. This speeds up repeated calls to the document transform to avoid unnecessary work.
The DocumentTransform
class takes this further and records all transformed documents. That means that passing an already transformed document to the document transform will immediately return the GraphQL document.
1const transformed1 = documentTransform.transformDocument(document);
2const transformed2 = documentTransform.transformDocument(transformed1);
3
4transformed1 === transformed2; // => true
In practice, this optimization is invisible to you. Apollo Client calls the
transformDocument
function on the document transform for you. This optimization primarily benefits the internals of Apollo Client where the transformed document is passed around several areas of the code base.
Non-deterministic document transforms
In rare circumstances, you may need to rely on a runtime condition from outside the transform function that changes the result of the document transform. Due to the automatic caching of the document transform, this becomes a problem when that runtime condition changes between calls to your document transform.
Instead of completely disabling the document cache in these situations, you can provide a custom cache key that will be used to cache the result of the document transform. This ensures your transform is only called as often as necessary while maintaining the flexibility of the runtime condition.
To customize the cache key, pass the getCacheKey
function as an option to the second argument of the DocumentTransform
constructor. This function receives the document
that will be passed to your transform function and is expected to return an array.
As an example, here is a document transform that depends on whether the user is connected to the network.
1const documentTransform = new DocumentTransform(
2 (document) => {
3 if (window.navigator.onLine) {
4 // Transform the document when the user is online
5 } else {
6 // Transform the document when the user is offline
7 }
8 },
9 {
10 getCacheKey: (document) => [document, window.navigator.onLine]
11 }
12);
⚠️ It is highly recommended you use the
document
as part of your cache key. In this example, if thedocument
is omitted from the cache key, the document transform will only output two transformed documents: one for thetrue
condition and one for thefalse
condition. Using thedocument
in the cache key ensures that each unique document in your application will be transformed accordingly.
You may conditionally disable the cache for select GraphQL documents by returning undefined
from the getCacheKey
function. This will force the document transform to run, regardless of whether the input GraphQL document has been seen.
1const documentTransform = new DocumentTransform(
2 (document) => {
3 // ...
4 },
5 {
6 getCacheKey: (document) => {
7 // Always run the transform function when `shouldCache` is `false`
8 if (shouldCache(document)) {
9 return [document]
10 }
11 }
12 }
13);
As a last resort, you may completely disable document caching to force your transform function to run each time your document transform is used. Set the cache
option to false
to disable the cache.
1const documentTransform = new DocumentTransform(
2 (document) => {
3 // ...
4 },
5 {
6 cache: false
7 }
8);
Caching within combined transforms
When you combine multiple document transforms using the concat()
function, each document transform's cache configuration is honored. This allows you to mix and match transforms that contain varying cache configurations and be confident the resulting GraphQL document is correctly transformed.
1const cachedTransform = new DocumentTransform(transform);
2
3const varyingTransform = new DocumentTransform(transform, {
4 getCacheKey: (document) => [document, window.navigator.onLine]
5});
6
7const conditionalCachedTransform = new DocumentTransform(transform, {
8 getCacheKey: (document) => {
9 if (shouldCache(document)) {
10 return [document]
11 }
12 }
13});
14
15const nonCachedTransform = new DocumentTransform(transform, {
16 cache: false
17});
18
19const documentTransform =
20 cachedTransform
21 .concat(varyingTransform)
22 .concat(conditionalCachedTransform)
23 .concat(nonCachedTransform);
We recommend adding non-cached document transforms to the end of the
concat()
chain. Document caching relies on referential equality to determine if the GraphQL document has been seen. If a non-cached document transform is defined before a cached transform, the cached transform will store new GraphQL documents created by the non-cached document transform each run. This could result in a memory leak.
TypeScript and GraphQL Code Generator
is a popular tool that generates TypeScript types for your GraphQL documents. It does this by statically analyzing your code to search for GraphQL query strings.Document transforms present a challenge for this tool. Because document transforms are used at runtime, there's no way for static analysis to understand the changes applied to GraphQL documents from within document transforms.
Thankfully, GraphQL Code Generator provides a document transform
feature that allows you to connect Apollo Client document transforms to GraphQL Code Generator. Use your document transform inside thetransform
function passed to the GraphQL Code Generator config:1import type { CodegenConfig } from '@graphql-codegen/cli';
2import { documentTransform } from './path/to/your/transform';
3
4const config: CodegenConfig = {
5 schema: 'https://localhost:4000/graphql',
6 documents: ['src/**/*.tsx'],
7 generates: {
8 './src/gql/': {
9 preset: 'client',
10 documentTransforms: [
11 {
12 transform: ({ documents }) => {
13 return documents.map((documentFile) => {
14 documentFile.document = documentTransform
15 .transformDocument(documentFile.document);
16
17 return documentFile;
18 });
19 }
20 }
21 ]
22 }
23 }
24}
You might not need document transforms
Document transforms are a powerful feature of Apollo Client. After reading this article, you may be rushing to find as many use cases for this feature as possible. While we encourage you to use this feature where it makes sense in your application, there can be a hidden cost to using it.
Consider what happens when working in a large production application that spans many teams within your organization. Document transforms are typically defined in the code base far from where GraphQL queries are defined. Not all developers may be aware of their existence nor understand their impact on the final GraphQL document.
Document transforms can make endless modifications to GraphQL documents before they are sent to the network. You may find yourself in a position where the result returned from the GraphQL operation does not match the original GraphQL document. This can get especially confusing when document transforms remove fields or make other destructive changes.
Consider leaning on existing techniques as a first resort, such as linting. For example, if you require that every selection set in your GraphQL document should include an id
field, you may find it more useful to create a lint rule that complains when you forget to include it. This makes it more obvious exactly what to expect from your GraphQL queries since the lint rule is applied where your GraphQL queries are defined. Adding an id
via a document transform makes this relationship an implicit one.
We encourage you to document your own document transforms to create a shared knowledge base to help avoid confusion. This doesn't mean we consider this feature dangerous. After all, Apollo Client has been performing document transformations for nearly its entire existence and they are necessary for its core functionality.
Can I use this to define my own custom directives?
At a glance, document transforms seem like a great place to create and define custom directives since they can detect their presence in the GraphQL document. Document transforms, however, don't have access to the cache, nor can they interact with the data returned from your GraphQL server. If your custom directive needs access to these features in Apollo Client, you will have difficulty finding ways to make this work.
Custom directives are limited to use cases that depend on modifications to the GraphQL document itself.
Here is an example that uses a DSL-like directive that depends on a feature flagging system to conditionally include fields in queries. The document transform modifies a custom @feature
directive to a regular @include
directive and adds a variable definition to the query.
1const query = gql`
2 query MyQuery {
3 myCustomField @feature(name: "custom", version: 2)
4 }
5`;
6
7const documentTransform = new DocumentTransform((document) => {
8 // convert `@feature` directives to `@include` directives and update variable definitions
9});
10
11documentTransform.transformDocument(query);
12// query MyQuery($feature_custom_v2: Boolean!) {
13// myCustomField @include(if: $feature_custom_v2)
14// }
API Reference
Options
boolean
Determines whether to cache the transformed GraphQL document. Caching can speed up repeated calls to the document transform for the same input document. Set to false
to completely disable caching for the document transform. When disabled, this option takes precedence over the getCacheKey
option.
The default value is true
.
(document: DocumentNode) => DocumentTransformCacheKey | undefined
Defines a custom cache key for a GraphQL document that will determine whether to re-run the document transform when given the same input GraphQL document. Returns an array that defines the cache key. Return undefined
to disable caching for that GraphQL document.
Note: The items in the array may be any type, but also need to be referentially stable to guarantee a stable cache key.
The default implementation of this function returns the document
as the cache key.