'use strict'; const { safeHTTPMethods } = require('../core/util'); /** * @param {import('../../types/dispatcher.d.ts').default.DispatchOptions} opts */ function makeCacheKey(opts) { if (!opts.origin) { throw new Error('opts.origin is undefined'); } const headers = normaliseHeaders(opts); return { origin: opts.origin.toString(), method: opts.method, path: opts.path, headers, }; } /** * @param {Record} * @return {Record} */ function normaliseHeaders(opts) { let headers; if (opts.headers == null) { headers = {}; } else if (typeof opts.headers[Symbol.iterator] === 'function') { headers = {}; for (const x of opts.headers) { if (!Array.isArray(x)) { throw new Error('opts.headers is not a valid header map'); } const [key, val] = x; if (typeof key !== 'string' || typeof val !== 'string') { throw new Error('opts.headers is not a valid header map'); } headers[key.toLowerCase()] = val; } } else if (typeof opts.headers === 'object') { headers = {}; for (const key of Object.keys(opts.headers)) { headers[key.toLowerCase()] = opts.headers[key]; } } else { throw new Error('opts.headers is not an object'); } return headers; } /** * @param {any} key */ function assertCacheKey(key) { if (typeof key !== 'object') { throw new TypeError(`expected key to be object, got ${typeof key}`); } for (const property of ['origin', 'method', 'path']) { if (typeof key[property] !== 'string') { throw new TypeError( `expected key.${property} to be string, got ${typeof key[property]}` ); } } if (key.headers !== undefined && typeof key.headers !== 'object') { throw new TypeError(`expected headers to be object, got ${typeof key}`); } } /** * @param {any} value */ function assertCacheValue(value) { if (typeof value !== 'object') { throw new TypeError(`expected value to be object, got ${typeof value}`); } for (const property of ['statusCode', 'cachedAt', 'staleAt', 'deleteAt']) { if (typeof value[property] !== 'number') { throw new TypeError( `expected value.${property} to be number, got ${typeof value[property]}` ); } } if (typeof value.statusMessage !== 'string') { throw new TypeError( `expected value.statusMessage to be string, got ${typeof value.statusMessage}` ); } if (value.headers != null && typeof value.headers !== 'object') { throw new TypeError( `expected value.rawHeaders to be object, got ${typeof value.headers}` ); } if (value.vary !== undefined && typeof value.vary !== 'object') { throw new TypeError( `expected value.vary to be object, got ${typeof value.vary}` ); } if (value.etag !== undefined && typeof value.etag !== 'string') { throw new TypeError( `expected value.etag to be string, got ${typeof value.etag}` ); } } /** * @see https://www.rfc-editor.org/rfc/rfc9111.html#name-cache-control * @see https://www.iana.org/assignments/http-cache-directives/http-cache-directives.xhtml * @param {string | string[]} header * @returns {import('../../types/cache-interceptor.d.ts').default.CacheControlDirectives} */ function parseCacheControlHeader(header) { /** * @type {import('../../types/cache-interceptor.d.ts').default.CacheControlDirectives} */ const output = {}; let directives; if (Array.isArray(header)) { directives = []; for (const directive of header) { directives.push(...directive.split(',')); } } else { directives = header.split(','); } for (let i = 0; i < directives.length; i++) { const directive = directives[i].toLowerCase(); const keyValueDelimiter = directive.indexOf('='); let key; let value; if (keyValueDelimiter !== -1) { key = directive.substring(0, keyValueDelimiter).trimStart(); value = directive.substring(keyValueDelimiter + 1); } else { key = directive.trim(); } switch (key) { case 'min-fresh': case 'max-stale': case 'max-age': case 's-maxage': case 'stale-while-revalidate': case 'stale-if-error': { if (value === undefined || value[0] === ' ') { continue; } if ( value.length >= 2 && value[0] === '"' && value[value.length - 1] === '"' ) { value = value.substring(1, value.length - 1); } const parsedValue = parseInt(value, 10); // eslint-disable-next-line no-self-compare if (parsedValue !== parsedValue) { continue; } if (key === 'max-age' && key in output && output[key] >= parsedValue) { continue; } output[key] = parsedValue; break; } case 'private': case 'no-cache': { if (value) { // The private and no-cache directives can be unqualified (aka just // `private` or `no-cache`) or qualified (w/ a value). When they're // qualified, it's a list of headers like `no-cache=header1`, // `no-cache="header1"`, or `no-cache="header1, header2"` // If we're given multiple headers, the comma messes us up since // we split the full header by commas. So, let's loop through the // remaining parts in front of us until we find one that ends in a // quote. We can then just splice all of the parts in between the // starting quote and the ending quote out of the directives array // and continue parsing like normal. // https://www.rfc-editor.org/rfc/rfc9111.html#name-no-cache-2 if (value[0] === '"') { // Something like `no-cache="some-header"` OR `no-cache="some-header, another-header"`. // Add the first header on and cut off the leading quote const headers = [value.substring(1)]; let foundEndingQuote = value[value.length - 1] === '"'; if (!foundEndingQuote) { // Something like `no-cache="some-header, another-header"` // This can still be something invalid, e.g. `no-cache="some-header, ...` for (let j = i + 1; j < directives.length; j++) { const nextPart = directives[j]; const nextPartLength = nextPart.length; headers.push(nextPart.trim()); if ( nextPartLength !== 0 && nextPart[nextPartLength - 1] === '"' ) { foundEndingQuote = true; break; } } } if (foundEndingQuote) { let lastHeader = headers[headers.length - 1]; if (lastHeader[lastHeader.length - 1] === '"') { lastHeader = lastHeader.substring(0, lastHeader.length - 1); headers[headers.length - 1] = lastHeader; } if (key in output) { output[key] = output[key].concat(headers); } else { output[key] = headers; } } } else { // Something like `no-cache=some-header` if (key in output) { output[key] = output[key].concat(value); } else { output[key] = [value]; } } break; } } // eslint-disable-next-line no-fallthrough case 'public': case 'no-store': case 'must-revalidate': case 'proxy-revalidate': case 'immutable': case 'no-transform': case 'must-understand': case 'only-if-cached': if (value) { // These are qualified (something like `public=...`) when they aren't // allowed to be, skip continue; } output[key] = true; break; default: // Ignore unknown directives as per https://www.rfc-editor.org/rfc/rfc9111.html#section-5.2.3-1 continue; } } return output; } /** * @param {string | string[]} varyHeader Vary header from the server * @param {Record} headers Request headers * @returns {Record} */ function parseVaryHeader(varyHeader, headers) { if (typeof varyHeader === 'string' && varyHeader.includes('*')) { return headers; } const output = /** @type {Record} */ ({}); const varyingHeaders = typeof varyHeader === 'string' ? varyHeader.split(',') : varyHeader; for (const header of varyingHeaders) { const trimmedHeader = header.trim().toLowerCase(); output[trimmedHeader] = headers[trimmedHeader] ?? null; } return output; } /** * Note: this deviates from the spec a little. Empty etags ("", W/"") are valid, * however, including them in cached resposnes serves little to no purpose. * * @see https://www.rfc-editor.org/rfc/rfc9110.html#name-etag * * @param {string} etag * @returns {boolean} */ function isEtagUsable(etag) { if (etag.length <= 2) { // Shortest an etag can be is two chars (just ""). This is where we deviate // from the spec requiring a min of 3 chars however return false; } if (etag[0] === '"' && etag[etag.length - 1] === '"') { // ETag: ""asd123"" or ETag: "W/"asd123"", kinda undefined behavior in the // spec. Some servers will accept these while others don't. // ETag: "asd123" return !(etag[1] === '"' || etag.startsWith('"W/')); } if (etag.startsWith('W/"') && etag[etag.length - 1] === '"') { // ETag: W/"", also where we deviate from the spec & require a min of 3 // chars // ETag: for W/"", W/"asd123" return etag.length !== 4; } // Anything else return false; } /** * @param {unknown} store * @returns {asserts store is import('../../types/cache-interceptor.d.ts').default.CacheStore} */ function assertCacheStore(store, name = 'CacheStore') { if (typeof store !== 'object' || store === null) { throw new TypeError( `expected type of ${name} to be a CacheStore, got ${store === null ? 'null' : typeof store}` ); } for (const fn of ['get', 'createWriteStream', 'delete']) { if (typeof store[fn] !== 'function') { throw new TypeError(`${name} needs to have a \`${fn}()\` function`); } } } /** * @param {unknown} methods * @returns {asserts methods is import('../../types/cache-interceptor.d.ts').default.CacheMethods[]} */ function assertCacheMethods(methods, name = 'CacheMethods') { if (!Array.isArray(methods)) { throw new TypeError( `expected type of ${name} needs to be an array, got ${methods === null ? 'null' : typeof methods}` ); } if (methods.length === 0) { throw new TypeError(`${name} needs to have at least one method`); } for (const method of methods) { if (!safeHTTPMethods.includes(method)) { throw new TypeError( `element of ${name}-array needs to be one of following values: ${safeHTTPMethods.join(', ')}, got ${method}` ); } } } module.exports = { makeCacheKey, normaliseHeaders, assertCacheKey, assertCacheValue, parseCacheControlHeader, parseVaryHeader, isEtagUsable, assertCacheMethods, assertCacheStore, };