1.4.2 • Published 3 years ago

@ayyi/apollo-response-cache v1.4.2

Weekly downloads
7
License
MIT
Repository
gitlab
Last release
3 years ago

apollo-response-cache

Publish Status Test Status

Caching and invalidation mechanisms (plugins, directives) for Apollo GraphQL.

apollo-response-cache is forked from thematters/apollo-response-cache is forked from apollo-server-plugin-response-cache.

Unlike the original, it uses a simple in-memory cache instead of Redis.

How does it work?

On each query request,

  1. responseCachePlugin creates an empty key set, and injects it to the context.
  2. @logCache collects nodes on its field, then add to the key set.
  3. responseCachePlugin writes query response cache (fqc) and node-fqc key mapping to in-memory data store.

Once a mutation updates this node, @purgeCache will purge related fqc.

Usage

Install package:

npm i @ayyi/apollo-response-cache

Add plugin and directives to the constructor:

import {
  responseCachePlugin,
  LogCacheDirective,
  PurgeCacheDirective,
} from '@ayyi/apollo-response-cache'

const server = new ApolloServer({
  plugins: [responseCachePlugin()],
})

const schema = makeExecutableSchema({
  schemaDirectives: {
    logCache: LogCacheDirective(),
    purgeCache: PurgeCacheDirective(),
  },
})

Add definitions to your schema:

directive @logCache(
  type: String!
  identifier: String = "id"
) on FIELD_DEFINITION

directive @purgeCache(
  type: String!
  identifier: String = "id"
) on FIELD_DEFINITION

Use in the schema:

type Query {
  article(id: ID!): Article! @logCache(type: "Article")
}

type Mutation {
  archiveArticle(id: ID!): Article! @purgeCache(type: "Article")
}

You can also purge cache in the resolver:

const schema = makeExecutableSchema({
  schemaDirectives: {
    purgeCache: PurgeCacheDirective({ extraNodesPath: '__invalid_nodes__' }),
  },
})

const resolvers = {
  Mutation: {
    archiveArticle: (parent, args, context) => {
      // ...
      article.__invalid_nodes__ = [
        {
          id: '2',
          type: 'Article',
        },
        {
          id: '3',
          type: 'Comment',
        },
      ]

      return article
    },
  },
}

Customize node type & id resolvers

You might want a custom function to resolve node's type and id since it may be a union or interface type.

const typeResolver = (type: string, result: any) => {
  if (['Node', 'Response'].indexOf(type) >= 0) {
    return result.__type
  }
  return type
}
const idResolver = (type: string, result: any) => {
  if (['Node', 'Response'].indexOf(type) >= 0) {
    return result.__unusual_id__
  }
  return result.id
}

const schema = makeExecutableSchema({
  schemaDirectives: {
    purgeCache: PurgeCacheDirective({ typeResolver, idResolver }),
  },
})