Skip to content

Commit

Permalink
cache: fix stale-while-revalidate and stale-if-error
Browse files Browse the repository at this point in the history
Closes #3853

Signed-off-by: flakey5 <[email protected]>
  • Loading branch information
flakey5 committed Nov 25, 2024
1 parent ac30c29 commit 10c87fd
Show file tree
Hide file tree
Showing 9 changed files with 601 additions and 92 deletions.
1 change: 1 addition & 0 deletions lib/cache/memory-cache-store.js
Original file line number Diff line number Diff line change
Expand Up @@ -90,6 +90,7 @@ class MemoryCacheStore {
headers: entry.headers,
body: entry.body,
etag: entry.etag,
cacheControlDirectives: entry.cacheControlDirectives,
cachedAt: entry.cachedAt,
staleAt: entry.staleAt,
deleteAt: entry.deleteAt
Expand Down
25 changes: 20 additions & 5 deletions lib/handler/cache-handler.js
Original file line number Diff line number Diff line change
Expand Up @@ -113,6 +113,7 @@ class CacheHandler {
statusMessage,
headers: strippedHeaders,
vary: varyDirectives,
cacheControlDirectives,
cachedAt: now,
staleAt,
deleteAt
Expand Down Expand Up @@ -170,7 +171,7 @@ class CacheHandler {
*
* @param {number} statusCode
* @param {Record<string, string | string[]>} headers
* @param {import('../util/cache.js').CacheControlDirectives} cacheControlDirectives
* @param {import('../../types/cache-interceptor.d.ts').default.CacheControlDirectives} cacheControlDirectives
*/
function canCacheResponse (statusCode, headers, cacheControlDirectives) {
if (statusCode !== 200 && statusCode !== 307) {
Expand Down Expand Up @@ -217,7 +218,7 @@ function canCacheResponse (statusCode, headers, cacheControlDirectives) {
/**
* @param {number} now
* @param {Record<string, string | string[]>} headers
* @param {import('../util/cache.js').CacheControlDirectives} cacheControlDirectives
* @param {import('../../types/cache-interceptor.d.ts').default.CacheControlDirectives} cacheControlDirectives
*
* @returns {number | undefined} time that the value is stale at or undefined if it shouldn't be cached
*/
Expand Down Expand Up @@ -253,22 +254,36 @@ function determineStaleAt (now, headers, cacheControlDirectives) {

/**
* @param {number} now
* @param {import('../util/cache.js').CacheControlDirectives} cacheControlDirectives
* @param {import('../../types/cache-interceptor.d.ts').default.CacheControlDirectives} cacheControlDirectives
* @param {number} staleAt
*/
function determineDeleteAt (now, cacheControlDirectives, staleAt) {
let staleWhileRevalidate = -Infinity
let staleIfError = -Infinity

if (cacheControlDirectives['stale-while-revalidate']) {
return now + (cacheControlDirectives['stale-while-revalidate'] * 1000)
staleWhileRevalidate = now + (cacheControlDirectives['stale-while-revalidate'] * 1000)
}

if (cacheControlDirectives['stale-if-error']) {
staleIfError = now + (cacheControlDirectives['stale-if-error'] * 1000)
}

return staleAt
return Math.max(staleAt, staleWhileRevalidate, staleIfError)
}

/**
* Strips headers required to be removed in cached responses
<<<<<<< HEAD
* @param {Record<string, string | string[]>} headers
* @param {import('../util/cache.js').CacheControlDirectives} cacheControlDirectives
* @returns {Record<string, string | string []>}
=======
* @param {Buffer[]} rawHeaders
* @param {string[]} parsedRawHeaders
* @param {import('../../types/cache-interceptor.d.ts').default.CacheControlDirectives} cacheControlDirectives
* @returns {Buffer[]}
>>>>>>> 45c0793e (cache: fix stale-while-revalidate and stale-if-error)
*/
function stripNecessaryHeaders (headers, cacheControlDirectives) {
const headersToRemove = ['connection']
Expand Down
26 changes: 25 additions & 1 deletion lib/handler/cache-revalidation-handler.js
Original file line number Diff line number Diff line change
Expand Up @@ -17,28 +17,45 @@ const assert = require('node:assert')
*/
class CacheRevalidationHandler {
#successful = false

/**
* @type {((boolean, any) => void) | null}
*/
#callback

/**
* @type {(import('../../types/dispatcher.d.ts').default.DispatchHandler)}
*/
#handler

<<<<<<< HEAD
#context

/**
* @param {(boolean, any) => void} callback Function to call if the cached value is valid
* @param {import('../../types/dispatcher.d.ts').default.DispatchHandler} handler
=======
/**
* @type {boolean}
*/
#allowErrorStatusCodes

#abort

/**
* @param {(boolean) => void} callback Function to call if the cached value is valid
* @param {import('../../types/dispatcher.d.ts').default.DispatchHandlers} handler
* @param {boolean} allowErrorStatusCodes
>>>>>>> 45c0793e (cache: fix stale-while-revalidate and stale-if-error)
*/
constructor (callback, handler) {
constructor (callback, handler, allowErrorStatusCodes) {
if (typeof callback !== 'function') {
throw new TypeError('callback must be a function')
}

this.#callback = callback
this.#handler = handler
this.#allowErrorStatusCodes = allowErrorStatusCodes
}

onRequestStart (controller, context) {
Expand All @@ -59,8 +76,15 @@ class CacheRevalidationHandler {
assert(this.#callback != null)

// https://www.rfc-editor.org/rfc/rfc9111.html#name-handling-a-validation-respo
<<<<<<< HEAD
this.#successful = statusCode === 304
this.#callback(this.#successful, this.#context)
=======
// https://datatracker.ietf.org/doc/html/rfc5861#section-4
this.#successful = statusCode === 304 ||
(this.#allowErrorStatusCodes && statusCode >= 500 && statusCode <= 504)
this.#callback(this.#successful)
>>>>>>> 45c0793e (cache: fix stale-while-revalidate and stale-if-error)
this.#callback = null

if (this.#successful) {
Expand Down
Loading

0 comments on commit 10c87fd

Please sign in to comment.