feat: add cache headers to assets in Vercel adapter (#7729)

* feat: cache assets in Vercel adapter

* Update tidy-tips-doubt.md

* chore: update lockfile

* Update packages/integrations/vercel/test/static-assets.test.js

* Update packages/integrations/vercel/test/static-assets.test.js

* Update packages/integrations/vercel/test/static-assets.test.js

* chore: update split test

---------

Co-authored-by: Kid <44045911+kidonng@users.noreply.github.com>
Co-authored-by: Nate Moore <nate@astro.build>
Co-authored-by: Nate Moore <natemoo-re@users.noreply.github.com>
This commit is contained in:
Hee 2023-08-02 08:06:50 +09:00 committed by GitHub
parent 4dd6c7900c
commit 560d0dab1c
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
10 changed files with 144 additions and 3 deletions

View file

@ -0,0 +1,5 @@
---
'@astrojs/vercel': minor
---
Add cache headers to assets in Vercel adapter

View file

@ -154,6 +154,11 @@ export default function vercelEdge({
version: 3, version: 3,
routes: [ routes: [
...getRedirects(routes, _config), ...getRedirects(routes, _config),
{
src: `^/${_config.build.assets}/(.*)$`,
headers: { 'cache-control': 'public, max-age=31536000, immutable' },
continue: true,
},
{ handle: 'filesystem' }, { handle: 'filesystem' },
{ src: '/.*', dest: 'render' }, { src: '/.*', dest: 'render' },
], ],

View file

@ -187,7 +187,16 @@ export default function vercelServerless({
// https://vercel.com/docs/build-output-api/v3#build-output-configuration // https://vercel.com/docs/build-output-api/v3#build-output-configuration
await writeJson(new URL(`./config.json`, _config.outDir), { await writeJson(new URL(`./config.json`, _config.outDir), {
version: 3, version: 3,
routes: [...getRedirects(routes, _config), { handle: 'filesystem' }, ...routeDefinitions], routes: [
...getRedirects(routes, _config),
{
src: `^/${_config.build.assets}/(.*)$`,
headers: { 'cache-control': 'public, max-age=31536000, immutable' },
continue: true,
},
{ handle: 'filesystem' },
...routeDefinitions,
],
...(imageService || imagesConfig ...(imageService || imagesConfig
? { images: imagesConfig ? imagesConfig : defaultImageConfig } ? { images: imagesConfig ? imagesConfig : defaultImageConfig }
: {}), : {}),

View file

@ -71,7 +71,15 @@ export default function vercelStatic({
// https://vercel.com/docs/build-output-api/v3#build-output-configuration // https://vercel.com/docs/build-output-api/v3#build-output-configuration
await writeJson(new URL(`./config.json`, getVercelOutput(_config.root)), { await writeJson(new URL(`./config.json`, getVercelOutput(_config.root)), {
version: 3, version: 3,
routes: [...getRedirects(routes, _config), { handle: 'filesystem' }], routes: [
...getRedirects(routes, _config),
{
src: `^/${_config.build.assets}/(.*)$`,
headers: { 'cache-control': 'public, max-age=31536000, immutable' },
continue: true,
},
{ handle: 'filesystem' },
],
...(imageService || imagesConfig ...(imageService || imagesConfig
? { images: imagesConfig ? imagesConfig : defaultImageConfig } ? { images: imagesConfig ? imagesConfig : defaultImageConfig }
: {}), : {}),

View file

@ -0,0 +1,4 @@
import { defineConfig } from 'astro/config';
export default defineConfig({
});

View file

@ -0,0 +1,9 @@
{
"name": "@test/astro-vercel-static-assets",
"version": "0.0.0",
"private": true,
"dependencies": {
"@astrojs/vercel": "workspace:*",
"astro": "workspace:*"
}
}

View file

@ -0,0 +1,8 @@
<html>
<head>
<title>Testing</title>
</head>
<body>
<h1>Testing</h1>
</body>
</html>

View file

@ -24,6 +24,6 @@ describe('build: split', () => {
it('creates the route definitions in the config.json', async () => { it('creates the route definitions in the config.json', async () => {
const json = await fixture.readFile('../.vercel/output/config.json'); const json = await fixture.readFile('../.vercel/output/config.json');
const config = JSON.parse(json); const config = JSON.parse(json);
expect(config.routes).to.have.a.lengthOf(3); expect(config.routes).to.have.a.lengthOf(4);
}); });
}); });

View file

@ -0,0 +1,84 @@
import { expect } from 'chai';
import { loadFixture } from './test-utils.js';
describe('Static Assets', () => {
/** @type {import('../../../astro/test/test-utils.js').Fixture} */
let fixture;
const VALID_CACHE_CONTROL = 'public, max-age=31536000, immutable';
async function build({ adapter, assets }) {
fixture = await loadFixture({
root: './fixtures/static-assets/',
adapter,
build: {
assets,
}
});
await fixture.build();
}
async function getConfig() {
const json = await fixture.readFile('../.vercel/output/config.json');
const config = JSON.parse(json);
return config;
}
async function getAssets() {
return fixture.config.build.assets;
}
async function checkValidCacheControl(assets) {
const config = await getConfig();
const route = config.routes.find((r) => r.src === `^/${assets ?? getAssets()}/(.*)$`);
expect(route.headers['cache-control']).to.equal(VALID_CACHE_CONTROL);
expect(route.continue).to.equal(true);
}
describe('static adapter', async () => {
const adapter = await import('@astrojs/vercel/static');
it('has cache control', async () => {
await build({ adapter });
checkValidCacheControl();
});
it('has cache control other assets', async () => {
const assets = '_foo';
await build({ adapter, assets });
checkValidCacheControl(assets);
});
});
describe('serverless adapter', async () => {
const adapter = await import('@astrojs/vercel/serverless');
it('has cache control', async () => {
await build({ adapter });
checkValidCacheControl();
});
it('has cache control other assets', async () => {
const assets = '_foo';
await build({ adapter, assets });
checkValidCacheControl(assets);
});
});
describe('edge adapter', async () => {
const adapter = await import('@astrojs/vercel/edge');
it('has cache control', async () => {
await build({ adapter });
checkValidCacheControl();
});
it('has cache control other assets', async () => {
const assets = '_foo';
await build({ adapter, assets });
checkValidCacheControl(assets);
});
});
});

View file

@ -5047,6 +5047,15 @@ importers:
specifier: workspace:* specifier: workspace:*
version: link:../../../../../astro version: link:../../../../../astro
packages/integrations/vercel/test/fixtures/static-assets:
dependencies:
'@astrojs/vercel':
specifier: workspace:*
version: link:../../..
astro:
specifier: workspace:*
version: link:../../../../../astro
packages/integrations/vercel/test/hosted/hosted-astro-project: packages/integrations/vercel/test/hosted/hosted-astro-project:
dependencies: dependencies:
'@astrojs/vercel': '@astrojs/vercel':