This repository was archived by the owner on Jun 9, 2025. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 18
feat: add purgeCache
helper
#433
Merged
Merged
Changes from all commits
Commits
Show all changes
8 commits
Select commit
Hold shift + click to select a range
656cfb5
feat: add `purgeCache` helper
eduardoboucas 9eff5eb
fix: oops
eduardoboucas f6368ca
fix: use correct environment variable
eduardoboucas 51998f3
refactor: accept additional parameters
eduardoboucas 7782a3b
refactor: check response status
eduardoboucas e70b66f
refactor: adjust parameters
eduardoboucas 542c520
chore: add tests
eduardoboucas 514d90d
chore: skip tests on Node <18
eduardoboucas 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 hidden or 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
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains hidden or 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 hidden or 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 |
---|---|---|
@@ -0,0 +1,82 @@ | ||
import { env } from 'process' | ||
|
||
interface BasePurgeCacheOptions { | ||
apiURL?: string | ||
deployAlias?: string | ||
tags?: string[] | ||
token?: string | ||
} | ||
|
||
interface PurgeCacheOptionsWithSiteID extends BasePurgeCacheOptions { | ||
siteID?: string | ||
} | ||
|
||
interface PurgeCacheOptionsWithSiteSlug extends BasePurgeCacheOptions { | ||
siteSlug: string | ||
} | ||
|
||
interface PurgeCacheOptionsWithDomain extends BasePurgeCacheOptions { | ||
domain: string | ||
} | ||
|
||
type PurgeCacheOptions = PurgeCacheOptionsWithSiteID | PurgeCacheOptionsWithSiteSlug | PurgeCacheOptionsWithDomain | ||
|
||
interface PurgeAPIPayload { | ||
cache_tags?: string[] | ||
deploy_alias?: string | ||
domain?: string | ||
site_id?: string | ||
site_slug?: string | ||
} | ||
|
||
export const purgeCache = async (options: PurgeCacheOptions = {}) => { | ||
if (globalThis.fetch === undefined) { | ||
throw new Error( | ||
"`fetch` is not available. Please ensure you're using Node.js version 18.0.0 or above. Refer to https://ntl.fyi/functions-runtime for more information.", | ||
) | ||
} | ||
|
||
const payload: PurgeAPIPayload = { | ||
cache_tags: options.tags, | ||
deploy_alias: options.deployAlias, | ||
} | ||
const token = env.NETLIFY_PURGE_API_TOKEN || options.token | ||
|
||
if ('siteSlug' in options) { | ||
payload.site_slug = options.siteSlug | ||
} else if ('domain' in options) { | ||
payload.domain = options.domain | ||
} else { | ||
// The `siteID` from `options` takes precedence over the one from the | ||
// environment. | ||
const siteID = options.siteID || env.SITE_ID | ||
|
||
if (!siteID) { | ||
throw new Error( | ||
'The Netlify site ID was not found in the execution environment. Please supply it manually using the `siteID` property.', | ||
) | ||
} | ||
|
||
payload.site_id = siteID | ||
} | ||
|
||
if (!token) { | ||
throw new Error( | ||
'The cache purge API token was not found in the execution environment. Please supply it manually using the `token` property.', | ||
) | ||
} | ||
|
||
const apiURL = options.apiURL || 'https://api.netlify.com' | ||
const response = await fetch(`${apiURL}/api/v1/purge`, { | ||
method: 'POST', | ||
headers: { | ||
'Content-Type': 'application/json; charset=utf8', | ||
Authorization: `Bearer ${token}`, | ||
}, | ||
body: JSON.stringify(payload), | ||
}) | ||
|
||
if (!response.ok) { | ||
throw new Error(`Cache purge API call returned an unexpected status code: ${response.status}`) | ||
} | ||
} |
This file contains hidden or 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 hidden or 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 |
---|---|---|
@@ -0,0 +1,74 @@ | ||
const assert = require('assert') | ||
|
||
module.exports = class MockFetch { | ||
constructor() { | ||
this.requests = [] | ||
} | ||
|
||
addExpectedRequest({ body, headers = {}, method, response, url }) { | ||
this.requests.push({ body, fulfilled: false, headers, method, response, url }) | ||
|
||
return this | ||
} | ||
|
||
delete(options) { | ||
return this.addExpectedRequest({ ...options, method: 'delete' }) | ||
} | ||
|
||
get(options) { | ||
return this.addExpectedRequest({ ...options, method: 'get' }) | ||
} | ||
|
||
post(options) { | ||
return this.addExpectedRequest({ ...options, method: 'post' }) | ||
} | ||
|
||
put(options) { | ||
return this.addExpectedRequest({ ...options, method: 'put' }) | ||
} | ||
|
||
get fetcher() { | ||
// eslint-disable-next-line require-await | ||
return async (...args) => { | ||
const [url, options] = args | ||
const headers = options?.headers | ||
const urlString = url.toString() | ||
const match = this.requests.find( | ||
(request) => | ||
request.method.toLowerCase() === options?.method.toLowerCase() && | ||
request.url === urlString && | ||
!request.fulfilled, | ||
) | ||
|
||
if (!match) { | ||
throw new Error(`Unexpected fetch call: ${url}`) | ||
} | ||
|
||
for (const key in match.headers) { | ||
assert.equal(headers[key], match.headers[key]) | ||
} | ||
|
||
if (typeof match.body === 'string') { | ||
assert.equal(options?.body, match.body) | ||
} else if (typeof match.body === 'function') { | ||
const bodyFn = match.body | ||
|
||
bodyFn(options?.body) | ||
} else { | ||
assert.equal(options?.body, undefined) | ||
} | ||
|
||
match.fulfilled = true | ||
|
||
if (match.response instanceof Error) { | ||
throw match.response | ||
} | ||
|
||
return match.response | ||
} | ||
} | ||
|
||
get fulfilled() { | ||
return this.requests.every((request) => request.fulfilled) | ||
} | ||
} |
This file contains hidden or 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 hidden or 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 |
---|---|---|
@@ -0,0 +1,92 @@ | ||
const process = require('process') | ||
|
||
const test = require('ava') | ||
const semver = require('semver') | ||
|
||
const { purgeCache } = require('../../dist/lib/purge_cache') | ||
const { invokeLambda } = require('../helpers/main') | ||
const MockFetch = require('../helpers/mock_fetch') | ||
|
||
const globalFetch = globalThis.fetch | ||
const hasFetchAPI = semver.gte(process.version, '18.0.0') | ||
|
||
test.beforeEach(() => { | ||
delete process.env.NETLIFY_PURGE_API_TOKEN | ||
delete process.env.SITE_ID | ||
}) | ||
|
||
test.afterEach(() => { | ||
globalThis.fetch = globalFetch | ||
}) | ||
|
||
test.serial('Calls the purge API endpoint and returns `undefined` if the operation was successful', async (t) => { | ||
if (!hasFetchAPI) { | ||
console.warn('Skipping test requires the fetch API') | ||
|
||
return t.pass() | ||
} | ||
|
||
const mockSiteID = '123456789' | ||
const mockToken = '1q2w3e4r5t6y7u8i9o0p' | ||
|
||
process.env.NETLIFY_PURGE_API_TOKEN = mockToken | ||
process.env.SITE_ID = mockSiteID | ||
|
||
const mockAPI = new MockFetch().post({ | ||
body: (payload) => { | ||
const data = JSON.parse(payload) | ||
|
||
t.is(data.site_id, mockSiteID) | ||
}, | ||
headers: { Authorization: `Bearer ${mockToken}` }, | ||
method: 'post', | ||
response: new Response(null, { status: 202 }), | ||
url: `https://api.netlify.com/api/v1/purge`, | ||
}) | ||
const myFunction = async () => { | ||
await purgeCache() | ||
} | ||
|
||
globalThis.fetch = mockAPI.fetcher | ||
|
||
const response = await invokeLambda(myFunction) | ||
|
||
t.is(response, undefined) | ||
t.true(mockAPI.fulfilled) | ||
}) | ||
|
||
test.serial('Throws if the API response does not have a successful status code', async (t) => { | ||
if (!hasFetchAPI) { | ||
console.warn('Skipping test requires the fetch API') | ||
|
||
return t.pass() | ||
} | ||
|
||
const mockSiteID = '123456789' | ||
const mockToken = '1q2w3e4r5t6y7u8i9o0p' | ||
|
||
process.env.NETLIFY_PURGE_API_TOKEN = mockToken | ||
process.env.SITE_ID = mockSiteID | ||
|
||
const mockAPI = new MockFetch().post({ | ||
body: (payload) => { | ||
const data = JSON.parse(payload) | ||
|
||
t.is(data.site_id, mockSiteID) | ||
}, | ||
headers: { Authorization: `Bearer ${mockToken}` }, | ||
method: 'post', | ||
response: new Response(null, { status: 500 }), | ||
url: `https://api.netlify.com/api/v1/purge`, | ||
}) | ||
const myFunction = async () => { | ||
await purgeCache() | ||
} | ||
|
||
globalThis.fetch = mockAPI.fetcher | ||
|
||
await t.throwsAsync( | ||
async () => await invokeLambda(myFunction), | ||
'Cache purge API call returned an unexpected status code: 500', | ||
) | ||
}) |
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.
Uh oh!
There was an error while loading. Please reload this page.