Implement the Astro.response RFC (#3289)
* Implement the Astro.response RFC * Adds a changeset * Mark Astro.response.headers as readonly
This commit is contained in:
parent
d0a4064d80
commit
61e1a267a8
19 changed files with 177 additions and 15 deletions
5
.changeset/angry-rats-turn.md
Normal file
5
.changeset/angry-rats-turn.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Implements the Astro.response RFC
|
|
@ -131,6 +131,18 @@ export interface AstroGlobal extends AstroGlobalPartial {
|
||||||
* [Astro reference](https://docs.astro.build/en/reference/api-reference/#astrorequest)
|
* [Astro reference](https://docs.astro.build/en/reference/api-reference/#astrorequest)
|
||||||
*/
|
*/
|
||||||
request: Request;
|
request: Request;
|
||||||
|
/** Information about the outgoing response. This is a standard [ResponseInit](https://developer.mozilla.org/en-US/docs/Web/API/Response/Response#init) object
|
||||||
|
*
|
||||||
|
* For example, to change the status code you can set a different status on this object:
|
||||||
|
* ```typescript
|
||||||
|
* Astro.response.status = 404;
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
* [Astro reference](https://docs.astro.build/en/reference/api-reference/#astroresponse)
|
||||||
|
*/
|
||||||
|
response: ResponseInit & {
|
||||||
|
readonly headers: Headers
|
||||||
|
};
|
||||||
/** Redirect to another page (**SSR Only**)
|
/** Redirect to another page (**SSR Only**)
|
||||||
*
|
*
|
||||||
* Example usage:
|
* Example usage:
|
||||||
|
@ -1031,5 +1043,6 @@ export interface SSRResult {
|
||||||
slots: Record<string, any> | null
|
slots: Record<string, any> | null
|
||||||
): AstroGlobal;
|
): AstroGlobal;
|
||||||
resolve: (s: string) => Promise<string>;
|
resolve: (s: string) => Promise<string>;
|
||||||
|
response: ResponseInit;
|
||||||
_metadata: SSRMetadata;
|
_metadata: SSRMetadata;
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,14 +108,12 @@ export class App {
|
||||||
}
|
}
|
||||||
|
|
||||||
let html = result.html;
|
let html = result.html;
|
||||||
|
let init = result.response;
|
||||||
|
let headers = init.headers as Headers;
|
||||||
let bytes = this.#encoder.encode(html);
|
let bytes = this.#encoder.encode(html);
|
||||||
return new Response(bytes, {
|
headers.set('Content-Type', 'text/html');
|
||||||
status: 200,
|
headers.set('Content-Length', bytes.byteLength.toString())
|
||||||
headers: {
|
return new Response(bytes, init);
|
||||||
'Content-Type': 'text/html',
|
|
||||||
'Content-Length': bytes.byteLength.toString(),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async #callEndpoint(
|
async #callEndpoint(
|
||||||
|
|
|
@ -84,7 +84,7 @@ export interface RenderOptions {
|
||||||
|
|
||||||
export async function render(
|
export async function render(
|
||||||
opts: RenderOptions
|
opts: RenderOptions
|
||||||
): Promise<{ type: 'html'; html: string } | { type: 'response'; response: Response }> {
|
): Promise<{ type: 'html'; html: string, response: ResponseInit; } | { type: 'response'; response: Response }> {
|
||||||
const {
|
const {
|
||||||
links,
|
links,
|
||||||
logging,
|
logging,
|
||||||
|
@ -162,5 +162,6 @@ export async function render(
|
||||||
return {
|
return {
|
||||||
type: 'html',
|
type: 'html',
|
||||||
html,
|
html,
|
||||||
|
response: result.response
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,7 +45,7 @@ export interface SSROptions {
|
||||||
export type ComponentPreload = [SSRLoadedRenderer[], ComponentInstance];
|
export type ComponentPreload = [SSRLoadedRenderer[], ComponentInstance];
|
||||||
|
|
||||||
export type RenderResponse =
|
export type RenderResponse =
|
||||||
| { type: 'html'; html: string }
|
| { type: 'html'; html: string, response: ResponseInit }
|
||||||
| { type: 'response'; response: Response };
|
| { type: 'response'; response: Response };
|
||||||
|
|
||||||
const svelteStylesRE = /svelte\?svelte&type=style/;
|
const svelteStylesRE = /svelte\?svelte&type=style/;
|
||||||
|
@ -199,6 +199,7 @@ export async function render(
|
||||||
return {
|
return {
|
||||||
type: 'html',
|
type: 'html',
|
||||||
html,
|
html,
|
||||||
|
response: content.response,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -104,6 +104,18 @@ export function createResult(args: CreateResultArgs): SSRResult {
|
||||||
|
|
||||||
const url = new URL(request.url);
|
const url = new URL(request.url);
|
||||||
const canonicalURL = createCanonicalURL('.' + pathname, site ?? url.origin);
|
const canonicalURL = createCanonicalURL('.' + pathname, site ?? url.origin);
|
||||||
|
const response: ResponseInit = {
|
||||||
|
status: 200,
|
||||||
|
statusText: 'OK',
|
||||||
|
headers: new Headers(),
|
||||||
|
};
|
||||||
|
|
||||||
|
// Make headers be read-only
|
||||||
|
Object.defineProperty(response, 'headers', {
|
||||||
|
value: response.headers,
|
||||||
|
enumerable: true,
|
||||||
|
writable: false,
|
||||||
|
});
|
||||||
|
|
||||||
// Create the result object that will be passed into the render function.
|
// Create the result object that will be passed into the render function.
|
||||||
// This object starts here as an empty shell (not yet the result) but then
|
// This object starts here as an empty shell (not yet the result) but then
|
||||||
|
@ -168,6 +180,7 @@ ${extra}`
|
||||||
// Intentionally return an empty string so that it is not relied upon.
|
// Intentionally return an empty string so that it is not relied upon.
|
||||||
return '';
|
return '';
|
||||||
},
|
},
|
||||||
|
response,
|
||||||
slots: astroSlots,
|
slots: astroSlots,
|
||||||
} as unknown as AstroGlobal;
|
} as unknown as AstroGlobal;
|
||||||
|
|
||||||
|
@ -202,6 +215,7 @@ ${extra}`
|
||||||
renderers,
|
renderers,
|
||||||
pathname,
|
pathname,
|
||||||
},
|
},
|
||||||
|
response
|
||||||
};
|
};
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -94,8 +94,13 @@ async function writeSSRResult(
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const { html } = result;
|
const { html, response: init } = result;
|
||||||
writeHtmlResponse(res, statusCode, html);
|
const headers = init.headers as Headers;
|
||||||
|
|
||||||
|
headers.set('Content-Type', 'text/html; charset=utf-8');
|
||||||
|
headers.set('Content-Length', Buffer.byteLength(html, 'utf-8').toString());
|
||||||
|
|
||||||
|
return writeWebResponse(res, new Response(html, init));
|
||||||
}
|
}
|
||||||
|
|
||||||
async function handle404Response(
|
async function handle404Response(
|
||||||
|
|
9
packages/astro/test/fixtures/ssr-response/src/components/OverwriteHeader.astro
vendored
Normal file
9
packages/astro/test/fixtures/ssr-response/src/components/OverwriteHeader.astro
vendored
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
---
|
||||||
|
let gotError = false;
|
||||||
|
try {
|
||||||
|
Astro.response.headers = new Headers();
|
||||||
|
} catch(err) {
|
||||||
|
gotError = true;
|
||||||
|
}
|
||||||
|
---
|
||||||
|
<div id="overwrite-error">{gotError}</div>
|
3
packages/astro/test/fixtures/ssr-response/src/components/SetAHeader.astro
vendored
Normal file
3
packages/astro/test/fixtures/ssr-response/src/components/SetAHeader.astro
vendored
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
---
|
||||||
|
Astro.response.headers.set('Seven-Eight', 'nine');
|
||||||
|
---
|
3
packages/astro/test/fixtures/ssr-response/src/components/SetStatus.astro
vendored
Normal file
3
packages/astro/test/fixtures/ssr-response/src/components/SetStatus.astro
vendored
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
---
|
||||||
|
Astro.response.status = 403;
|
||||||
|
---
|
12
packages/astro/test/fixtures/ssr-response/src/pages/child-set-status.astro
vendored
Normal file
12
packages/astro/test/fixtures/ssr-response/src/pages/child-set-status.astro
vendored
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
---
|
||||||
|
import SetStatus from '../components/SetStatus.astro';
|
||||||
|
---
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>Testing</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>Testing</h1>
|
||||||
|
<SetStatus />
|
||||||
|
</body>
|
||||||
|
</html>
|
14
packages/astro/test/fixtures/ssr-response/src/pages/child-tries-to-overwrite.astro
vendored
Normal file
14
packages/astro/test/fixtures/ssr-response/src/pages/child-tries-to-overwrite.astro
vendored
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
---
|
||||||
|
import SetAHeader from '../components/SetAHeader.astro';
|
||||||
|
import OverwriteHeader from '../components/OverwriteHeader.astro';
|
||||||
|
---
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>Testing</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>Testing</h1>
|
||||||
|
<SetAHeader />
|
||||||
|
<OverwriteHeader />
|
||||||
|
</body>
|
||||||
|
</html>
|
14
packages/astro/test/fixtures/ssr-response/src/pages/some-header.astro
vendored
Normal file
14
packages/astro/test/fixtures/ssr-response/src/pages/some-header.astro
vendored
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
---
|
||||||
|
import SetAHeader from '../components/SetAHeader.astro';
|
||||||
|
Astro.response.headers.set('One-Two', 'three');
|
||||||
|
Astro.response.headers.set('Four-Five', 'six');
|
||||||
|
---
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>Testing</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>Testing</h1>
|
||||||
|
<SetAHeader />
|
||||||
|
</body>
|
||||||
|
</html>
|
12
packages/astro/test/fixtures/ssr-response/src/pages/status-code.astro
vendored
Normal file
12
packages/astro/test/fixtures/ssr-response/src/pages/status-code.astro
vendored
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
---
|
||||||
|
Astro.response.status = 404;
|
||||||
|
Astro.response.statusText = 'Oops';
|
||||||
|
---
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>Testing</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>Testing</h1>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -3,7 +3,6 @@ import { load as cheerioLoad } from 'cheerio';
|
||||||
import { loadFixture } from './test-utils.js';
|
import { loadFixture } from './test-utils.js';
|
||||||
import { viteID } from '../dist/core/util.js';
|
import { viteID } from '../dist/core/util.js';
|
||||||
|
|
||||||
// Asset bundling
|
|
||||||
describe('Integration buildConfig hook', () => {
|
describe('Integration buildConfig hook', () => {
|
||||||
/** @type {import('./test-utils').Fixture} */
|
/** @type {import('./test-utils').Fixture} */
|
||||||
let fixture;
|
let fixture;
|
||||||
|
|
|
@ -2,7 +2,6 @@ import { expect } from 'chai';
|
||||||
import { loadFixture } from './test-utils.js';
|
import { loadFixture } from './test-utils.js';
|
||||||
import testAdapter from './test-adapter.js';
|
import testAdapter from './test-adapter.js';
|
||||||
|
|
||||||
// Asset bundling
|
|
||||||
describe('API routes in SSR', () => {
|
describe('API routes in SSR', () => {
|
||||||
/** @type {import('./test-utils').Fixture} */
|
/** @type {import('./test-utils').Fixture} */
|
||||||
let fixture;
|
let fixture;
|
||||||
|
|
|
@ -3,7 +3,6 @@ import { load as cheerioLoad } from 'cheerio';
|
||||||
import { loadFixture } from './test-utils.js';
|
import { loadFixture } from './test-utils.js';
|
||||||
import testAdapter from './test-adapter.js';
|
import testAdapter from './test-adapter.js';
|
||||||
|
|
||||||
// Asset bundling
|
|
||||||
describe('Dynamic pages in SSR', () => {
|
describe('Dynamic pages in SSR', () => {
|
||||||
/** @type {import('./test-utils').Fixture} */
|
/** @type {import('./test-utils').Fixture} */
|
||||||
let fixture;
|
let fixture;
|
||||||
|
|
|
@ -3,7 +3,6 @@ import { load as cheerioLoad } from 'cheerio';
|
||||||
import { loadFixture } from './test-utils.js';
|
import { loadFixture } from './test-utils.js';
|
||||||
import testAdapter from './test-adapter.js';
|
import testAdapter from './test-adapter.js';
|
||||||
|
|
||||||
// Asset bundling
|
|
||||||
describe('Using Astro.request in SSR', () => {
|
describe('Using Astro.request in SSR', () => {
|
||||||
/** @type {import('./test-utils').Fixture} */
|
/** @type {import('./test-utils').Fixture} */
|
||||||
let fixture;
|
let fixture;
|
||||||
|
|
62
packages/astro/test/ssr-response.test.js
Normal file
62
packages/astro/test/ssr-response.test.js
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
import { expect } from 'chai';
|
||||||
|
import { load as cheerioLoad } from 'cheerio';
|
||||||
|
import { loadFixture } from './test-utils.js';
|
||||||
|
import testAdapter from './test-adapter.js';
|
||||||
|
|
||||||
|
describe('Using Astro.response in SSR', () => {
|
||||||
|
/** @type {import('./test-utils').Fixture} */
|
||||||
|
let fixture;
|
||||||
|
|
||||||
|
before(async () => {
|
||||||
|
fixture = await loadFixture({
|
||||||
|
root: './fixtures/ssr-response/',
|
||||||
|
adapter: testAdapter(),
|
||||||
|
experimental: {
|
||||||
|
ssr: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
await fixture.build();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can set the status', async () => {
|
||||||
|
const app = await fixture.loadTestAdapterApp();
|
||||||
|
const request = new Request('http://example.com/status-code');
|
||||||
|
const response = await app.render(request);
|
||||||
|
expect(response.status).to.equal(404);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can set the statusText', async () => {
|
||||||
|
const app = await fixture.loadTestAdapterApp();
|
||||||
|
const request = new Request('http://example.com/status-code');
|
||||||
|
const response = await app.render(request);
|
||||||
|
expect(response.statusText).to.equal('Oops');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Child component can set status', async () => {
|
||||||
|
const app = await fixture.loadTestAdapterApp();
|
||||||
|
const request = new Request('http://example.com/child-set-status');
|
||||||
|
const response = await app.render(request);
|
||||||
|
expect(response.status).to.equal(403);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can add headers', async () => {
|
||||||
|
const app = await fixture.loadTestAdapterApp();
|
||||||
|
const request = new Request('http://example.com/some-header');
|
||||||
|
const response = await app.render(request);
|
||||||
|
const headers = response.headers;
|
||||||
|
expect(headers.get('one-two')).to.equal('three');
|
||||||
|
expect(headers.get('four-five')).to.equal('six');
|
||||||
|
expect(headers.get('seven-eight')).to.equal('nine');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Child component cannot override headers object', async () => {
|
||||||
|
const app = await fixture.loadTestAdapterApp();
|
||||||
|
const request = new Request('http://example.com/child-tries-to-overwrite');
|
||||||
|
const response = await app.render(request);
|
||||||
|
const headers = response.headers;
|
||||||
|
expect(headers.get('seven-eight')).to.equal('nine');
|
||||||
|
const html = await response.text();
|
||||||
|
const $ = cheerioLoad(html);
|
||||||
|
expect($('#overwrite-error').html()).to.equal('true');
|
||||||
|
});
|
||||||
|
});
|
Loading…
Reference in a new issue