@graphql-mesh/plugin-http-cache
Version:
149 lines (148 loc) • 6.96 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.default = useHTTPCache;
const tslib_1 = require("tslib");
const http_cache_semantics_1 = tslib_1.__importDefault(require("http-cache-semantics"));
const utils_1 = require("@graphql-mesh/utils");
const fetch_1 = require("@whatwg-node/fetch");
const promise_helpers_1 = require("@whatwg-node/promise-helpers");
function useHTTPCache({ cache, matches, ignores, logger, }) {
if (!cache) {
throw new Error('HTTP Cache plugin requires a cache instance');
}
let matchesPatterns;
let ignoresPatterns;
let URLPatternCtor = fetch_1.URLPattern;
let ResponseCtor = fetch_1.Response;
function shouldSkip(url) {
ignoresPatterns ||= ignores?.map(match => new URLPatternCtor(match)) || [];
if (ignoresPatterns?.length) {
for (const pattern of ignoresPatterns) {
if (pattern.test(url)) {
logger?.debug(`Ignore pattern ${pattern} matched for ${url}`);
return true;
}
}
}
matchesPatterns ||= matches?.map(match => new URLPatternCtor(match)) || [];
if (matchesPatterns?.length) {
for (const pattern of matchesPatterns) {
if (pattern.test(url)) {
logger?.debug(`Match pattern ${pattern} matched for ${url}`);
return false;
}
}
logger?.debug(`No match pattern matched for ${url}`);
return true;
}
return false;
}
const pluginLogger = logger?.child({ plugin: 'HTTP Cache' });
return {
onYogaInit({ yoga }) {
if (yoga.fetchAPI.URLPattern) {
URLPatternCtor = yoga.fetchAPI.URLPattern;
}
if (yoga.fetchAPI.Response) {
ResponseCtor = yoga.fetchAPI.Response;
}
// @ts-expect-error - Logger type mismatch
logger ||= yoga.logger;
},
onFetch({ url, options, setOptions, context, endResponse }) {
if (shouldSkip(url) || typeof options.body === 'object') {
pluginLogger?.debug(`Skipping cache for ${url}`);
return;
}
const cacheKey = `http-cache-${url}-${options.method}-${options.body}`;
const policyRequest = {
url,
headers: (0, utils_1.getHeadersObj)(options.headers),
};
return (0, promise_helpers_1.handleMaybePromise)(() => cache.get(cacheKey), function handleCacheEntry(cacheEntry) {
let policy;
function returnCachedResponse(endResponse) {
return endResponse(new ResponseCtor(cacheEntry.body, {
status: cacheEntry.response.status,
// @ts-expect-error - Headers type mismatch
headers: policy.responseHeaders(),
}));
}
if (cacheEntry?.policy) {
pluginLogger?.debug(`Cache hit for ${url}`);
policy = http_cache_semantics_1.default.fromObject(cacheEntry.policy);
if (policy?.satisfiesWithoutRevalidation(policyRequest)) {
pluginLogger?.debug(`Cache hit is fresh for ${url}`);
return returnCachedResponse(endResponse);
}
else if (policy?.revalidationHeaders) {
pluginLogger?.debug(`Cache will be revalidated for ${url}`);
setOptions({
...options,
// @ts-expect-error - Headers type mismatch
headers: policy.revalidationHeaders(policyRequest),
});
}
}
return function handleResponse({ response, setResponse }) {
let body$;
const policyResponse = {
status: response.status,
headers: (0, utils_1.getHeadersObj)(response.headers),
};
function updateCacheEntry() {
const store$ = (0, promise_helpers_1.handleMaybePromise)(() => body$, body => {
const ttl = policy.timeToLive();
if (ttl) {
pluginLogger?.debug(`TTL: ${ttl}ms`);
}
return cache.set(cacheKey, {
policy: policy.toObject(),
response: policyResponse,
body,
}, ttl
? {
ttl: ttl / 1000,
}
: undefined);
});
context?.waitUntil?.(store$);
}
if (policy) {
const revalidationPolicy = policy.revalidatedPolicy(policyRequest, policyResponse);
policy = revalidationPolicy.policy;
if (revalidationPolicy.matches) {
pluginLogger?.debug(`Response not modified for ${url}`);
body$ = cacheEntry.body;
updateCacheEntry();
return returnCachedResponse(setResponse);
}
pluginLogger?.debug(`Updating the cache entry cache for ${url}`);
}
else {
pluginLogger?.debug(`Creating the cache entry for ${url}`);
policy = new http_cache_semantics_1.default(policyRequest, policyResponse);
}
if (policy.storable()) {
pluginLogger?.debug(`Storing the cache entry for ${url}`);
body$ = response.text();
updateCacheEntry();
return body$.then(body => setResponse(new ResponseCtor(body, {
...response,
// @ts-expect-error - Headers type mismatch
headers: policy.responseHeaders(),
})));
}
else {
if (cacheEntry) {
pluginLogger?.debug(`Deleting the cache entry for ${url}`);
const delete$ = cache.delete(cacheKey);
// @ts-expect-error - Promise type mismatch
context?.waitUntil?.(delete$);
}
}
};
});
},
};
}
;