2023-03-20 16:02:07 +00:00
|
|
|
import { builder, type Handler } from '@netlify/functions';
|
2023-03-10 15:19:57 +00:00
|
|
|
import type { SSRManifest } from 'astro';
|
2022-03-25 16:08:02 +00:00
|
|
|
import { App } from 'astro/app';
|
2023-07-27 16:24:39 +00:00
|
|
|
import { applyPolyfills } from 'astro/app/node';
|
2023-07-17 14:53:10 +00:00
|
|
|
import { ASTRO_LOCALS_HEADER } from './integration-functions.js';
|
2022-03-25 16:08:02 +00:00
|
|
|
|
2023-07-27 16:24:39 +00:00
|
|
|
applyPolyfills();
|
2022-03-25 16:08:02 +00:00
|
|
|
|
2022-06-15 19:49:09 +00:00
|
|
|
export interface Args {
|
2023-01-27 15:20:34 +00:00
|
|
|
builders?: boolean;
|
2022-06-15 19:49:09 +00:00
|
|
|
binaryMediaTypes?: string[];
|
2023-07-28 09:11:13 +00:00
|
|
|
edgeMiddleware: boolean;
|
|
|
|
functionPerRoute: boolean;
|
2022-06-15 19:49:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function parseContentType(header?: string) {
|
|
|
|
return header?.split(';')[0] ?? '';
|
|
|
|
}
|
2022-03-25 16:08:02 +00:00
|
|
|
|
2022-07-19 20:10:15 +00:00
|
|
|
const clientAddressSymbol = Symbol.for('astro.clientAddress');
|
|
|
|
|
2022-03-25 16:08:02 +00:00
|
|
|
export const createExports = (manifest: SSRManifest, args: Args) => {
|
|
|
|
const app = new App(manifest);
|
|
|
|
|
2023-01-27 15:20:34 +00:00
|
|
|
const builders = args.builders ?? false;
|
2022-06-15 19:49:09 +00:00
|
|
|
const binaryMediaTypes = args.binaryMediaTypes ?? [];
|
|
|
|
const knownBinaryMediaTypes = new Set([
|
|
|
|
'audio/3gpp',
|
|
|
|
'audio/3gpp2',
|
|
|
|
'audio/aac',
|
|
|
|
'audio/midi',
|
|
|
|
'audio/mpeg',
|
|
|
|
'audio/ogg',
|
|
|
|
'audio/opus',
|
|
|
|
'audio/wav',
|
|
|
|
'audio/webm',
|
|
|
|
'audio/x-midi',
|
|
|
|
'image/avif',
|
|
|
|
'image/bmp',
|
|
|
|
'image/gif',
|
|
|
|
'image/vnd.microsoft.icon',
|
2022-09-20 19:33:01 +00:00
|
|
|
'image/heif',
|
2022-06-15 19:49:09 +00:00
|
|
|
'image/jpeg',
|
|
|
|
'image/png',
|
|
|
|
'image/svg+xml',
|
|
|
|
'image/tiff',
|
|
|
|
'image/webp',
|
|
|
|
'video/3gpp',
|
|
|
|
'video/3gpp2',
|
|
|
|
'video/mp2t',
|
|
|
|
'video/mp4',
|
|
|
|
'video/mpeg',
|
|
|
|
'video/ogg',
|
|
|
|
'video/x-msvideo',
|
|
|
|
'video/webm',
|
|
|
|
...binaryMediaTypes,
|
|
|
|
]);
|
|
|
|
|
2023-01-27 15:20:34 +00:00
|
|
|
const myHandler: Handler = async (event) => {
|
2022-04-10 21:34:49 +00:00
|
|
|
const { httpMethod, headers, rawUrl, body: requestBody, isBase64Encoded } = event;
|
|
|
|
const init: RequestInit = {
|
|
|
|
method: httpMethod,
|
|
|
|
headers: new Headers(headers as any),
|
|
|
|
};
|
2022-12-19 11:00:00 +00:00
|
|
|
// Attach the event body the request, with proper encoding.
|
2022-04-10 21:34:49 +00:00
|
|
|
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:02 +00:00
|
|
|
|
2023-08-01 14:52:16 +00:00
|
|
|
const routeData = app.match(request);
|
2022-07-19 20:10:15 +00:00
|
|
|
const ip = headers['x-nf-client-connection-ip'];
|
|
|
|
Reflect.set(request, clientAddressSymbol, ip);
|
2023-07-17 14:53:10 +00:00
|
|
|
let locals = {};
|
|
|
|
if (request.headers.has(ASTRO_LOCALS_HEADER)) {
|
|
|
|
let localsAsString = request.headers.get(ASTRO_LOCALS_HEADER);
|
|
|
|
if (localsAsString) {
|
|
|
|
locals = JSON.parse(localsAsString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const response: Response = await app.render(request, routeData, locals);
|
2022-04-12 20:50:10 +00:00
|
|
|
const responseHeaders = Object.fromEntries(response.headers.entries());
|
2022-06-15 19:50:36 +00:00
|
|
|
|
2022-06-15 19:49:09 +00:00
|
|
|
const responseContentType = parseContentType(responseHeaders['content-type']);
|
|
|
|
const responseIsBase64Encoded = knownBinaryMediaTypes.has(responseContentType);
|
|
|
|
|
2022-09-20 19:33:01 +00:00
|
|
|
let responseBody: string;
|
2022-09-20 19:34:51 +00:00
|
|
|
if (responseIsBase64Encoded) {
|
2022-09-20 19:33:01 +00:00
|
|
|
const ab = await response.arrayBuffer();
|
|
|
|
responseBody = Buffer.from(ab).toString('base64');
|
|
|
|
} else {
|
|
|
|
responseBody = await response.text();
|
|
|
|
}
|
2022-06-15 19:50:36 +00:00
|
|
|
|
2022-04-12 20:50:10 +00:00
|
|
|
const fnResponse: any = {
|
|
|
|
statusCode: response.status,
|
|
|
|
headers: responseHeaders,
|
2022-04-10 21:34:49 +00:00
|
|
|
body: responseBody,
|
2022-06-15 19:49:09 +00:00
|
|
|
isBase64Encoded: responseIsBase64Encoded,
|
2022-03-25 16:08:02 +00:00
|
|
|
};
|
2022-04-12 20:50:10 +00:00
|
|
|
|
2023-02-21 13:18:17 +00:00
|
|
|
const cookies = response.headers.get('set-cookie');
|
|
|
|
if (cookies) {
|
|
|
|
fnResponse.multiValueHeaders = {
|
|
|
|
'set-cookie': Array.isArray(cookies) ? cookies : splitCookiesString(cookies),
|
|
|
|
};
|
2022-04-12 20:50:10 +00:00
|
|
|
}
|
|
|
|
|
2022-09-28 20:55:27 +00:00
|
|
|
// Apply cookies set via Astro.cookies.set/delete
|
2022-09-28 20:57:35 +00:00
|
|
|
if (app.setCookieHeaders) {
|
2022-09-28 20:55:27 +00:00
|
|
|
const setCookieHeaders = Array.from(app.setCookieHeaders(response));
|
|
|
|
fnResponse.multiValueHeaders = fnResponse.multiValueHeaders || {};
|
2022-09-28 20:57:35 +00:00
|
|
|
if (!fnResponse.multiValueHeaders['set-cookie']) {
|
2022-09-28 20:55:27 +00:00
|
|
|
fnResponse.multiValueHeaders['set-cookie'] = [];
|
|
|
|
}
|
|
|
|
fnResponse.multiValueHeaders['set-cookie'].push(...setCookieHeaders);
|
|
|
|
}
|
|
|
|
|
2022-04-12 20:50:10 +00:00
|
|
|
return fnResponse;
|
2022-03-25 16:08:51 +00:00
|
|
|
};
|
2022-03-25 16:08:02 +00:00
|
|
|
|
2023-01-27 15:22:30 +00:00
|
|
|
const handler = builders ? builder(myHandler) : myHandler;
|
2023-01-27 15:20:34 +00:00
|
|
|
|
2022-03-25 16:08:02 +00:00
|
|
|
return { handler };
|
|
|
|
};
|
2023-01-06 17:01:54 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
From: https://github.com/nfriedly/set-cookie-parser/blob/5cae030d8ef0f80eec58459e3583d43a07b984cb/lib/set-cookie.js#L144
|
|
|
|
Set-Cookie header field-values are sometimes comma joined in one string. This splits them without choking on commas
|
|
|
|
that are within a single set-cookie field-value, such as in the Expires portion.
|
|
|
|
This is uncommon, but explicitly allowed - see https://tools.ietf.org/html/rfc2616#section-4.2
|
|
|
|
Node.js does this for every header *except* set-cookie - see https://github.com/nodejs/node/blob/d5e363b77ebaf1caf67cd7528224b651c86815c1/lib/_http_incoming.js#L128
|
|
|
|
React Native's fetch does this for *every* header, including set-cookie.
|
|
|
|
Based on: https://github.com/google/j2objc/commit/16820fdbc8f76ca0c33472810ce0cb03d20efe25
|
|
|
|
Credits to: https://github.com/tomball for original and https://github.com/chrusart for JavaScript implementation
|
|
|
|
*/
|
|
|
|
function splitCookiesString(cookiesString: string): string[] {
|
|
|
|
if (Array.isArray(cookiesString)) {
|
|
|
|
return cookiesString;
|
|
|
|
}
|
|
|
|
if (typeof cookiesString !== 'string') {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
let cookiesStrings = [];
|
|
|
|
let pos = 0;
|
|
|
|
let start;
|
|
|
|
let ch;
|
|
|
|
let lastComma;
|
|
|
|
let nextStart;
|
|
|
|
let cookiesSeparatorFound;
|
|
|
|
|
|
|
|
function skipWhitespace() {
|
|
|
|
while (pos < cookiesString.length && /\s/.test(cookiesString.charAt(pos))) {
|
|
|
|
pos += 1;
|
|
|
|
}
|
|
|
|
return pos < cookiesString.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
function notSpecialChar() {
|
|
|
|
ch = cookiesString.charAt(pos);
|
|
|
|
|
|
|
|
return ch !== '=' && ch !== ';' && ch !== ',';
|
|
|
|
}
|
|
|
|
|
|
|
|
while (pos < cookiesString.length) {
|
|
|
|
start = pos;
|
|
|
|
cookiesSeparatorFound = false;
|
|
|
|
|
|
|
|
while (skipWhitespace()) {
|
|
|
|
ch = cookiesString.charAt(pos);
|
|
|
|
if (ch === ',') {
|
|
|
|
// ',' is a cookie separator if we have later first '=', not ';' or ','
|
|
|
|
lastComma = pos;
|
|
|
|
pos += 1;
|
|
|
|
|
|
|
|
skipWhitespace();
|
|
|
|
nextStart = pos;
|
|
|
|
|
|
|
|
while (pos < cookiesString.length && notSpecialChar()) {
|
|
|
|
pos += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// currently special character
|
|
|
|
if (pos < cookiesString.length && cookiesString.charAt(pos) === '=') {
|
|
|
|
// we found cookies separator
|
|
|
|
cookiesSeparatorFound = true;
|
|
|
|
// pos is inside the next cookie, so back up and return it.
|
|
|
|
pos = nextStart;
|
|
|
|
cookiesStrings.push(cookiesString.substring(start, lastComma));
|
|
|
|
start = pos;
|
|
|
|
} else {
|
|
|
|
// in param ',' or param separator ';',
|
|
|
|
// we continue from that comma
|
|
|
|
pos = lastComma + 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
pos += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cookiesSeparatorFound || pos >= cookiesString.length) {
|
|
|
|
cookiesStrings.push(cookiesString.substring(start, cookiesString.length));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return cookiesStrings;
|
|
|
|
}
|