Cache Transform
The cache
transform allows you to easily apply caching over your data sources.
It allows you to configure custom invalidation rules (by TTL/mutation) and selective caching according to your needs.
This transform is being applied globally because of the need to communicate with other GraphQL operations from your mesh.
To get started with this transform, install it:
yarn add @graphql-mesh/transform-cache
GraphQL Mesh uses a default localforage cache (it is fallback to LRU cache in NodeJS), but you can replace it with any other key=>value cache mechanism. See cache section for more info. We'd recommend this transform be used on the root level, not the handler level. It might not work as expected if you use it on a handler level.
How to use?
Simple caching
To cache some of your queries, apply the following transform rule. The following example will cache all Query.*
in your schema forever:
transforms:
- cache:
- field: Query.*
You can also apply it to a specific field or multiple fields:
transforms:
- cache:
- field: Query.users
- cache:
- field: Query.posts
The Cache Key
Each cache record is stored with a key. The default way of creating this key is to use the GraphQL type name, the GraphQL field name, and a hash of the args
object. This is in order to make that we can distinct the cache key according to the data it's storing.
You can customize the cacheKey
according to your needs, and you can use custom helpers to help you create those cache keys dynamically.
The following example creates a cacheKey
by a GraphQL query argument called userId
per day:
transforms:
- cache:
- field: Query.user
cacheKey: user-{args.id}-{yyyy-mm-dd | date}
Cache Invalidation
TTL
Invalidation by TTL is the simplest way to deal with your cache. You can specify any time (in seconds) to keep your cache.
transforms:
- cache:
- field: Query.*
invalidate:
ttl: 3600 # 1 hour
Operation-based
GraphQL Mesh has a built-in operation-based caching solution.
You can cache data easily and invalidate it only when it changes by a mutation.
For example, given the following schema:
type Query {
user(id: ID!): User!
}
type Mutation {
updateUser(userIdToUpdate: ID!, setFields: UpdateUserInput!): User!
}
type User {
id: ID!
email: String!
name: String
}
input UpdateUserInput {
email: String
name: String
}
You can set a simple caching based on a user id:
transforms:
- cache:
- field: Query.user
cacheKey: user-{args.id}
And you can add operation-based invalidation, so when updateUser
is done successfully, it will invalidate the matching cache record to make sure the data will be fetched next time from the remote source:
transforms:
- cache:
- field: Query.user
cacheKey: user-{args.id}
invalidate:
effectingOperations:
- operation: Mutation.updateUser
matchKey: user-{args.userIdToUpdate}
This way, when someone uses updateUser
with a specific user id, it will automatically update the data record and then invalidate the cache
Programmatic
The getBuiltMesh
method of GraphQL Mesh artifacts returns the general key=>value cache it uses at the moment, so you can easily access it and invalidate records according to your needs:
const { getBuiltMesh } = require('./.mesh')
const { schema, execute, cache } = getBuiltMesh()
cache.delete(SOME_KEY)
Codesandbox Example
You can check the "Location Weather" example that uses OpenAPI handler with cache transform;
Config API Reference
field
(type:String
, required) - The type and field to apply cache to, you can use wild cards as well, for example:Query.*
cacheKey
(type:String
) - Cache key to use to store your resolvers responses. The default is:{typeName}-{fieldName}-{argsHash}-{fieldNamesHash}
Available variables:
{args.argName}
- use resolver argument{typeName}
- use name of the type{fieldName}
- use name of the field{argsHash}
- a hash based on the 'args' object{fieldNamesHash}
- a hash based on the field names selected by the client{info}
- the GraphQLResolveInfo of the resolver
Available interpolations:
{format|date}
- returns the current date with a specific format
invalidate
(type:Object
) - Invalidation rules:effectingOperations
(type:Array of Object
, required) - Invalidate the cache when a specific operation is done without an error:operation
(type:String
, required) - Path to the operation that could effect it. In a form: Mutation.something. Note that wildcard is not supported in this field.matchKey
(type:String
) - Cache key to invalidate on successful resolver (no error), seecacheKey
for list of available options in this field.
ttl
(type:Int
) - Specified in seconds, the time-to-live (TTL) value limits the lifespan