-
Notifications
You must be signed in to change notification settings - Fork 87
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: make CDN SWR background revalidation discard stale cache content in order to produce fresh responses #2765
Merged
Merged
Changes from all commits
Commits
Show all changes
3 commits
Select commit
Hold shift + click to select a range
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -390,6 +390,63 @@ test.describe('Simple Page Router (no basePath, no i18n)', () => { | |
expect(beforeFetch.localeCompare(date2)).toBeLessThan(0) | ||
}) | ||
|
||
test('Background SWR invocations can store fresh responses in CDN cache', async ({ | ||
page, | ||
pageRouter, | ||
}) => { | ||
const slug = Date.now() | ||
const pathname = `/revalidate-60/${slug}` | ||
|
||
const beforeFirstFetch = new Date().toISOString() | ||
|
||
const response1 = await page.goto(new URL(pathname, pageRouter.url).href) | ||
expect(response1?.status()).toBe(200) | ||
expect(response1?.headers()['cache-status']).toMatch( | ||
/"Netlify (Edge|Durable)"; fwd=(uri-miss(; stored)?|miss)/m, | ||
) | ||
expect(response1?.headers()['netlify-cdn-cache-control']).toMatch( | ||
/s-maxage=60, stale-while-revalidate=[0-9]+, durable/, | ||
) | ||
|
||
// ensure response was NOT produced before invocation | ||
const date1 = (await page.textContent('[data-testid="date-now"]')) ?? '' | ||
expect(date1.localeCompare(beforeFirstFetch)).toBeGreaterThan(0) | ||
|
||
// allow page to get stale | ||
await page.waitForTimeout(60_000) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. :( need for Durable so reduce potential flakiness if different edge nodes would be used - tested page has 60s revalidate time because of that as well |
||
|
||
const response2 = await page.goto(new URL(pathname, pageRouter.url).href) | ||
expect(response2?.status()).toBe(200) | ||
expect(response2?.headers()['cache-status']).toMatch( | ||
/"Netlify (Edge|Durable)"; hit; fwd=stale/m, | ||
) | ||
expect(response2?.headers()['netlify-cdn-cache-control']).toMatch( | ||
/s-maxage=60, stale-while-revalidate=[0-9]+, durable/, | ||
) | ||
|
||
const date2 = (await page.textContent('[data-testid="date-now"]')) ?? '' | ||
expect(date2).toBe(date1) | ||
|
||
// wait a bit to ensure background work has a chance to finish | ||
// (it should take at least 5 seconds to regenerate, so we should wait at least that much to get fresh response) | ||
await page.waitForTimeout(10_000) | ||
|
||
// subsequent request should be served with fresh response from cdn cache, as previous request | ||
// should result in background SWR invocation that serves fresh response that was stored in CDN cache | ||
const response3 = await page.goto(new URL(pathname, pageRouter.url).href) | ||
expect(response3?.status()).toBe(200) | ||
expect(response3?.headers()['cache-status']).toMatch( | ||
// hit, without being followed by ';fwd=stale' | ||
/"Netlify (Edge|Durable)"; hit(?!; fwd=stale)/m, | ||
) | ||
expect(response3?.headers()['netlify-cdn-cache-control']).toMatch( | ||
/s-maxage=60, stale-while-revalidate=[0-9]+, durable/, | ||
) | ||
|
||
const date3 = (await page.textContent('[data-testid="date-now"]')) ?? '' | ||
expect(date3.localeCompare(date2)).toBeGreaterThan(0) | ||
}) | ||
|
||
test('should serve 404 page when requesting non existing page (no matching route)', async ({ | ||
page, | ||
pageRouter, | ||
|
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this list exhaustive?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is all
kind
s that we do support (and not hit this caseopennextjs-netlify/src/run/handlers/cache.cts
Lines 285 to 286 in f3e24b1
But right now this is manually constructed and there is no guard rail ensuring this list gets updated and in sync in the future.
Will think about best way of ensuring that all supported
kind
s are in here (but I would do this probably as follow up in case I don't figure out quick/easy solution)There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok, there is a way but it's rather not pretty - full
tsc
check results in pretty awful errors then that are really non-descript (if we miss kind) and the whole setup looks pretty iffy, so I would rather not go this road.https://typescript-eslint.io/rules/switch-exhaustiveness-check/ would be great for this ... but it doesn't really work with version of eslint-typescript we have - we would need some major upgrades of this ... which would require major upgrades of bunch of other things. Those major upgrades also change default configs.
And lastly - we don't even control those versions ourselves - and shared config we use was last released ... over 2 years ago ( https://github.com/netlify/eslint-config-node/releases )
So, I won't be touching this in this PR and will keep it as is