astro/packages/integrations/netlify/src/netlify-functions.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

71 lines
2.2 KiB
TypeScript
Raw Normal View History

2022-06-06 16:49:53 +00:00
import { polyfill } from '@astrojs/webapi';
2022-03-25 16:08:51 +00:00
import type { Handler } from '@netlify/functions';
2022-06-06 16:49:53 +00:00
import { SSRManifest } from 'astro';
import { App } from 'astro/app';
polyfill(globalThis, {
exclude: 'window document',
});
interface Args {}
export const createExports = (manifest: SSRManifest, args: Args) => {
const app = new App(manifest);
const handler: Handler = async (event) => {
const { httpMethod, headers, rawUrl, body: requestBody, isBase64Encoded } = event;
const init: RequestInit = {
method: httpMethod,
headers: new Headers(headers as any),
};
// Attach the event body the the request, with proper encoding.
if (httpMethod !== 'GET' && httpMethod !== 'HEAD') {
const encoding = isBase64Encoded ? 'base64' : 'utf-8';
init.body =
typeof requestBody === 'string' ? Buffer.from(requestBody, encoding) : requestBody;
}
const request = new Request(rawUrl, init);
2022-03-25 16:08:51 +00:00
if (!app.match(request)) {
return {
statusCode: 404,
2022-03-25 16:08:51 +00:00
body: 'Not found',
};
}
const response: Response = await app.render(request);
const responseBody = await response.text();
const responseHeaders = Object.fromEntries(response.headers.entries());
const fnResponse: any = {
statusCode: response.status,
headers: responseHeaders,
body: responseBody,
};
// Special-case set-cookie which has to be set an different way :/
// The fetch API does not have a way to get multiples of a single header, but instead concatenates
// them. There are non-standard ways to do it, and node-fetch gives us headers.raw()
// See https://github.com/whatwg/fetch/issues/973 for discussion
if (response.headers.has('set-cookie') && 'raw' in response.headers) {
// Node fetch allows you to get the raw headers, which includes multiples of the same type.
// This is needed because Set-Cookie *must* be called for each cookie, and can't be
// concatenated together.
type HeadersWithRaw = Headers & {
raw: () => Record<string, string[]>;
};
const rawPacked = (response.headers as HeadersWithRaw).raw();
2022-04-12 20:50:59 +00:00
if ('set-cookie' in rawPacked) {
fnResponse.multiValueHeaders = {
2022-04-12 20:50:59 +00:00
'set-cookie': rawPacked['set-cookie'],
};
}
}
return fnResponse;
2022-03-25 16:08:51 +00:00
};
return { handler };
};