-
Notifications
You must be signed in to change notification settings - Fork 16
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
**Which problem is this pull request solving?** Adds a `purgeCache` helper for https://www.notion.so/netlify/Cache-Purge-API-12b8eb7359c549a4aad56d528f19feb0, using the environment variable added in netlify/serverless-functions-api#161. I still need to add some tests, but wanted to get the PR up sooner rather than later to get feedback on the approach.
- Loading branch information
1 parent
82f6c12
commit f6098c0
Showing
8 changed files
with
257 additions
and
2 deletions.
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
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
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 |
---|---|---|
@@ -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 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 |
---|---|---|
@@ -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 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 |
---|---|---|
@@ -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', | ||
) | ||
}) |