UNPKG

@graphql-mesh/transform-cache

Version:
151 lines (147 loc) 6.4 kB
import { composeResolvers } from '@graphql-tools/resolvers-composition'; import { hashObject, stringInterpolator } from '@graphql-mesh/string-interpolation'; import { process } from '@graphql-mesh/cross-helpers'; import { extractResolvers } from '@graphql-mesh/utils'; import { addResolversToSchema } from '@graphql-tools/schema'; function computeCacheKey(options) { const argsHash = options.args ? hashObject(options.args) : ''; const fieldNamesHash = hashObject(options.info.fieldNodes); if (!options.keyStr) { return `${options.info.parentType.name}-${options.info.fieldName}-${argsHash}-${fieldNamesHash}`; } const templateData = { typeName: options.info.parentType.name, fieldName: options.info.fieldName, args: options.args, argsHash, fieldNamesHash, info: options.info || null, env: process.env, }; return stringInterpolator.parse(options.keyStr, templateData); } class CacheTransform { constructor(options) { this.options = options; this.noWrap = true; this.shouldWaitLocal = {}; } transformSchema(schema) { var _a; const { config, cache } = this.options; const sourceResolvers = extractResolvers(schema); const compositions = {}; for (const cacheItem of config) { const effectingOperations = ((_a = cacheItem.invalidate) === null || _a === void 0 ? void 0 : _a.effectingOperations) || []; for (const { operation, matchKey } of effectingOperations) { compositions[operation] = originalResolver => async (root, args, context, info) => { const result = await originalResolver(root, args, context, info); const cacheKey = computeCacheKey({ keyStr: matchKey, args, info, }); await cache.delete(cacheKey); return result; }; } compositions[cacheItem.field] = originalResolver => async (root, args, context, info) => { var _a, _b; const cacheKey = computeCacheKey({ keyStr: cacheItem.cacheKey, args, info, }); const cachedValue = await cache.get(cacheKey); if (cachedValue) { return cachedValue; } const shouldWaitCacheKey = `${cacheKey}_shouldWait`; const pubsubTopic = `${cacheKey}_resolved`; const shouldWait = await this.shouldWait(shouldWaitCacheKey); if (shouldWait) { return this.waitAndReturn(pubsubTopic); } this.setShouldWait(shouldWaitCacheKey); try { const result = await originalResolver(root, args, context, info); await cache.set(cacheKey, result, { ttl: (_a = cacheItem.invalidate) === null || _a === void 0 ? void 0 : _a.ttl, }); // do not await setting the cache here, otherwise we would delay returnig the result unnecessarily // instead await as part of shouldWait cleanup const setCachePromise = this.options.cache.set(cacheKey, result, { ttl: (_b = cacheItem.invalidate) === null || _b === void 0 ? void 0 : _b.ttl, }); // do not wait for cleanup to complete this.cleanupShouldWait({ shouldWaitCacheKey, pubsubTopic, data: { result }, setCachePromise, }); return result; } catch (error) { this.cleanupShouldWait({ shouldWaitCacheKey, pubsubTopic, data: { error }, }); throw error; } }; } const wrappedResolvers = composeResolvers(sourceResolvers, compositions); return addResolversToSchema({ schema, resolvers: wrappedResolvers, updateResolversInPlace: true, }); } async shouldWait(shouldWaitCacheKey) { // this is to prevent a call to a the cache (which might be distributed) // when the should wait was set from current instance const shouldWaitLocal = this.shouldWaitLocal[shouldWaitCacheKey]; if (shouldWaitLocal) { return true; } const shouldWaitGlobal = await this.options.cache.get(shouldWaitCacheKey); if (shouldWaitGlobal) { return true; } // requried to be called after async check to eliminate local race condition return this.shouldWaitLocal[shouldWaitCacheKey]; } setShouldWait(shouldWaitCacheKey) { this.options.cache.set(shouldWaitCacheKey, true); this.shouldWaitLocal[shouldWaitCacheKey] = true; } async cleanupShouldWait({ shouldWaitCacheKey, pubsubTopic, data, setCachePromise, }) { if (setCachePromise) { // we need to wait for cache to be filled before removing the shouldWait await setCachePromise; } // the below order is deliberate and important // we need to delete the shouldWait keys first // this ensures that no new subscriptions for topic are created after publish is called // since the cache is async we need to await the delete await this.options.cache.delete(shouldWaitCacheKey); delete this.shouldWaitLocal[shouldWaitCacheKey]; this.options.pubsub.publish(pubsubTopic, data); } waitAndReturn(pubsubTopic) { return new Promise((resolve, reject) => { const subId = this.options.pubsub.subscribe(pubsubTopic, ({ result, error }) => { this.options.pubsub.unsubscribe(subId); if (error) { reject(error); } if (result) { resolve(result); } }); }); } } export default CacheTransform;