Spaces:
Sleeping
Sleeping
import { H as HttpError, j as json, t as text, b as base, R as Redirect, S as SvelteKitError, o as override, r as reset, a as assets, A as ActionFailure } from './chunks/index-JNnR1J8_.js'; | |
import { d as decode_pathname, h as has_data_suffix, s as strip_data_suffix, a as decode_params, n as normalize_path, b as disable_search, c as add_data_suffix, m as make_trackable, r as resolve } from './chunks/exports-mq_1S73-.js'; | |
import { w as writable, r as readable } from './chunks/index2-mSDuUbym.js'; | |
import { p as public_env, s as safe_public_env, a as set_private_env, b as set_public_env, c as set_safe_public_env } from './chunks/shared-server-49TKSBDM.js'; | |
import { c as create_ssr_component, s as setContext, v as validate_component, m as missing_component } from './chunks/ssr-gTr-ZOTT.js'; | |
import './chunks/utils-UCt__SlX.js'; | |
/** @type {Record<string, string>} */ | |
const escaped = { | |
'<': '\\u003C', | |
'\\': '\\\\', | |
'\b': '\\b', | |
'\f': '\\f', | |
'\n': '\\n', | |
'\r': '\\r', | |
'\t': '\\t', | |
'\u2028': '\\u2028', | |
'\u2029': '\\u2029' | |
}; | |
class DevalueError extends Error { | |
/** | |
* @param {string} message | |
* @param {string[]} keys | |
*/ | |
constructor(message, keys) { | |
super(message); | |
this.name = 'DevalueError'; | |
this.path = keys.join(''); | |
} | |
} | |
/** @param {any} thing */ | |
function is_primitive(thing) { | |
return Object(thing) !== thing; | |
} | |
const object_proto_names = /* @__PURE__ */ Object.getOwnPropertyNames( | |
Object.prototype | |
) | |
.sort() | |
.join('\0'); | |
/** @param {any} thing */ | |
function is_plain_object(thing) { | |
const proto = Object.getPrototypeOf(thing); | |
return ( | |
proto === Object.prototype || | |
proto === null || | |
Object.getOwnPropertyNames(proto).sort().join('\0') === object_proto_names | |
); | |
} | |
/** @param {any} thing */ | |
function get_type(thing) { | |
return Object.prototype.toString.call(thing).slice(8, -1); | |
} | |
/** @param {string} char */ | |
function get_escaped_char(char) { | |
switch (char) { | |
case '"': | |
return '\\"'; | |
case '<': | |
return '\\u003C'; | |
case '\\': | |
return '\\\\'; | |
case '\n': | |
return '\\n'; | |
case '\r': | |
return '\\r'; | |
case '\t': | |
return '\\t'; | |
case '\b': | |
return '\\b'; | |
case '\f': | |
return '\\f'; | |
case '\u2028': | |
return '\\u2028'; | |
case '\u2029': | |
return '\\u2029'; | |
default: | |
return char < ' ' | |
? `\\u${char.charCodeAt(0).toString(16).padStart(4, '0')}` | |
: ''; | |
} | |
} | |
/** @param {string} str */ | |
function stringify_string(str) { | |
let result = ''; | |
let last_pos = 0; | |
const len = str.length; | |
for (let i = 0; i < len; i += 1) { | |
const char = str[i]; | |
const replacement = get_escaped_char(char); | |
if (replacement) { | |
result += str.slice(last_pos, i) + replacement; | |
last_pos = i + 1; | |
} | |
} | |
return `"${last_pos === 0 ? str : result + str.slice(last_pos)}"`; | |
} | |
const chars$1 = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ_$'; | |
const unsafe_chars = /[<\b\f\n\r\t\0\u2028\u2029]/g; | |
const reserved = | |
/^(?:do|if|in|for|int|let|new|try|var|byte|case|char|else|enum|goto|long|this|void|with|await|break|catch|class|const|final|float|short|super|throw|while|yield|delete|double|export|import|native|return|switch|throws|typeof|boolean|default|extends|finally|package|private|abstract|continue|debugger|function|volatile|interface|protected|transient|implements|instanceof|synchronized)$/; | |
/** | |
* Turn a value into the JavaScript that creates an equivalent value | |
* @param {any} value | |
* @param {(value: any) => string | void} [replacer] | |
*/ | |
function uneval(value, replacer) { | |
const counts = new Map(); | |
/** @type {string[]} */ | |
const keys = []; | |
const custom = new Map(); | |
/** @param {any} thing */ | |
function walk(thing) { | |
if (typeof thing === 'function') { | |
throw new DevalueError(`Cannot stringify a function`, keys); | |
} | |
if (!is_primitive(thing)) { | |
if (counts.has(thing)) { | |
counts.set(thing, counts.get(thing) + 1); | |
return; | |
} | |
counts.set(thing, 1); | |
if (replacer) { | |
const str = replacer(thing); | |
if (typeof str === 'string') { | |
custom.set(thing, str); | |
return; | |
} | |
} | |
const type = get_type(thing); | |
switch (type) { | |
case 'Number': | |
case 'BigInt': | |
case 'String': | |
case 'Boolean': | |
case 'Date': | |
case 'RegExp': | |
return; | |
case 'Array': | |
/** @type {any[]} */ (thing).forEach((value, i) => { | |
keys.push(`[${i}]`); | |
walk(value); | |
keys.pop(); | |
}); | |
break; | |
case 'Set': | |
Array.from(thing).forEach(walk); | |
break; | |
case 'Map': | |
for (const [key, value] of thing) { | |
keys.push( | |
`.get(${is_primitive(key) ? stringify_primitive$1(key) : '...'})` | |
); | |
walk(value); | |
keys.pop(); | |
} | |
break; | |
default: | |
if (!is_plain_object(thing)) { | |
throw new DevalueError( | |
`Cannot stringify arbitrary non-POJOs`, | |
keys | |
); | |
} | |
if (Object.getOwnPropertySymbols(thing).length > 0) { | |
throw new DevalueError( | |
`Cannot stringify POJOs with symbolic keys`, | |
keys | |
); | |
} | |
for (const key in thing) { | |
keys.push(`.${key}`); | |
walk(thing[key]); | |
keys.pop(); | |
} | |
} | |
} | |
} | |
walk(value); | |
const names = new Map(); | |
Array.from(counts) | |
.filter((entry) => entry[1] > 1) | |
.sort((a, b) => b[1] - a[1]) | |
.forEach((entry, i) => { | |
names.set(entry[0], get_name(i)); | |
}); | |
/** | |
* @param {any} thing | |
* @returns {string} | |
*/ | |
function stringify(thing) { | |
if (names.has(thing)) { | |
return names.get(thing); | |
} | |
if (is_primitive(thing)) { | |
return stringify_primitive$1(thing); | |
} | |
if (custom.has(thing)) { | |
return custom.get(thing); | |
} | |
const type = get_type(thing); | |
switch (type) { | |
case 'Number': | |
case 'String': | |
case 'Boolean': | |
return `Object(${stringify(thing.valueOf())})`; | |
case 'RegExp': | |
return `new RegExp(${stringify_string(thing.source)}, "${ | |
thing.flags | |
}")`; | |
case 'Date': | |
return `new Date(${thing.getTime()})`; | |
case 'Array': | |
const members = /** @type {any[]} */ (thing).map((v, i) => | |
i in thing ? stringify(v) : '' | |
); | |
const tail = thing.length === 0 || thing.length - 1 in thing ? '' : ','; | |
return `[${members.join(',')}${tail}]`; | |
case 'Set': | |
case 'Map': | |
return `new ${type}([${Array.from(thing).map(stringify).join(',')}])`; | |
default: | |
const obj = `{${Object.keys(thing) | |
.map((key) => `${safe_key(key)}:${stringify(thing[key])}`) | |
.join(',')}}`; | |
const proto = Object.getPrototypeOf(thing); | |
if (proto === null) { | |
return Object.keys(thing).length > 0 | |
? `Object.assign(Object.create(null),${obj})` | |
: `Object.create(null)`; | |
} | |
return obj; | |
} | |
} | |
const str = stringify(value); | |
if (names.size) { | |
/** @type {string[]} */ | |
const params = []; | |
/** @type {string[]} */ | |
const statements = []; | |
/** @type {string[]} */ | |
const values = []; | |
names.forEach((name, thing) => { | |
params.push(name); | |
if (custom.has(thing)) { | |
values.push(/** @type {string} */ (custom.get(thing))); | |
return; | |
} | |
if (is_primitive(thing)) { | |
values.push(stringify_primitive$1(thing)); | |
return; | |
} | |
const type = get_type(thing); | |
switch (type) { | |
case 'Number': | |
case 'String': | |
case 'Boolean': | |
values.push(`Object(${stringify(thing.valueOf())})`); | |
break; | |
case 'RegExp': | |
values.push(thing.toString()); | |
break; | |
case 'Date': | |
values.push(`new Date(${thing.getTime()})`); | |
break; | |
case 'Array': | |
values.push(`Array(${thing.length})`); | |
/** @type {any[]} */ (thing).forEach((v, i) => { | |
statements.push(`${name}[${i}]=${stringify(v)}`); | |
}); | |
break; | |
case 'Set': | |
values.push(`new Set`); | |
statements.push( | |
`${name}.${Array.from(thing) | |
.map((v) => `add(${stringify(v)})`) | |
.join('.')}` | |
); | |
break; | |
case 'Map': | |
values.push(`new Map`); | |
statements.push( | |
`${name}.${Array.from(thing) | |
.map(([k, v]) => `set(${stringify(k)}, ${stringify(v)})`) | |
.join('.')}` | |
); | |
break; | |
default: | |
values.push( | |
Object.getPrototypeOf(thing) === null ? 'Object.create(null)' : '{}' | |
); | |
Object.keys(thing).forEach((key) => { | |
statements.push( | |
`${name}${safe_prop(key)}=${stringify(thing[key])}` | |
); | |
}); | |
} | |
}); | |
statements.push(`return ${str}`); | |
return `(function(${params.join(',')}){${statements.join( | |
';' | |
)}}(${values.join(',')}))`; | |
} else { | |
return str; | |
} | |
} | |
/** @param {number} num */ | |
function get_name(num) { | |
let name = ''; | |
do { | |
name = chars$1[num % chars$1.length] + name; | |
num = ~~(num / chars$1.length) - 1; | |
} while (num >= 0); | |
return reserved.test(name) ? `${name}0` : name; | |
} | |
/** @param {string} c */ | |
function escape_unsafe_char(c) { | |
return escaped[c] || c; | |
} | |
/** @param {string} str */ | |
function escape_unsafe_chars(str) { | |
return str.replace(unsafe_chars, escape_unsafe_char); | |
} | |
/** @param {string} key */ | |
function safe_key(key) { | |
return /^[_$a-zA-Z][_$a-zA-Z0-9]*$/.test(key) | |
? key | |
: escape_unsafe_chars(JSON.stringify(key)); | |
} | |
/** @param {string} key */ | |
function safe_prop(key) { | |
return /^[_$a-zA-Z][_$a-zA-Z0-9]*$/.test(key) | |
? `.${key}` | |
: `[${escape_unsafe_chars(JSON.stringify(key))}]`; | |
} | |
/** @param {any} thing */ | |
function stringify_primitive$1(thing) { | |
if (typeof thing === 'string') return stringify_string(thing); | |
if (thing === void 0) return 'void 0'; | |
if (thing === 0 && 1 / thing < 0) return '-0'; | |
const str = String(thing); | |
if (typeof thing === 'number') return str.replace(/^(-)?0\./, '$1.'); | |
if (typeof thing === 'bigint') return thing + 'n'; | |
return str; | |
} | |
const UNDEFINED = -1; | |
const HOLE = -2; | |
const NAN = -3; | |
const POSITIVE_INFINITY = -4; | |
const NEGATIVE_INFINITY = -5; | |
const NEGATIVE_ZERO = -6; | |
/** | |
* Turn a value into a JSON string that can be parsed with `devalue.parse` | |
* @param {any} value | |
* @param {Record<string, (value: any) => any>} [reducers] | |
*/ | |
function stringify(value, reducers) { | |
/** @type {any[]} */ | |
const stringified = []; | |
/** @type {Map<any, number>} */ | |
const indexes = new Map(); | |
/** @type {Array<{ key: string, fn: (value: any) => any }>} */ | |
const custom = []; | |
for (const key in reducers) { | |
custom.push({ key, fn: reducers[key] }); | |
} | |
/** @type {string[]} */ | |
const keys = []; | |
let p = 0; | |
/** @param {any} thing */ | |
function flatten(thing) { | |
if (typeof thing === 'function') { | |
throw new DevalueError(`Cannot stringify a function`, keys); | |
} | |
if (indexes.has(thing)) return indexes.get(thing); | |
if (thing === undefined) return UNDEFINED; | |
if (Number.isNaN(thing)) return NAN; | |
if (thing === Infinity) return POSITIVE_INFINITY; | |
if (thing === -Infinity) return NEGATIVE_INFINITY; | |
if (thing === 0 && 1 / thing < 0) return NEGATIVE_ZERO; | |
const index = p++; | |
indexes.set(thing, index); | |
for (const { key, fn } of custom) { | |
const value = fn(thing); | |
if (value) { | |
stringified[index] = `["${key}",${flatten(value)}]`; | |
return index; | |
} | |
} | |
let str = ''; | |
if (is_primitive(thing)) { | |
str = stringify_primitive(thing); | |
} else { | |
const type = get_type(thing); | |
switch (type) { | |
case 'Number': | |
case 'String': | |
case 'Boolean': | |
str = `["Object",${stringify_primitive(thing)}]`; | |
break; | |
case 'BigInt': | |
str = `["BigInt",${thing}]`; | |
break; | |
case 'Date': | |
str = `["Date","${thing.toISOString()}"]`; | |
break; | |
case 'RegExp': | |
const { source, flags } = thing; | |
str = flags | |
? `["RegExp",${stringify_string(source)},"${flags}"]` | |
: `["RegExp",${stringify_string(source)}]`; | |
break; | |
case 'Array': | |
str = '['; | |
for (let i = 0; i < thing.length; i += 1) { | |
if (i > 0) str += ','; | |
if (i in thing) { | |
keys.push(`[${i}]`); | |
str += flatten(thing[i]); | |
keys.pop(); | |
} else { | |
str += HOLE; | |
} | |
} | |
str += ']'; | |
break; | |
case 'Set': | |
str = '["Set"'; | |
for (const value of thing) { | |
str += `,${flatten(value)}`; | |
} | |
str += ']'; | |
break; | |
case 'Map': | |
str = '["Map"'; | |
for (const [key, value] of thing) { | |
keys.push( | |
`.get(${is_primitive(key) ? stringify_primitive(key) : '...'})` | |
); | |
str += `,${flatten(key)},${flatten(value)}`; | |
} | |
str += ']'; | |
break; | |
default: | |
if (!is_plain_object(thing)) { | |
throw new DevalueError( | |
`Cannot stringify arbitrary non-POJOs`, | |
keys | |
); | |
} | |
if (Object.getOwnPropertySymbols(thing).length > 0) { | |
throw new DevalueError( | |
`Cannot stringify POJOs with symbolic keys`, | |
keys | |
); | |
} | |
if (Object.getPrototypeOf(thing) === null) { | |
str = '["null"'; | |
for (const key in thing) { | |
keys.push(`.${key}`); | |
str += `,${stringify_string(key)},${flatten(thing[key])}`; | |
keys.pop(); | |
} | |
str += ']'; | |
} else { | |
str = '{'; | |
let started = false; | |
for (const key in thing) { | |
if (started) str += ','; | |
started = true; | |
keys.push(`.${key}`); | |
str += `${stringify_string(key)}:${flatten(thing[key])}`; | |
keys.pop(); | |
} | |
str += '}'; | |
} | |
} | |
} | |
stringified[index] = str; | |
return index; | |
} | |
const index = flatten(value); | |
// special case — value is represented as a negative index | |
if (index < 0) return `${index}`; | |
return `[${stringified.join(',')}]`; | |
} | |
/** | |
* @param {any} thing | |
* @returns {string} | |
*/ | |
function stringify_primitive(thing) { | |
const type = typeof thing; | |
if (type === 'string') return stringify_string(thing); | |
if (thing instanceof String) return stringify_string(thing.toString()); | |
if (thing === void 0) return UNDEFINED.toString(); | |
if (thing === 0 && 1 / thing < 0) return NEGATIVE_ZERO.toString(); | |
if (type === 'bigint') return `["BigInt","${thing}"]`; | |
return String(thing); | |
} | |
var cookie = {}; | |
/*! | |
* cookie | |
* Copyright(c) 2012-2014 Roman Shtylman | |
* Copyright(c) 2015 Douglas Christopher Wilson | |
* MIT Licensed | |
*/ | |
var hasRequiredCookie; | |
function requireCookie () { | |
if (hasRequiredCookie) return cookie; | |
hasRequiredCookie = 1; | |
/** | |
* Module exports. | |
* @public | |
*/ | |
cookie.parse = parse; | |
cookie.serialize = serialize; | |
/** | |
* Module variables. | |
* @private | |
*/ | |
var __toString = Object.prototype.toString; | |
/** | |
* RegExp to match field-content in RFC 7230 sec 3.2 | |
* | |
* field-content = field-vchar [ 1*( SP / HTAB ) field-vchar ] | |
* field-vchar = VCHAR / obs-text | |
* obs-text = %x80-FF | |
*/ | |
var fieldContentRegExp = /^[\u0009\u0020-\u007e\u0080-\u00ff]+$/; | |
/** | |
* Parse a cookie header. | |
* | |
* Parse the given cookie header string into an object | |
* The object has the various cookies as keys(names) => values | |
* | |
* @param {string} str | |
* @param {object} [options] | |
* @return {object} | |
* @public | |
*/ | |
function parse(str, options) { | |
if (typeof str !== 'string') { | |
throw new TypeError('argument str must be a string'); | |
} | |
var obj = {}; | |
var opt = options || {}; | |
var dec = opt.decode || decode; | |
var index = 0; | |
while (index < str.length) { | |
var eqIdx = str.indexOf('=', index); | |
// no more cookie pairs | |
if (eqIdx === -1) { | |
break | |
} | |
var endIdx = str.indexOf(';', index); | |
if (endIdx === -1) { | |
endIdx = str.length; | |
} else if (endIdx < eqIdx) { | |
// backtrack on prior semicolon | |
index = str.lastIndexOf(';', eqIdx - 1) + 1; | |
continue | |
} | |
var key = str.slice(index, eqIdx).trim(); | |
// only assign once | |
if (undefined === obj[key]) { | |
var val = str.slice(eqIdx + 1, endIdx).trim(); | |
// quoted values | |
if (val.charCodeAt(0) === 0x22) { | |
val = val.slice(1, -1); | |
} | |
obj[key] = tryDecode(val, dec); | |
} | |
index = endIdx + 1; | |
} | |
return obj; | |
} | |
/** | |
* Serialize data into a cookie header. | |
* | |
* Serialize the a name value pair into a cookie string suitable for | |
* http headers. An optional options object specified cookie parameters. | |
* | |
* serialize('foo', 'bar', { httpOnly: true }) | |
* => "foo=bar; httpOnly" | |
* | |
* @param {string} name | |
* @param {string} val | |
* @param {object} [options] | |
* @return {string} | |
* @public | |
*/ | |
function serialize(name, val, options) { | |
var opt = options || {}; | |
var enc = opt.encode || encode; | |
if (typeof enc !== 'function') { | |
throw new TypeError('option encode is invalid'); | |
} | |
if (!fieldContentRegExp.test(name)) { | |
throw new TypeError('argument name is invalid'); | |
} | |
var value = enc(val); | |
if (value && !fieldContentRegExp.test(value)) { | |
throw new TypeError('argument val is invalid'); | |
} | |
var str = name + '=' + value; | |
if (null != opt.maxAge) { | |
var maxAge = opt.maxAge - 0; | |
if (isNaN(maxAge) || !isFinite(maxAge)) { | |
throw new TypeError('option maxAge is invalid') | |
} | |
str += '; Max-Age=' + Math.floor(maxAge); | |
} | |
if (opt.domain) { | |
if (!fieldContentRegExp.test(opt.domain)) { | |
throw new TypeError('option domain is invalid'); | |
} | |
str += '; Domain=' + opt.domain; | |
} | |
if (opt.path) { | |
if (!fieldContentRegExp.test(opt.path)) { | |
throw new TypeError('option path is invalid'); | |
} | |
str += '; Path=' + opt.path; | |
} | |
if (opt.expires) { | |
var expires = opt.expires; | |
if (!isDate(expires) || isNaN(expires.valueOf())) { | |
throw new TypeError('option expires is invalid'); | |
} | |
str += '; Expires=' + expires.toUTCString(); | |
} | |
if (opt.httpOnly) { | |
str += '; HttpOnly'; | |
} | |
if (opt.secure) { | |
str += '; Secure'; | |
} | |
if (opt.partitioned) { | |
str += '; Partitioned'; | |
} | |
if (opt.priority) { | |
var priority = typeof opt.priority === 'string' | |
? opt.priority.toLowerCase() | |
: opt.priority; | |
switch (priority) { | |
case 'low': | |
str += '; Priority=Low'; | |
break | |
case 'medium': | |
str += '; Priority=Medium'; | |
break | |
case 'high': | |
str += '; Priority=High'; | |
break | |
default: | |
throw new TypeError('option priority is invalid') | |
} | |
} | |
if (opt.sameSite) { | |
var sameSite = typeof opt.sameSite === 'string' | |
? opt.sameSite.toLowerCase() : opt.sameSite; | |
switch (sameSite) { | |
case true: | |
str += '; SameSite=Strict'; | |
break; | |
case 'lax': | |
str += '; SameSite=Lax'; | |
break; | |
case 'strict': | |
str += '; SameSite=Strict'; | |
break; | |
case 'none': | |
str += '; SameSite=None'; | |
break; | |
default: | |
throw new TypeError('option sameSite is invalid'); | |
} | |
} | |
return str; | |
} | |
/** | |
* URL-decode string value. Optimized to skip native call when no %. | |
* | |
* @param {string} str | |
* @returns {string} | |
*/ | |
function decode (str) { | |
return str.indexOf('%') !== -1 | |
? decodeURIComponent(str) | |
: str | |
} | |
/** | |
* URL-encode value. | |
* | |
* @param {string} val | |
* @returns {string} | |
*/ | |
function encode (val) { | |
return encodeURIComponent(val) | |
} | |
/** | |
* Determine if value is a Date. | |
* | |
* @param {*} val | |
* @private | |
*/ | |
function isDate (val) { | |
return __toString.call(val) === '[object Date]' || | |
val instanceof Date | |
} | |
/** | |
* Try decoding a string using a decoding function. | |
* | |
* @param {string} str | |
* @param {function} decode | |
* @private | |
*/ | |
function tryDecode(str, decode) { | |
try { | |
return decode(str); | |
} catch (e) { | |
return str; | |
} | |
} | |
return cookie; | |
} | |
var cookieExports = requireCookie(); | |
var setCookie = {exports: {}}; | |
var hasRequiredSetCookie; | |
function requireSetCookie () { | |
if (hasRequiredSetCookie) return setCookie.exports; | |
hasRequiredSetCookie = 1; | |
var defaultParseOptions = { | |
decodeValues: true, | |
map: false, | |
silent: false, | |
}; | |
function isNonEmptyString(str) { | |
return typeof str === "string" && !!str.trim(); | |
} | |
function parseString(setCookieValue, options) { | |
var parts = setCookieValue.split(";").filter(isNonEmptyString); | |
var nameValuePairStr = parts.shift(); | |
var parsed = parseNameValuePair(nameValuePairStr); | |
var name = parsed.name; | |
var value = parsed.value; | |
options = options | |
? Object.assign({}, defaultParseOptions, options) | |
: defaultParseOptions; | |
try { | |
value = options.decodeValues ? decodeURIComponent(value) : value; // decode cookie value | |
} catch (e) { | |
console.error( | |
"set-cookie-parser encountered an error while decoding a cookie with value '" + | |
value + | |
"'. Set options.decodeValues to false to disable this feature.", | |
e | |
); | |
} | |
var cookie = { | |
name: name, | |
value: value, | |
}; | |
parts.forEach(function (part) { | |
var sides = part.split("="); | |
var key = sides.shift().trimLeft().toLowerCase(); | |
var value = sides.join("="); | |
if (key === "expires") { | |
cookie.expires = new Date(value); | |
} else if (key === "max-age") { | |
cookie.maxAge = parseInt(value, 10); | |
} else if (key === "secure") { | |
cookie.secure = true; | |
} else if (key === "httponly") { | |
cookie.httpOnly = true; | |
} else if (key === "samesite") { | |
cookie.sameSite = value; | |
} else { | |
cookie[key] = value; | |
} | |
}); | |
return cookie; | |
} | |
function parseNameValuePair(nameValuePairStr) { | |
// Parses name-value-pair according to rfc6265bis draft | |
var name = ""; | |
var value = ""; | |
var nameValueArr = nameValuePairStr.split("="); | |
if (nameValueArr.length > 1) { | |
name = nameValueArr.shift(); | |
value = nameValueArr.join("="); // everything after the first =, joined by a "=" if there was more than one part | |
} else { | |
value = nameValuePairStr; | |
} | |
return { name: name, value: value }; | |
} | |
function parse(input, options) { | |
options = options | |
? Object.assign({}, defaultParseOptions, options) | |
: defaultParseOptions; | |
if (!input) { | |
if (!options.map) { | |
return []; | |
} else { | |
return {}; | |
} | |
} | |
if (input.headers) { | |
if (typeof input.headers.getSetCookie === "function") { | |
// for fetch responses - they combine headers of the same type in the headers array, | |
// but getSetCookie returns an uncombined array | |
input = input.headers.getSetCookie(); | |
} else if (input.headers["set-cookie"]) { | |
// fast-path for node.js (which automatically normalizes header names to lower-case | |
input = input.headers["set-cookie"]; | |
} else { | |
// slow-path for other environments - see #25 | |
var sch = | |
input.headers[ | |
Object.keys(input.headers).find(function (key) { | |
return key.toLowerCase() === "set-cookie"; | |
}) | |
]; | |
// warn if called on a request-like object with a cookie header rather than a set-cookie header - see #34, 36 | |
if (!sch && input.headers.cookie && !options.silent) { | |
console.warn( | |
"Warning: set-cookie-parser appears to have been called on a request object. It is designed to parse Set-Cookie headers from responses, not Cookie headers from requests. Set the option {silent: true} to suppress this warning." | |
); | |
} | |
input = sch; | |
} | |
} | |
if (!Array.isArray(input)) { | |
input = [input]; | |
} | |
options = options | |
? Object.assign({}, defaultParseOptions, options) | |
: defaultParseOptions; | |
if (!options.map) { | |
return input.filter(isNonEmptyString).map(function (str) { | |
return parseString(str, options); | |
}); | |
} else { | |
var cookies = {}; | |
return input.filter(isNonEmptyString).reduce(function (cookies, str) { | |
var cookie = parseString(str, options); | |
cookies[cookie.name] = cookie; | |
return cookies; | |
}, cookies); | |
} | |
} | |
/* | |
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) { | |
if (Array.isArray(cookiesString)) { | |
return cookiesString; | |
} | |
if (typeof cookiesString !== "string") { | |
return []; | |
} | |
var cookiesStrings = []; | |
var pos = 0; | |
var start; | |
var ch; | |
var lastComma; | |
var nextStart; | |
var 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; | |
} | |
setCookie.exports = parse; | |
setCookie.exports.parse = parse; | |
setCookie.exports.parseString = parseString; | |
setCookie.exports.splitCookiesString = splitCookiesString; | |
return setCookie.exports; | |
} | |
var setCookieExports = requireSetCookie(); | |
function afterUpdate() { | |
} | |
const Root = create_ssr_component(($$result, $$props, $$bindings, slots) => { | |
let { stores } = $$props; | |
let { page } = $$props; | |
let { constructors } = $$props; | |
let { components = [] } = $$props; | |
let { form } = $$props; | |
let { data_0 = null } = $$props; | |
let { data_1 = null } = $$props; | |
{ | |
setContext("__svelte__", stores); | |
} | |
afterUpdate(stores.page.notify); | |
if ($$props.stores === void 0 && $$bindings.stores && stores !== void 0) | |
$$bindings.stores(stores); | |
if ($$props.page === void 0 && $$bindings.page && page !== void 0) | |
$$bindings.page(page); | |
if ($$props.constructors === void 0 && $$bindings.constructors && constructors !== void 0) | |
$$bindings.constructors(constructors); | |
if ($$props.components === void 0 && $$bindings.components && components !== void 0) | |
$$bindings.components(components); | |
if ($$props.form === void 0 && $$bindings.form && form !== void 0) | |
$$bindings.form(form); | |
if ($$props.data_0 === void 0 && $$bindings.data_0 && data_0 !== void 0) | |
$$bindings.data_0(data_0); | |
if ($$props.data_1 === void 0 && $$bindings.data_1 && data_1 !== void 0) | |
$$bindings.data_1(data_1); | |
let $$settled; | |
let $$rendered; | |
let previous_head = $$result.head; | |
do { | |
$$settled = true; | |
$$result.head = previous_head; | |
{ | |
stores.page.set(page); | |
} | |
$$rendered = ` ${constructors[1] ? `${validate_component(constructors[0] || missing_component, "svelte:component").$$render( | |
$$result, | |
{ data: data_0, this: components[0] }, | |
{ | |
this: ($$value) => { | |
components[0] = $$value; | |
$$settled = false; | |
} | |
}, | |
{ | |
default: () => { | |
return `${validate_component(constructors[1] || missing_component, "svelte:component").$$render( | |
$$result, | |
{ data: data_1, form, this: components[1] }, | |
{ | |
this: ($$value) => { | |
components[1] = $$value; | |
$$settled = false; | |
} | |
}, | |
{} | |
)}`; | |
} | |
} | |
)}` : `${validate_component(constructors[0] || missing_component, "svelte:component").$$render( | |
$$result, | |
{ data: data_0, form, this: components[0] }, | |
{ | |
this: ($$value) => { | |
components[0] = $$value; | |
$$settled = false; | |
} | |
}, | |
{} | |
)}`} ${``}`; | |
} while (!$$settled); | |
return $$rendered; | |
}); | |
const options = { | |
app_dir: "_app", | |
app_template_contains_nonce: false, | |
csp: { "mode": "auto", "directives": { "upgrade-insecure-requests": false, "block-all-mixed-content": false }, "reportOnly": { "upgrade-insecure-requests": false, "block-all-mixed-content": false } }, | |
csrf_check_origin: true, | |
embedded: false, | |
env_public_prefix: "PUBLIC_", | |
env_private_prefix: "", | |
hooks: null, | |
// added lazily, via `get_hooks` | |
preload_strategy: "modulepreload", | |
root: Root, | |
service_worker: false, | |
templates: { | |
app: ({ head, body, assets, nonce, env }) => '<!doctype html>\r\n<html lang="en">\r\n <head>\r\n <meta charset="utf-8" />\r\n <link rel="icon" href="' + assets + '/favicon.png" />\r\n <meta name="viewport" content="width=device-width, initial-scale=1" />\r\n ' + head + '\r\n </head>\r\n <body data-sveltekit-preload-data="hover" data-theme="skeleton" style="overflow-x: hidden;">\r\n <div style="display: contents">' + body + "</div>\r\n </body>\r\n</html>\r\n", | |
error: ({ status, message }) => '<!doctype html>\n<html lang="en">\n <head>\n <meta charset="utf-8" />\n <title>' + message + `</title> | |
<style> | |
body { | |
--bg: white; | |
--fg: #222; | |
--divider: #ccc; | |
background: var(--bg); | |
color: var(--fg); | |
font-family: | |
system-ui, | |
-apple-system, | |
BlinkMacSystemFont, | |
'Segoe UI', | |
Roboto, | |
Oxygen, | |
Ubuntu, | |
Cantarell, | |
'Open Sans', | |
'Helvetica Neue', | |
sans-serif; | |
display: flex; | |
align-items: center; | |
justify-content: center; | |
height: 100vh; | |
margin: 0; | |
} | |
.error { | |
display: flex; | |
align-items: center; | |
max-width: 32rem; | |
margin: 0 1rem; | |
} | |
.status { | |
font-weight: 200; | |
font-size: 3rem; | |
line-height: 1; | |
position: relative; | |
top: -0.05rem; | |
} | |
.message { | |
border-left: 1px solid var(--divider); | |
padding: 0 0 0 1rem; | |
margin: 0 0 0 1rem; | |
min-height: 2.5rem; | |
display: flex; | |
align-items: center; | |
} | |
.message h1 { | |
font-weight: 400; | |
font-size: 1em; | |
margin: 0; | |
} | |
@media (prefers-color-scheme: dark) { | |
body { | |
--bg: #222; | |
--fg: #ddd; | |
--divider: #666; | |
} | |
} | |
</style> | |
</head> | |
<body> | |
<div class="error"> | |
<span class="status">` + status + '</span>\n <div class="message">\n <h1>' + message + "</h1>\n </div>\n </div>\n </body>\n</html>\n" | |
}, | |
version_hash: "c39tcv" | |
}; | |
async function get_hooks() { | |
return { | |
...await import('./chunks/hooks.server-PjMICSnm.js') | |
}; | |
} | |
const DEV = false; | |
const SVELTE_KIT_ASSETS = "/_svelte_kit_assets"; | |
const ENDPOINT_METHODS = ["GET", "POST", "PUT", "PATCH", "DELETE", "OPTIONS", "HEAD"]; | |
const PAGE_METHODS = ["GET", "POST", "HEAD"]; | |
function negotiate(accept, types) { | |
const parts = []; | |
accept.split(",").forEach((str, i) => { | |
const match = /([^/]+)\/([^;]+)(?:;q=([0-9.]+))?/.exec(str); | |
if (match) { | |
const [, type, subtype, q = "1"] = match; | |
parts.push({ type, subtype, q: +q, i }); | |
} | |
}); | |
parts.sort((a, b) => { | |
if (a.q !== b.q) { | |
return b.q - a.q; | |
} | |
if (a.subtype === "*" !== (b.subtype === "*")) { | |
return a.subtype === "*" ? 1 : -1; | |
} | |
if (a.type === "*" !== (b.type === "*")) { | |
return a.type === "*" ? 1 : -1; | |
} | |
return a.i - b.i; | |
}); | |
let accepted; | |
let min_priority = Infinity; | |
for (const mimetype of types) { | |
const [type, subtype] = mimetype.split("/"); | |
const priority = parts.findIndex( | |
(part) => (part.type === type || part.type === "*") && (part.subtype === subtype || part.subtype === "*") | |
); | |
if (priority !== -1 && priority < min_priority) { | |
accepted = mimetype; | |
min_priority = priority; | |
} | |
} | |
return accepted; | |
} | |
function is_content_type(request, ...types) { | |
const type = request.headers.get("content-type")?.split(";", 1)[0].trim() ?? ""; | |
return types.includes(type.toLowerCase()); | |
} | |
function is_form_content_type(request) { | |
return is_content_type( | |
request, | |
"application/x-www-form-urlencoded", | |
"multipart/form-data", | |
"text/plain" | |
); | |
} | |
function coalesce_to_error(err) { | |
return err instanceof Error || err && /** @type {any} */ | |
err.name && /** @type {any} */ | |
err.message ? ( | |
/** @type {Error} */ | |
err | |
) : new Error(JSON.stringify(err)); | |
} | |
function normalize_error(error) { | |
return ( | |
/** @type {import('../runtime/control.js').Redirect | HttpError | SvelteKitError | Error} */ | |
error | |
); | |
} | |
function get_status(error) { | |
return error instanceof HttpError || error instanceof SvelteKitError ? error.status : 500; | |
} | |
function get_message(error) { | |
return error instanceof SvelteKitError ? error.text : "Internal Error"; | |
} | |
function method_not_allowed(mod, method) { | |
return text(`${method} method not allowed`, { | |
status: 405, | |
headers: { | |
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Status/405 | |
// "The server must generate an Allow header field in a 405 status code response" | |
allow: allowed_methods(mod).join(", ") | |
} | |
}); | |
} | |
function allowed_methods(mod) { | |
const allowed = ENDPOINT_METHODS.filter((method) => method in mod); | |
if ("GET" in mod || "HEAD" in mod) | |
allowed.push("HEAD"); | |
return allowed; | |
} | |
function static_error_page(options2, status, message) { | |
let page = options2.templates.error({ status, message }); | |
return text(page, { | |
headers: { "content-type": "text/html; charset=utf-8" }, | |
status | |
}); | |
} | |
async function handle_fatal_error(event, options2, error) { | |
error = error instanceof HttpError ? error : coalesce_to_error(error); | |
const status = get_status(error); | |
const body2 = await handle_error_and_jsonify(event, options2, error); | |
const type = negotiate(event.request.headers.get("accept") || "text/html", [ | |
"application/json", | |
"text/html" | |
]); | |
if (event.isDataRequest || type === "application/json") { | |
return json(body2, { | |
status | |
}); | |
} | |
return static_error_page(options2, status, body2.message); | |
} | |
async function handle_error_and_jsonify(event, options2, error) { | |
if (error instanceof HttpError) { | |
return error.body; | |
} | |
const status = get_status(error); | |
const message = get_message(error); | |
return await options2.hooks.handleError({ error, event, status, message }) ?? { message }; | |
} | |
function redirect_response(status, location) { | |
const response = new Response(void 0, { | |
status, | |
headers: { location } | |
}); | |
return response; | |
} | |
function clarify_devalue_error(event, error) { | |
if (error.path) { | |
return `Data returned from \`load\` while rendering ${event.route.id} is not serializable: ${error.message} (data${error.path})`; | |
} | |
if (error.path === "") { | |
return `Data returned from \`load\` while rendering ${event.route.id} is not a plain object`; | |
} | |
return error.message; | |
} | |
function stringify_uses(node) { | |
const uses = []; | |
if (node.uses && node.uses.dependencies.size > 0) { | |
uses.push(`"dependencies":${JSON.stringify(Array.from(node.uses.dependencies))}`); | |
} | |
if (node.uses && node.uses.search_params.size > 0) { | |
uses.push(`"search_params":${JSON.stringify(Array.from(node.uses.search_params))}`); | |
} | |
if (node.uses && node.uses.params.size > 0) { | |
uses.push(`"params":${JSON.stringify(Array.from(node.uses.params))}`); | |
} | |
if (node.uses?.parent) | |
uses.push('"parent":1'); | |
if (node.uses?.route) | |
uses.push('"route":1'); | |
if (node.uses?.url) | |
uses.push('"url":1'); | |
return `"uses":{${uses.join(",")}}`; | |
} | |
async function render_endpoint(event, mod, state) { | |
const method = ( | |
/** @type {import('types').HttpMethod} */ | |
event.request.method | |
); | |
let handler = mod[method] || mod.fallback; | |
if (method === "HEAD" && mod.GET && !mod.HEAD) { | |
handler = mod.GET; | |
} | |
if (!handler) { | |
return method_not_allowed(mod, method); | |
} | |
const prerender = mod.prerender ?? state.prerender_default; | |
if (prerender && (mod.POST || mod.PATCH || mod.PUT || mod.DELETE)) { | |
throw new Error("Cannot prerender endpoints that have mutative methods"); | |
} | |
if (state.prerendering && !prerender) { | |
if (state.depth > 0) { | |
throw new Error(`${event.route.id} is not prerenderable`); | |
} else { | |
return new Response(void 0, { status: 204 }); | |
} | |
} | |
try { | |
let response = await handler( | |
/** @type {import('@sveltejs/kit').RequestEvent<Record<string, any>>} */ | |
event | |
); | |
if (!(response instanceof Response)) { | |
throw new Error( | |
`Invalid response from route ${event.url.pathname}: handler should return a Response object` | |
); | |
} | |
if (state.prerendering) { | |
response = new Response(response.body, { | |
status: response.status, | |
statusText: response.statusText, | |
headers: new Headers(response.headers) | |
}); | |
response.headers.set("x-sveltekit-prerender", String(prerender)); | |
} | |
return response; | |
} catch (e) { | |
if (e instanceof Redirect) { | |
return new Response(void 0, { | |
status: e.status, | |
headers: { location: e.location } | |
}); | |
} | |
throw e; | |
} | |
} | |
function is_endpoint_request(event) { | |
const { method, headers: headers2 } = event.request; | |
if (ENDPOINT_METHODS.includes(method) && !PAGE_METHODS.includes(method)) { | |
return true; | |
} | |
if (method === "POST" && headers2.get("x-sveltekit-action") === "true") | |
return false; | |
const accept = event.request.headers.get("accept") ?? "*/*"; | |
return negotiate(accept, ["*", "text/html"]) !== "text/html"; | |
} | |
function compact(arr) { | |
return arr.filter( | |
/** @returns {val is NonNullable<T>} */ | |
(val) => val != null | |
); | |
} | |
function is_action_json_request(event) { | |
const accept = negotiate(event.request.headers.get("accept") ?? "*/*", [ | |
"application/json", | |
"text/html" | |
]); | |
return accept === "application/json" && event.request.method === "POST"; | |
} | |
async function handle_action_json_request(event, options2, server) { | |
const actions = server?.actions; | |
if (!actions) { | |
const no_actions_error = new SvelteKitError( | |
405, | |
"Method Not Allowed", | |
"POST method not allowed. No actions exist for this page" | |
); | |
return action_json( | |
{ | |
type: "error", | |
error: await handle_error_and_jsonify(event, options2, no_actions_error) | |
}, | |
{ | |
status: no_actions_error.status, | |
headers: { | |
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Status/405 | |
// "The server must generate an Allow header field in a 405 status code response" | |
allow: "GET" | |
} | |
} | |
); | |
} | |
check_named_default_separate(actions); | |
try { | |
const data = await call_action(event, actions); | |
if (false) | |
; | |
if (data instanceof ActionFailure) { | |
return action_json({ | |
type: "failure", | |
status: data.status, | |
// @ts-expect-error we assign a string to what is supposed to be an object. That's ok | |
// because we don't use the object outside, and this way we have better code navigation | |
// through knowing where the related interface is used. | |
data: stringify_action_response( | |
data.data, | |
/** @type {string} */ | |
event.route.id | |
) | |
}); | |
} else { | |
return action_json({ | |
type: "success", | |
status: data ? 200 : 204, | |
// @ts-expect-error see comment above | |
data: stringify_action_response( | |
data, | |
/** @type {string} */ | |
event.route.id | |
) | |
}); | |
} | |
} catch (e) { | |
const err = normalize_error(e); | |
if (err instanceof Redirect) { | |
return action_json_redirect(err); | |
} | |
return action_json( | |
{ | |
type: "error", | |
error: await handle_error_and_jsonify(event, options2, check_incorrect_fail_use(err)) | |
}, | |
{ | |
status: get_status(err) | |
} | |
); | |
} | |
} | |
function check_incorrect_fail_use(error) { | |
return error instanceof ActionFailure ? new Error('Cannot "throw fail()". Use "return fail()"') : error; | |
} | |
function action_json_redirect(redirect) { | |
return action_json({ | |
type: "redirect", | |
status: redirect.status, | |
location: redirect.location | |
}); | |
} | |
function action_json(data, init2) { | |
return json(data, init2); | |
} | |
function is_action_request(event) { | |
return event.request.method === "POST"; | |
} | |
async function handle_action_request(event, server) { | |
const actions = server?.actions; | |
if (!actions) { | |
event.setHeaders({ | |
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Status/405 | |
// "The server must generate an Allow header field in a 405 status code response" | |
allow: "GET" | |
}); | |
return { | |
type: "error", | |
error: new SvelteKitError( | |
405, | |
"Method Not Allowed", | |
"POST method not allowed. No actions exist for this page" | |
) | |
}; | |
} | |
check_named_default_separate(actions); | |
try { | |
const data = await call_action(event, actions); | |
if (false) | |
; | |
if (data instanceof ActionFailure) { | |
return { | |
type: "failure", | |
status: data.status, | |
data: data.data | |
}; | |
} else { | |
return { | |
type: "success", | |
status: 200, | |
// @ts-expect-error this will be removed upon serialization, so `undefined` is the same as omission | |
data | |
}; | |
} | |
} catch (e) { | |
const err = normalize_error(e); | |
if (err instanceof Redirect) { | |
return { | |
type: "redirect", | |
status: err.status, | |
location: err.location | |
}; | |
} | |
return { | |
type: "error", | |
error: check_incorrect_fail_use(err) | |
}; | |
} | |
} | |
function check_named_default_separate(actions) { | |
if (actions.default && Object.keys(actions).length > 1) { | |
throw new Error( | |
"When using named actions, the default action cannot be used. See the docs for more info: https://kit.svelte.dev/docs/form-actions#named-actions" | |
); | |
} | |
} | |
async function call_action(event, actions) { | |
const url = new URL(event.request.url); | |
let name = "default"; | |
for (const param of url.searchParams) { | |
if (param[0].startsWith("/")) { | |
name = param[0].slice(1); | |
if (name === "default") { | |
throw new Error('Cannot use reserved action name "default"'); | |
} | |
break; | |
} | |
} | |
const action = actions[name]; | |
if (!action) { | |
throw new SvelteKitError(404, "Not Found", `No action with name '${name}' found`); | |
} | |
if (!is_form_content_type(event.request)) { | |
throw new SvelteKitError( | |
415, | |
"Unsupported Media Type", | |
`Form actions expect form-encoded data — received ${event.request.headers.get( | |
"content-type" | |
)}` | |
); | |
} | |
return action(event); | |
} | |
function uneval_action_response(data, route_id) { | |
return try_deserialize(data, uneval, route_id); | |
} | |
function stringify_action_response(data, route_id) { | |
return try_deserialize(data, stringify, route_id); | |
} | |
function try_deserialize(data, fn, route_id) { | |
try { | |
return fn(data); | |
} catch (e) { | |
const error = ( | |
/** @type {any} */ | |
e | |
); | |
if ("path" in error) { | |
let message = `Data returned from action inside ${route_id} is not serializable: ${error.message}`; | |
if (error.path !== "") | |
message += ` (data.${error.path})`; | |
throw new Error(message); | |
} | |
throw error; | |
} | |
} | |
const INVALIDATED_PARAM = "x-sveltekit-invalidated"; | |
const TRAILING_SLASH_PARAM = "x-sveltekit-trailing-slash"; | |
async function load_server_data({ event, state, node, parent }) { | |
if (!node?.server) | |
return null; | |
let is_tracking = true; | |
const uses = { | |
dependencies: /* @__PURE__ */ new Set(), | |
params: /* @__PURE__ */ new Set(), | |
parent: false, | |
route: false, | |
url: false, | |
search_params: /* @__PURE__ */ new Set() | |
}; | |
const url = make_trackable( | |
event.url, | |
() => { | |
if (is_tracking) { | |
uses.url = true; | |
} | |
}, | |
(param) => { | |
if (is_tracking) { | |
uses.search_params.add(param); | |
} | |
} | |
); | |
if (state.prerendering) { | |
disable_search(url); | |
} | |
const result = await node.server.load?.call(null, { | |
...event, | |
fetch: (info, init2) => { | |
new URL(info instanceof Request ? info.url : info, event.url); | |
return event.fetch(info, init2); | |
}, | |
/** @param {string[]} deps */ | |
depends: (...deps) => { | |
for (const dep of deps) { | |
const { href } = new URL(dep, event.url); | |
uses.dependencies.add(href); | |
} | |
}, | |
params: new Proxy(event.params, { | |
get: (target, key2) => { | |
if (is_tracking) { | |
uses.params.add(key2); | |
} | |
return target[ | |
/** @type {string} */ | |
key2 | |
]; | |
} | |
}), | |
parent: async () => { | |
if (is_tracking) { | |
uses.parent = true; | |
} | |
return parent(); | |
}, | |
route: new Proxy(event.route, { | |
get: (target, key2) => { | |
if (is_tracking) { | |
uses.route = true; | |
} | |
return target[ | |
/** @type {'id'} */ | |
key2 | |
]; | |
} | |
}), | |
url, | |
untrack(fn) { | |
is_tracking = false; | |
try { | |
return fn(); | |
} finally { | |
is_tracking = true; | |
} | |
} | |
}); | |
return { | |
type: "data", | |
data: result ?? null, | |
uses, | |
slash: node.server.trailingSlash | |
}; | |
} | |
async function load_data({ | |
event, | |
fetched, | |
node, | |
parent, | |
server_data_promise, | |
state, | |
resolve_opts, | |
csr | |
}) { | |
const server_data_node = await server_data_promise; | |
if (!node?.universal?.load) { | |
return server_data_node?.data ?? null; | |
} | |
const result = await node.universal.load.call(null, { | |
url: event.url, | |
params: event.params, | |
data: server_data_node?.data ?? null, | |
route: event.route, | |
fetch: create_universal_fetch(event, state, fetched, csr, resolve_opts), | |
setHeaders: event.setHeaders, | |
depends: () => { | |
}, | |
parent, | |
untrack: (fn) => fn() | |
}); | |
return result ?? null; | |
} | |
function b64_encode(buffer) { | |
if (globalThis.Buffer) { | |
return Buffer.from(buffer).toString("base64"); | |
} | |
const little_endian = new Uint8Array(new Uint16Array([1]).buffer)[0] > 0; | |
return btoa( | |
new TextDecoder(little_endian ? "utf-16le" : "utf-16be").decode( | |
new Uint16Array(new Uint8Array(buffer)) | |
) | |
); | |
} | |
function create_universal_fetch(event, state, fetched, csr, resolve_opts) { | |
const universal_fetch = async (input, init2) => { | |
const cloned_body = input instanceof Request && input.body ? input.clone().body : null; | |
const cloned_headers = input instanceof Request && [...input.headers].length ? new Headers(input.headers) : init2?.headers; | |
let response = await event.fetch(input, init2); | |
const url = new URL(input instanceof Request ? input.url : input, event.url); | |
const same_origin = url.origin === event.url.origin; | |
let dependency; | |
if (same_origin) { | |
if (state.prerendering) { | |
dependency = { response, body: null }; | |
state.prerendering.dependencies.set(url.pathname, dependency); | |
} | |
} else { | |
const mode = input instanceof Request ? input.mode : init2?.mode ?? "cors"; | |
if (mode === "no-cors") { | |
response = new Response("", { | |
status: response.status, | |
statusText: response.statusText, | |
headers: response.headers | |
}); | |
} else { | |
const acao = response.headers.get("access-control-allow-origin"); | |
if (!acao || acao !== event.url.origin && acao !== "*") { | |
throw new Error( | |
`CORS error: ${acao ? "Incorrect" : "No"} 'Access-Control-Allow-Origin' header is present on the requested resource` | |
); | |
} | |
} | |
} | |
const proxy = new Proxy(response, { | |
get(response2, key2, _receiver) { | |
async function push_fetched(body2, is_b64) { | |
const status_number = Number(response2.status); | |
if (isNaN(status_number)) { | |
throw new Error( | |
`response.status is not a number. value: "${response2.status}" type: ${typeof response2.status}` | |
); | |
} | |
fetched.push({ | |
url: same_origin ? url.href.slice(event.url.origin.length) : url.href, | |
method: event.request.method, | |
request_body: ( | |
/** @type {string | ArrayBufferView | undefined} */ | |
input instanceof Request && cloned_body ? await stream_to_string(cloned_body) : init2?.body | |
), | |
request_headers: cloned_headers, | |
response_body: body2, | |
response: response2, | |
is_b64 | |
}); | |
} | |
if (key2 === "arrayBuffer") { | |
return async () => { | |
const buffer = await response2.arrayBuffer(); | |
if (dependency) { | |
dependency.body = new Uint8Array(buffer); | |
} | |
if (buffer instanceof ArrayBuffer) { | |
await push_fetched(b64_encode(buffer), true); | |
} | |
return buffer; | |
}; | |
} | |
async function text2() { | |
const body2 = await response2.text(); | |
if (!body2 || typeof body2 === "string") { | |
await push_fetched(body2, false); | |
} | |
if (dependency) { | |
dependency.body = body2; | |
} | |
return body2; | |
} | |
if (key2 === "text") { | |
return text2; | |
} | |
if (key2 === "json") { | |
return async () => { | |
return JSON.parse(await text2()); | |
}; | |
} | |
return Reflect.get(response2, key2, response2); | |
} | |
}); | |
if (csr) { | |
const get = response.headers.get; | |
response.headers.get = (key2) => { | |
const lower = key2.toLowerCase(); | |
const value = get.call(response.headers, lower); | |
if (value && !lower.startsWith("x-sveltekit-")) { | |
const included = resolve_opts.filterSerializedResponseHeaders(lower, value); | |
if (!included) { | |
throw new Error( | |
`Failed to get response header "${lower}" — it must be included by the \`filterSerializedResponseHeaders\` option: https://kit.svelte.dev/docs/hooks#server-hooks-handle (at ${event.route.id})` | |
); | |
} | |
} | |
return value; | |
}; | |
} | |
return proxy; | |
}; | |
return (input, init2) => { | |
const response = universal_fetch(input, init2); | |
response.catch(() => { | |
}); | |
return response; | |
}; | |
} | |
async function stream_to_string(stream) { | |
let result = ""; | |
const reader = stream.getReader(); | |
const decoder = new TextDecoder(); | |
while (true) { | |
const { done, value } = await reader.read(); | |
if (done) { | |
break; | |
} | |
result += decoder.decode(value); | |
} | |
return result; | |
} | |
function hash(...values) { | |
let hash2 = 5381; | |
for (const value of values) { | |
if (typeof value === "string") { | |
let i = value.length; | |
while (i) | |
hash2 = hash2 * 33 ^ value.charCodeAt(--i); | |
} else if (ArrayBuffer.isView(value)) { | |
const buffer = new Uint8Array(value.buffer, value.byteOffset, value.byteLength); | |
let i = buffer.length; | |
while (i) | |
hash2 = hash2 * 33 ^ buffer[--i]; | |
} else { | |
throw new TypeError("value must be a string or TypedArray"); | |
} | |
} | |
return (hash2 >>> 0).toString(36); | |
} | |
const escape_html_attr_dict = { | |
"&": "&", | |
'"': """ | |
}; | |
const escape_html_attr_regex = new RegExp( | |
// special characters | |
`[${Object.keys(escape_html_attr_dict).join("")}]|[\\ud800-\\udbff](?![\\udc00-\\udfff])|[\\ud800-\\udbff][\\udc00-\\udfff]|[\\udc00-\\udfff]`, | |
"g" | |
); | |
function escape_html_attr(str) { | |
const escaped_str = str.replace(escape_html_attr_regex, (match) => { | |
if (match.length === 2) { | |
return match; | |
} | |
return escape_html_attr_dict[match] ?? `&#${match.charCodeAt(0)};`; | |
}); | |
return `"${escaped_str}"`; | |
} | |
const replacements = { | |
"<": "\\u003C", | |
"\u2028": "\\u2028", | |
"\u2029": "\\u2029" | |
}; | |
const pattern = new RegExp(`[${Object.keys(replacements).join("")}]`, "g"); | |
function serialize_data(fetched, filter, prerendering2 = false) { | |
const headers2 = {}; | |
let cache_control = null; | |
let age = null; | |
let varyAny = false; | |
for (const [key2, value] of fetched.response.headers) { | |
if (filter(key2, value)) { | |
headers2[key2] = value; | |
} | |
if (key2 === "cache-control") | |
cache_control = value; | |
else if (key2 === "age") | |
age = value; | |
else if (key2 === "vary" && value.trim() === "*") | |
varyAny = true; | |
} | |
const payload = { | |
status: fetched.response.status, | |
statusText: fetched.response.statusText, | |
headers: headers2, | |
body: fetched.response_body | |
}; | |
const safe_payload = JSON.stringify(payload).replace(pattern, (match) => replacements[match]); | |
const attrs = [ | |
'type="application/json"', | |
"data-sveltekit-fetched", | |
`data-url=${escape_html_attr(fetched.url)}` | |
]; | |
if (fetched.is_b64) { | |
attrs.push("data-b64"); | |
} | |
if (fetched.request_headers || fetched.request_body) { | |
const values = []; | |
if (fetched.request_headers) { | |
values.push([...new Headers(fetched.request_headers)].join(",")); | |
} | |
if (fetched.request_body) { | |
values.push(fetched.request_body); | |
} | |
attrs.push(`data-hash="${hash(...values)}"`); | |
} | |
if (!prerendering2 && fetched.method === "GET" && cache_control && !varyAny) { | |
const match = /s-maxage=(\d+)/g.exec(cache_control) ?? /max-age=(\d+)/g.exec(cache_control); | |
if (match) { | |
const ttl = +match[1] - +(age ?? "0"); | |
attrs.push(`data-ttl="${ttl}"`); | |
} | |
} | |
return `<script ${attrs.join(" ")}>${safe_payload}<\/script>`; | |
} | |
const s = JSON.stringify; | |
const encoder$2 = new TextEncoder(); | |
function sha256(data) { | |
if (!key[0]) | |
precompute(); | |
const out = init.slice(0); | |
const array2 = encode(data); | |
for (let i = 0; i < array2.length; i += 16) { | |
const w = array2.subarray(i, i + 16); | |
let tmp; | |
let a; | |
let b; | |
let out0 = out[0]; | |
let out1 = out[1]; | |
let out2 = out[2]; | |
let out3 = out[3]; | |
let out4 = out[4]; | |
let out5 = out[5]; | |
let out6 = out[6]; | |
let out7 = out[7]; | |
for (let i2 = 0; i2 < 64; i2++) { | |
if (i2 < 16) { | |
tmp = w[i2]; | |
} else { | |
a = w[i2 + 1 & 15]; | |
b = w[i2 + 14 & 15]; | |
tmp = w[i2 & 15] = (a >>> 7 ^ a >>> 18 ^ a >>> 3 ^ a << 25 ^ a << 14) + (b >>> 17 ^ b >>> 19 ^ b >>> 10 ^ b << 15 ^ b << 13) + w[i2 & 15] + w[i2 + 9 & 15] | 0; | |
} | |
tmp = tmp + out7 + (out4 >>> 6 ^ out4 >>> 11 ^ out4 >>> 25 ^ out4 << 26 ^ out4 << 21 ^ out4 << 7) + (out6 ^ out4 & (out5 ^ out6)) + key[i2]; | |
out7 = out6; | |
out6 = out5; | |
out5 = out4; | |
out4 = out3 + tmp | 0; | |
out3 = out2; | |
out2 = out1; | |
out1 = out0; | |
out0 = tmp + (out1 & out2 ^ out3 & (out1 ^ out2)) + (out1 >>> 2 ^ out1 >>> 13 ^ out1 >>> 22 ^ out1 << 30 ^ out1 << 19 ^ out1 << 10) | 0; | |
} | |
out[0] = out[0] + out0 | 0; | |
out[1] = out[1] + out1 | 0; | |
out[2] = out[2] + out2 | 0; | |
out[3] = out[3] + out3 | 0; | |
out[4] = out[4] + out4 | 0; | |
out[5] = out[5] + out5 | 0; | |
out[6] = out[6] + out6 | 0; | |
out[7] = out[7] + out7 | 0; | |
} | |
const bytes = new Uint8Array(out.buffer); | |
reverse_endianness(bytes); | |
return base64(bytes); | |
} | |
const init = new Uint32Array(8); | |
const key = new Uint32Array(64); | |
function precompute() { | |
function frac(x) { | |
return (x - Math.floor(x)) * 4294967296; | |
} | |
let prime = 2; | |
for (let i = 0; i < 64; prime++) { | |
let is_prime = true; | |
for (let factor = 2; factor * factor <= prime; factor++) { | |
if (prime % factor === 0) { | |
is_prime = false; | |
break; | |
} | |
} | |
if (is_prime) { | |
if (i < 8) { | |
init[i] = frac(prime ** (1 / 2)); | |
} | |
key[i] = frac(prime ** (1 / 3)); | |
i++; | |
} | |
} | |
} | |
function reverse_endianness(bytes) { | |
for (let i = 0; i < bytes.length; i += 4) { | |
const a = bytes[i + 0]; | |
const b = bytes[i + 1]; | |
const c = bytes[i + 2]; | |
const d = bytes[i + 3]; | |
bytes[i + 0] = d; | |
bytes[i + 1] = c; | |
bytes[i + 2] = b; | |
bytes[i + 3] = a; | |
} | |
} | |
function encode(str) { | |
const encoded = encoder$2.encode(str); | |
const length = encoded.length * 8; | |
const size = 512 * Math.ceil((length + 65) / 512); | |
const bytes = new Uint8Array(size / 8); | |
bytes.set(encoded); | |
bytes[encoded.length] = 128; | |
reverse_endianness(bytes); | |
const words = new Uint32Array(bytes.buffer); | |
words[words.length - 2] = Math.floor(length / 4294967296); | |
words[words.length - 1] = length; | |
return words; | |
} | |
const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".split(""); | |
function base64(bytes) { | |
const l = bytes.length; | |
let result = ""; | |
let i; | |
for (i = 2; i < l; i += 3) { | |
result += chars[bytes[i - 2] >> 2]; | |
result += chars[(bytes[i - 2] & 3) << 4 | bytes[i - 1] >> 4]; | |
result += chars[(bytes[i - 1] & 15) << 2 | bytes[i] >> 6]; | |
result += chars[bytes[i] & 63]; | |
} | |
if (i === l + 1) { | |
result += chars[bytes[i - 2] >> 2]; | |
result += chars[(bytes[i - 2] & 3) << 4]; | |
result += "=="; | |
} | |
if (i === l) { | |
result += chars[bytes[i - 2] >> 2]; | |
result += chars[(bytes[i - 2] & 3) << 4 | bytes[i - 1] >> 4]; | |
result += chars[(bytes[i - 1] & 15) << 2]; | |
result += "="; | |
} | |
return result; | |
} | |
const array = new Uint8Array(16); | |
function generate_nonce() { | |
crypto.getRandomValues(array); | |
return base64(array); | |
} | |
const quoted = /* @__PURE__ */ new Set([ | |
"self", | |
"unsafe-eval", | |
"unsafe-hashes", | |
"unsafe-inline", | |
"none", | |
"strict-dynamic", | |
"report-sample", | |
"wasm-unsafe-eval", | |
"script" | |
]); | |
const crypto_pattern = /^(nonce|sha\d\d\d)-/; | |
class BaseProvider { | |
/** @type {boolean} */ | |
#use_hashes; | |
/** @type {boolean} */ | |
#script_needs_csp; | |
/** @type {boolean} */ | |
#style_needs_csp; | |
/** @type {import('types').CspDirectives} */ | |
#directives; | |
/** @type {import('types').Csp.Source[]} */ | |
#script_src; | |
/** @type {import('types').Csp.Source[]} */ | |
#script_src_elem; | |
/** @type {import('types').Csp.Source[]} */ | |
#style_src; | |
/** @type {import('types').Csp.Source[]} */ | |
#style_src_attr; | |
/** @type {import('types').Csp.Source[]} */ | |
#style_src_elem; | |
/** @type {string} */ | |
#nonce; | |
/** | |
* @param {boolean} use_hashes | |
* @param {import('types').CspDirectives} directives | |
* @param {string} nonce | |
*/ | |
constructor(use_hashes, directives, nonce) { | |
this.#use_hashes = use_hashes; | |
this.#directives = directives; | |
const d = this.#directives; | |
this.#script_src = []; | |
this.#script_src_elem = []; | |
this.#style_src = []; | |
this.#style_src_attr = []; | |
this.#style_src_elem = []; | |
const effective_script_src = d["script-src"] || d["default-src"]; | |
const script_src_elem = d["script-src-elem"]; | |
const effective_style_src = d["style-src"] || d["default-src"]; | |
const style_src_attr = d["style-src-attr"]; | |
const style_src_elem = d["style-src-elem"]; | |
this.#script_needs_csp = !!effective_script_src && effective_script_src.filter((value) => value !== "unsafe-inline").length > 0 || !!script_src_elem && script_src_elem.filter((value) => value !== "unsafe-inline").length > 0; | |
this.#style_needs_csp = !!effective_style_src && effective_style_src.filter((value) => value !== "unsafe-inline").length > 0 || !!style_src_attr && style_src_attr.filter((value) => value !== "unsafe-inline").length > 0 || !!style_src_elem && style_src_elem.filter((value) => value !== "unsafe-inline").length > 0; | |
this.script_needs_nonce = this.#script_needs_csp && !this.#use_hashes; | |
this.style_needs_nonce = this.#style_needs_csp && !this.#use_hashes; | |
this.#nonce = nonce; | |
} | |
/** @param {string} content */ | |
add_script(content) { | |
if (this.#script_needs_csp) { | |
const d = this.#directives; | |
if (this.#use_hashes) { | |
const hash2 = sha256(content); | |
this.#script_src.push(`sha256-${hash2}`); | |
if (d["script-src-elem"]?.length) { | |
this.#script_src_elem.push(`sha256-${hash2}`); | |
} | |
} else { | |
if (this.#script_src.length === 0) { | |
this.#script_src.push(`nonce-${this.#nonce}`); | |
} | |
if (d["script-src-elem"]?.length) { | |
this.#script_src_elem.push(`nonce-${this.#nonce}`); | |
} | |
} | |
} | |
} | |
/** @param {string} content */ | |
add_style(content) { | |
if (this.#style_needs_csp) { | |
const empty_comment_hash = "9OlNO0DNEeaVzHL4RZwCLsBHA8WBQ8toBp/4F5XV2nc="; | |
const d = this.#directives; | |
if (this.#use_hashes) { | |
const hash2 = sha256(content); | |
this.#style_src.push(`sha256-${hash2}`); | |
if (d["style-src-attr"]?.length) { | |
this.#style_src_attr.push(`sha256-${hash2}`); | |
} | |
if (d["style-src-elem"]?.length) { | |
if (hash2 !== empty_comment_hash && !d["style-src-elem"].includes(`sha256-${empty_comment_hash}`)) { | |
this.#style_src_elem.push(`sha256-${empty_comment_hash}`); | |
} | |
this.#style_src_elem.push(`sha256-${hash2}`); | |
} | |
} else { | |
if (this.#style_src.length === 0 && !d["style-src"]?.includes("unsafe-inline")) { | |
this.#style_src.push(`nonce-${this.#nonce}`); | |
} | |
if (d["style-src-attr"]?.length) { | |
this.#style_src_attr.push(`nonce-${this.#nonce}`); | |
} | |
if (d["style-src-elem"]?.length) { | |
if (!d["style-src-elem"].includes(`sha256-${empty_comment_hash}`)) { | |
this.#style_src_elem.push(`sha256-${empty_comment_hash}`); | |
} | |
this.#style_src_elem.push(`nonce-${this.#nonce}`); | |
} | |
} | |
} | |
} | |
/** | |
* @param {boolean} [is_meta] | |
*/ | |
get_header(is_meta = false) { | |
const header = []; | |
const directives = { ...this.#directives }; | |
if (this.#style_src.length > 0) { | |
directives["style-src"] = [ | |
...directives["style-src"] || directives["default-src"] || [], | |
...this.#style_src | |
]; | |
} | |
if (this.#style_src_attr.length > 0) { | |
directives["style-src-attr"] = [ | |
...directives["style-src-attr"] || [], | |
...this.#style_src_attr | |
]; | |
} | |
if (this.#style_src_elem.length > 0) { | |
directives["style-src-elem"] = [ | |
...directives["style-src-elem"] || [], | |
...this.#style_src_elem | |
]; | |
} | |
if (this.#script_src.length > 0) { | |
directives["script-src"] = [ | |
...directives["script-src"] || directives["default-src"] || [], | |
...this.#script_src | |
]; | |
} | |
if (this.#script_src_elem.length > 0) { | |
directives["script-src-elem"] = [ | |
...directives["script-src-elem"] || [], | |
...this.#script_src_elem | |
]; | |
} | |
for (const key2 in directives) { | |
if (is_meta && (key2 === "frame-ancestors" || key2 === "report-uri" || key2 === "sandbox")) { | |
continue; | |
} | |
const value = ( | |
/** @type {string[] | true} */ | |
directives[key2] | |
); | |
if (!value) | |
continue; | |
const directive = [key2]; | |
if (Array.isArray(value)) { | |
value.forEach((value2) => { | |
if (quoted.has(value2) || crypto_pattern.test(value2)) { | |
directive.push(`'${value2}'`); | |
} else { | |
directive.push(value2); | |
} | |
}); | |
} | |
header.push(directive.join(" ")); | |
} | |
return header.join("; "); | |
} | |
} | |
class CspProvider extends BaseProvider { | |
get_meta() { | |
const content = this.get_header(true); | |
if (!content) { | |
return; | |
} | |
return `<meta http-equiv="content-security-policy" content=${escape_html_attr(content)}>`; | |
} | |
} | |
class CspReportOnlyProvider extends BaseProvider { | |
/** | |
* @param {boolean} use_hashes | |
* @param {import('types').CspDirectives} directives | |
* @param {string} nonce | |
*/ | |
constructor(use_hashes, directives, nonce) { | |
super(use_hashes, directives, nonce); | |
if (Object.values(directives).filter((v) => !!v).length > 0) { | |
const has_report_to = directives["report-to"]?.length ?? 0 > 0; | |
const has_report_uri = directives["report-uri"]?.length ?? 0 > 0; | |
if (!has_report_to && !has_report_uri) { | |
throw Error( | |
"`content-security-policy-report-only` must be specified with either the `report-to` or `report-uri` directives, or both" | |
); | |
} | |
} | |
} | |
} | |
class Csp { | |
/** @readonly */ | |
nonce = generate_nonce(); | |
/** @type {CspProvider} */ | |
csp_provider; | |
/** @type {CspReportOnlyProvider} */ | |
report_only_provider; | |
/** | |
* @param {import('./types.js').CspConfig} config | |
* @param {import('./types.js').CspOpts} opts | |
*/ | |
constructor({ mode, directives, reportOnly }, { prerender }) { | |
const use_hashes = mode === "hash" || mode === "auto" && prerender; | |
this.csp_provider = new CspProvider(use_hashes, directives, this.nonce); | |
this.report_only_provider = new CspReportOnlyProvider(use_hashes, reportOnly, this.nonce); | |
} | |
get script_needs_nonce() { | |
return this.csp_provider.script_needs_nonce || this.report_only_provider.script_needs_nonce; | |
} | |
get style_needs_nonce() { | |
return this.csp_provider.style_needs_nonce || this.report_only_provider.style_needs_nonce; | |
} | |
/** @param {string} content */ | |
add_script(content) { | |
this.csp_provider.add_script(content); | |
this.report_only_provider.add_script(content); | |
} | |
/** @param {string} content */ | |
add_style(content) { | |
this.csp_provider.add_style(content); | |
this.report_only_provider.add_style(content); | |
} | |
} | |
function defer() { | |
let fulfil; | |
let reject; | |
const promise = new Promise((f, r) => { | |
fulfil = f; | |
reject = r; | |
}); | |
return { promise, fulfil, reject }; | |
} | |
function create_async_iterator() { | |
const deferred = [defer()]; | |
return { | |
iterator: { | |
[Symbol.asyncIterator]() { | |
return { | |
next: async () => { | |
const next = await deferred[0].promise; | |
if (!next.done) | |
deferred.shift(); | |
return next; | |
} | |
}; | |
} | |
}, | |
push: (value) => { | |
deferred[deferred.length - 1].fulfil({ | |
value, | |
done: false | |
}); | |
deferred.push(defer()); | |
}, | |
done: () => { | |
deferred[deferred.length - 1].fulfil({ done: true }); | |
} | |
}; | |
} | |
const updated = { | |
...readable(false), | |
check: () => false | |
}; | |
const encoder$1 = new TextEncoder(); | |
async function render_response({ | |
branch, | |
fetched, | |
options: options2, | |
manifest, | |
state, | |
page_config, | |
status, | |
error = null, | |
event, | |
resolve_opts, | |
action_result | |
}) { | |
if (state.prerendering) { | |
if (options2.csp.mode === "nonce") { | |
throw new Error('Cannot use prerendering if config.kit.csp.mode === "nonce"'); | |
} | |
if (options2.app_template_contains_nonce) { | |
throw new Error("Cannot use prerendering if page template contains %sveltekit.nonce%"); | |
} | |
} | |
const { client } = manifest._; | |
const modulepreloads = new Set(client.imports); | |
const stylesheets = new Set(client.stylesheets); | |
const fonts = new Set(client.fonts); | |
const link_header_preloads = /* @__PURE__ */ new Set(); | |
const inline_styles = /* @__PURE__ */ new Map(); | |
let rendered; | |
const form_value = action_result?.type === "success" || action_result?.type === "failure" ? action_result.data ?? null : null; | |
let base$1 = base; | |
let assets$1 = assets; | |
let base_expression = s(base); | |
if (!state.prerendering?.fallback) { | |
const segments = event.url.pathname.slice(base.length).split("/").slice(2); | |
base$1 = segments.map(() => "..").join("/") || "."; | |
base_expression = `new URL(${s(base$1)}, location).pathname.slice(0, -1)`; | |
if (!assets || assets[0] === "/" && assets !== SVELTE_KIT_ASSETS) { | |
assets$1 = base$1; | |
} | |
} | |
if (page_config.ssr) { | |
const props = { | |
stores: { | |
page: writable(null), | |
navigating: writable(null), | |
updated | |
}, | |
constructors: await Promise.all(branch.map(({ node }) => node.component())), | |
form: form_value | |
}; | |
let data2 = {}; | |
for (let i = 0; i < branch.length; i += 1) { | |
data2 = { ...data2, ...branch[i].data }; | |
props[`data_${i}`] = data2; | |
} | |
props.page = { | |
error, | |
params: ( | |
/** @type {Record<string, any>} */ | |
event.params | |
), | |
route: event.route, | |
status, | |
url: event.url, | |
data: data2, | |
form: form_value, | |
state: {} | |
}; | |
override({ base: base$1, assets: assets$1 }); | |
{ | |
try { | |
rendered = options2.root.render(props); | |
} finally { | |
reset(); | |
} | |
} | |
for (const { node } of branch) { | |
for (const url of node.imports) | |
modulepreloads.add(url); | |
for (const url of node.stylesheets) | |
stylesheets.add(url); | |
for (const url of node.fonts) | |
fonts.add(url); | |
if (node.inline_styles) { | |
Object.entries(await node.inline_styles()).forEach(([k, v]) => inline_styles.set(k, v)); | |
} | |
} | |
} else { | |
rendered = { head: "", html: "", css: { code: "", map: null } }; | |
} | |
let head = ""; | |
let body2 = rendered.html; | |
const csp = new Csp(options2.csp, { | |
prerender: !!state.prerendering | |
}); | |
const prefixed = (path) => { | |
if (path.startsWith("/")) { | |
return base + path; | |
} | |
return `${assets$1}/${path}`; | |
}; | |
if (inline_styles.size > 0) { | |
const content = Array.from(inline_styles.values()).join("\n"); | |
const attributes = []; | |
if (csp.style_needs_nonce) | |
attributes.push(` nonce="${csp.nonce}"`); | |
csp.add_style(content); | |
head += ` | |
<style${attributes.join("")}>${content}</style>`; | |
} | |
for (const dep of stylesheets) { | |
const path = prefixed(dep); | |
const attributes = ['rel="stylesheet"']; | |
if (inline_styles.has(dep)) { | |
attributes.push("disabled", 'media="(max-width: 0)"'); | |
} else { | |
if (resolve_opts.preload({ type: "css", path })) { | |
const preload_atts = ['rel="preload"', 'as="style"']; | |
link_header_preloads.add(`<${encodeURI(path)}>; ${preload_atts.join(";")}; nopush`); | |
} | |
} | |
head += ` | |
<link href="${path}" ${attributes.join(" ")}>`; | |
} | |
for (const dep of fonts) { | |
const path = prefixed(dep); | |
if (resolve_opts.preload({ type: "font", path })) { | |
const ext = dep.slice(dep.lastIndexOf(".") + 1); | |
const attributes = [ | |
'rel="preload"', | |
'as="font"', | |
`type="font/${ext}"`, | |
`href="${path}"`, | |
"crossorigin" | |
]; | |
head += ` | |
<link ${attributes.join(" ")}>`; | |
} | |
} | |
const global = `__sveltekit_${options2.version_hash}`; | |
const { data, chunks } = get_data( | |
event, | |
options2, | |
branch.map((b) => b.server_data), | |
global | |
); | |
if (page_config.ssr && page_config.csr) { | |
body2 += ` | |
${fetched.map( | |
(item) => serialize_data(item, resolve_opts.filterSerializedResponseHeaders, !!state.prerendering) | |
).join("\n ")}`; | |
} | |
if (page_config.csr) { | |
if (client.uses_env_dynamic_public && state.prerendering) { | |
modulepreloads.add(`${options2.app_dir}/env.js`); | |
} | |
const included_modulepreloads = Array.from(modulepreloads, (dep) => prefixed(dep)).filter( | |
(path) => resolve_opts.preload({ type: "js", path }) | |
); | |
for (const path of included_modulepreloads) { | |
link_header_preloads.add(`<${encodeURI(path)}>; rel="modulepreload"; nopush`); | |
if (options2.preload_strategy !== "modulepreload") { | |
head += ` | |
<link rel="preload" as="script" crossorigin="anonymous" href="${path}">`; | |
} else if (state.prerendering) { | |
head += ` | |
<link rel="modulepreload" href="${path}">`; | |
} | |
} | |
const blocks = []; | |
const load_env_eagerly = client.uses_env_dynamic_public && state.prerendering; | |
const properties = [`base: ${base_expression}`]; | |
if (assets) { | |
properties.push(`assets: ${s(assets)}`); | |
} | |
if (client.uses_env_dynamic_public) { | |
properties.push(`env: ${load_env_eagerly ? "null" : s(public_env)}`); | |
} | |
if (chunks) { | |
blocks.push("const deferred = new Map();"); | |
properties.push(`defer: (id) => new Promise((fulfil, reject) => { | |
deferred.set(id, { fulfil, reject }); | |
})`); | |
properties.push(`resolve: ({ id, data, error }) => { | |
const { fulfil, reject } = deferred.get(id); | |
deferred.delete(id); | |
if (error) reject(error); | |
else fulfil(data); | |
}`); | |
} | |
blocks.push(`${global} = { | |
${properties.join(",\n ")} | |
};`); | |
const args = ["app", "element"]; | |
blocks.push("const element = document.currentScript.parentElement;"); | |
if (page_config.ssr) { | |
const serialized = { form: "null", error: "null" }; | |
blocks.push(`const data = ${data};`); | |
if (form_value) { | |
serialized.form = uneval_action_response( | |
form_value, | |
/** @type {string} */ | |
event.route.id | |
); | |
} | |
if (error) { | |
serialized.error = uneval(error); | |
} | |
const hydrate = [ | |
`node_ids: [${branch.map(({ node }) => node.index).join(", ")}]`, | |
"data", | |
`form: ${serialized.form}`, | |
`error: ${serialized.error}` | |
]; | |
if (status !== 200) { | |
hydrate.push(`status: ${status}`); | |
} | |
if (options2.embedded) { | |
hydrate.push(`params: ${uneval(event.params)}`, `route: ${s(event.route)}`); | |
} | |
const indent = " ".repeat(load_env_eagerly ? 7 : 6); | |
args.push(`{ | |
${indent} ${hydrate.join(`, | |
${indent} `)} | |
${indent}}`); | |
} | |
if (load_env_eagerly) { | |
blocks.push(`import(${s(`${base$1}/${options2.app_dir}/env.js`)}).then(({ env }) => { | |
${global}.env = env; | |
Promise.all([ | |
import(${s(prefixed(client.start))}), | |
import(${s(prefixed(client.app))}) | |
]).then(([kit, app]) => { | |
kit.start(${args.join(", ")}); | |
}); | |
});`); | |
} else { | |
blocks.push(`Promise.all([ | |
import(${s(prefixed(client.start))}), | |
import(${s(prefixed(client.app))}) | |
]).then(([kit, app]) => { | |
kit.start(${args.join(", ")}); | |
});`); | |
} | |
if (options2.service_worker) { | |
const opts = ""; | |
blocks.push(`if ('serviceWorker' in navigator) { | |
addEventListener('load', function () { | |
navigator.serviceWorker.register('${prefixed("service-worker.js")}'${opts}); | |
}); | |
}`); | |
} | |
const init_app = ` | |
{ | |
${blocks.join("\n\n ")} | |
} | |
`; | |
csp.add_script(init_app); | |
body2 += ` | |
<script${csp.script_needs_nonce ? ` nonce="${csp.nonce}"` : ""}>${init_app}<\/script> | |
`; | |
} | |
const headers2 = new Headers({ | |
"x-sveltekit-page": "true", | |
"content-type": "text/html" | |
}); | |
if (state.prerendering) { | |
const http_equiv = []; | |
const csp_headers = csp.csp_provider.get_meta(); | |
if (csp_headers) { | |
http_equiv.push(csp_headers); | |
} | |
if (state.prerendering.cache) { | |
http_equiv.push(`<meta http-equiv="cache-control" content="${state.prerendering.cache}">`); | |
} | |
if (http_equiv.length > 0) { | |
head = http_equiv.join("\n") + head; | |
} | |
} else { | |
const csp_header = csp.csp_provider.get_header(); | |
if (csp_header) { | |
headers2.set("content-security-policy", csp_header); | |
} | |
const report_only_header = csp.report_only_provider.get_header(); | |
if (report_only_header) { | |
headers2.set("content-security-policy-report-only", report_only_header); | |
} | |
if (link_header_preloads.size) { | |
headers2.set("link", Array.from(link_header_preloads).join(", ")); | |
} | |
} | |
head += rendered.head; | |
const html = options2.templates.app({ | |
head, | |
body: body2, | |
assets: assets$1, | |
nonce: ( | |
/** @type {string} */ | |
csp.nonce | |
), | |
env: safe_public_env | |
}); | |
const transformed = await resolve_opts.transformPageChunk({ | |
html, | |
done: true | |
}) || ""; | |
if (!chunks) { | |
headers2.set("etag", `"${hash(transformed)}"`); | |
} | |
return !chunks ? text(transformed, { | |
status, | |
headers: headers2 | |
}) : new Response( | |
new ReadableStream({ | |
async start(controller) { | |
controller.enqueue(encoder$1.encode(transformed + "\n")); | |
for await (const chunk of chunks) { | |
controller.enqueue(encoder$1.encode(chunk)); | |
} | |
controller.close(); | |
}, | |
type: "bytes" | |
}), | |
{ | |
headers: { | |
"content-type": "text/html" | |
} | |
} | |
); | |
} | |
function get_data(event, options2, nodes, global) { | |
let promise_id = 1; | |
let count = 0; | |
const { iterator, push, done } = create_async_iterator(); | |
function replacer(thing) { | |
if (typeof thing?.then === "function") { | |
const id = promise_id++; | |
count += 1; | |
thing.then( | |
/** @param {any} data */ | |
(data) => ({ data }) | |
).catch( | |
/** @param {any} error */ | |
async (error) => ({ | |
error: await handle_error_and_jsonify(event, options2, error) | |
}) | |
).then( | |
/** | |
* @param {{data: any; error: any}} result | |
*/ | |
async ({ data, error }) => { | |
count -= 1; | |
let str; | |
try { | |
str = uneval({ id, data, error }, replacer); | |
} catch (e) { | |
error = await handle_error_and_jsonify( | |
event, | |
options2, | |
new Error(`Failed to serialize promise while rendering ${event.route.id}`) | |
); | |
data = void 0; | |
str = uneval({ id, data, error }, replacer); | |
} | |
push(`<script>${global}.resolve(${str})<\/script> | |
`); | |
if (count === 0) | |
done(); | |
} | |
); | |
return `${global}.defer(${id})`; | |
} | |
} | |
try { | |
const strings = nodes.map((node) => { | |
if (!node) | |
return "null"; | |
return `{"type":"data","data":${uneval(node.data, replacer)},${stringify_uses(node)}${node.slash ? `,"slash":${JSON.stringify(node.slash)}` : ""}}`; | |
}); | |
return { | |
data: `[${strings.join(",")}]`, | |
chunks: count > 0 ? iterator : null | |
}; | |
} catch (e) { | |
throw new Error(clarify_devalue_error( | |
event, | |
/** @type {any} */ | |
e | |
)); | |
} | |
} | |
function get_option(nodes, option) { | |
return nodes.reduce( | |
(value, node) => { | |
return ( | |
/** @type {Value} TypeScript's too dumb to understand this */ | |
node?.universal?.[option] ?? node?.server?.[option] ?? value | |
); | |
}, | |
/** @type {Value | undefined} */ | |
void 0 | |
); | |
} | |
async function respond_with_error({ | |
event, | |
options: options2, | |
manifest, | |
state, | |
status, | |
error, | |
resolve_opts | |
}) { | |
if (event.request.headers.get("x-sveltekit-error")) { | |
return static_error_page( | |
options2, | |
status, | |
/** @type {Error} */ | |
error.message | |
); | |
} | |
const fetched = []; | |
try { | |
const branch = []; | |
const default_layout = await manifest._.nodes[0](); | |
const ssr = get_option([default_layout], "ssr") ?? true; | |
const csr = get_option([default_layout], "csr") ?? true; | |
if (ssr) { | |
state.error = true; | |
const server_data_promise = load_server_data({ | |
event, | |
state, | |
node: default_layout, | |
parent: async () => ({}) | |
}); | |
const server_data = await server_data_promise; | |
const data = await load_data({ | |
event, | |
fetched, | |
node: default_layout, | |
parent: async () => ({}), | |
resolve_opts, | |
server_data_promise, | |
state, | |
csr | |
}); | |
branch.push( | |
{ | |
node: default_layout, | |
server_data, | |
data | |
}, | |
{ | |
node: await manifest._.nodes[1](), | |
// 1 is always the root error | |
data: null, | |
server_data: null | |
} | |
); | |
} | |
return await render_response({ | |
options: options2, | |
manifest, | |
state, | |
page_config: { | |
ssr, | |
csr | |
}, | |
status, | |
error: await handle_error_and_jsonify(event, options2, error), | |
branch, | |
fetched, | |
event, | |
resolve_opts | |
}); | |
} catch (e) { | |
if (e instanceof Redirect) { | |
return redirect_response(e.status, e.location); | |
} | |
return static_error_page( | |
options2, | |
get_status(e), | |
(await handle_error_and_jsonify(event, options2, e)).message | |
); | |
} | |
} | |
function once(fn) { | |
let done = false; | |
let result; | |
return () => { | |
if (done) | |
return result; | |
done = true; | |
return result = fn(); | |
}; | |
} | |
const encoder = new TextEncoder(); | |
async function render_data(event, route, options2, manifest, state, invalidated_data_nodes, trailing_slash) { | |
if (!route.page) { | |
return new Response(void 0, { | |
status: 404 | |
}); | |
} | |
try { | |
const node_ids = [...route.page.layouts, route.page.leaf]; | |
const invalidated = invalidated_data_nodes ?? node_ids.map(() => true); | |
let aborted = false; | |
const url = new URL(event.url); | |
url.pathname = normalize_path(url.pathname, trailing_slash); | |
const new_event = { ...event, url }; | |
const functions = node_ids.map((n, i) => { | |
return once(async () => { | |
try { | |
if (aborted) { | |
return ( | |
/** @type {import('types').ServerDataSkippedNode} */ | |
{ | |
type: "skip" | |
} | |
); | |
} | |
const node = n == void 0 ? n : await manifest._.nodes[n](); | |
return load_server_data({ | |
event: new_event, | |
state, | |
node, | |
parent: async () => { | |
const data2 = {}; | |
for (let j = 0; j < i; j += 1) { | |
const parent = ( | |
/** @type {import('types').ServerDataNode | null} */ | |
await functions[j]() | |
); | |
if (parent) { | |
Object.assign(data2, parent.data); | |
} | |
} | |
return data2; | |
} | |
}); | |
} catch (e) { | |
aborted = true; | |
throw e; | |
} | |
}); | |
}); | |
const promises = functions.map(async (fn, i) => { | |
if (!invalidated[i]) { | |
return ( | |
/** @type {import('types').ServerDataSkippedNode} */ | |
{ | |
type: "skip" | |
} | |
); | |
} | |
return fn(); | |
}); | |
let length = promises.length; | |
const nodes = await Promise.all( | |
promises.map( | |
(p, i) => p.catch(async (error) => { | |
if (error instanceof Redirect) { | |
throw error; | |
} | |
length = Math.min(length, i + 1); | |
return ( | |
/** @type {import('types').ServerErrorNode} */ | |
{ | |
type: "error", | |
error: await handle_error_and_jsonify(event, options2, error), | |
status: error instanceof HttpError || error instanceof SvelteKitError ? error.status : void 0 | |
} | |
); | |
}) | |
) | |
); | |
const { data, chunks } = get_data_json(event, options2, nodes); | |
if (!chunks) { | |
return json_response(data); | |
} | |
return new Response( | |
new ReadableStream({ | |
async start(controller) { | |
controller.enqueue(encoder.encode(data)); | |
for await (const chunk of chunks) { | |
controller.enqueue(encoder.encode(chunk)); | |
} | |
controller.close(); | |
}, | |
type: "bytes" | |
}), | |
{ | |
headers: { | |
// we use a proprietary content type to prevent buffering. | |
// the `text` prefix makes it inspectable | |
"content-type": "text/sveltekit-data", | |
"cache-control": "private, no-store" | |
} | |
} | |
); | |
} catch (e) { | |
const error = normalize_error(e); | |
if (error instanceof Redirect) { | |
return redirect_json_response(error); | |
} else { | |
return json_response(await handle_error_and_jsonify(event, options2, error), 500); | |
} | |
} | |
} | |
function json_response(json2, status = 200) { | |
return text(typeof json2 === "string" ? json2 : JSON.stringify(json2), { | |
status, | |
headers: { | |
"content-type": "application/json", | |
"cache-control": "private, no-store" | |
} | |
}); | |
} | |
function redirect_json_response(redirect) { | |
return json_response({ | |
type: "redirect", | |
location: redirect.location | |
}); | |
} | |
function get_data_json(event, options2, nodes) { | |
let promise_id = 1; | |
let count = 0; | |
const { iterator, push, done } = create_async_iterator(); | |
const reducers = { | |
/** @param {any} thing */ | |
Promise: (thing) => { | |
if (typeof thing?.then === "function") { | |
const id = promise_id++; | |
count += 1; | |
let key2 = "data"; | |
thing.catch( | |
/** @param {any} e */ | |
async (e) => { | |
key2 = "error"; | |
return handle_error_and_jsonify( | |
event, | |
options2, | |
/** @type {any} */ | |
e | |
); | |
} | |
).then( | |
/** @param {any} value */ | |
async (value) => { | |
let str; | |
try { | |
str = stringify(value, reducers); | |
} catch (e) { | |
const error = await handle_error_and_jsonify( | |
event, | |
options2, | |
new Error(`Failed to serialize promise while rendering ${event.route.id}`) | |
); | |
key2 = "error"; | |
str = stringify(error, reducers); | |
} | |
count -= 1; | |
push(`{"type":"chunk","id":${id},"${key2}":${str}} | |
`); | |
if (count === 0) | |
done(); | |
} | |
); | |
return id; | |
} | |
} | |
}; | |
try { | |
const strings = nodes.map((node) => { | |
if (!node) | |
return "null"; | |
if (node.type === "error" || node.type === "skip") { | |
return JSON.stringify(node); | |
} | |
return `{"type":"data","data":${stringify(node.data, reducers)},${stringify_uses( | |
node | |
)}${node.slash ? `,"slash":${JSON.stringify(node.slash)}` : ""}}`; | |
}); | |
return { | |
data: `{"type":"data","nodes":[${strings.join(",")}]} | |
`, | |
chunks: count > 0 ? iterator : null | |
}; | |
} catch (e) { | |
throw new Error(clarify_devalue_error( | |
event, | |
/** @type {any} */ | |
e | |
)); | |
} | |
} | |
const MAX_DEPTH = 10; | |
async function render_page(event, page, options2, manifest, state, resolve_opts) { | |
if (state.depth > MAX_DEPTH) { | |
return text(`Not found: ${event.url.pathname}`, { | |
status: 404 | |
// TODO in some cases this should be 500. not sure how to differentiate | |
}); | |
} | |
if (is_action_json_request(event)) { | |
const node = await manifest._.nodes[page.leaf](); | |
return handle_action_json_request(event, options2, node?.server); | |
} | |
try { | |
const nodes = await Promise.all([ | |
// we use == here rather than === because [undefined] serializes as "[null]" | |
...page.layouts.map((n) => n == void 0 ? n : manifest._.nodes[n]()), | |
manifest._.nodes[page.leaf]() | |
]); | |
const leaf_node = ( | |
/** @type {import('types').SSRNode} */ | |
nodes.at(-1) | |
); | |
let status = 200; | |
let action_result = void 0; | |
if (is_action_request(event)) { | |
action_result = await handle_action_request(event, leaf_node.server); | |
if (action_result?.type === "redirect") { | |
return redirect_response(action_result.status, action_result.location); | |
} | |
if (action_result?.type === "error") { | |
status = get_status(action_result.error); | |
} | |
if (action_result?.type === "failure") { | |
status = action_result.status; | |
} | |
} | |
const should_prerender_data = nodes.some((node) => node?.server?.load); | |
const data_pathname = add_data_suffix(event.url.pathname); | |
const should_prerender = get_option(nodes, "prerender") ?? false; | |
if (should_prerender) { | |
const mod = leaf_node.server; | |
if (mod?.actions) { | |
throw new Error("Cannot prerender pages with actions"); | |
} | |
} else if (state.prerendering) { | |
return new Response(void 0, { | |
status: 204 | |
}); | |
} | |
state.prerender_default = should_prerender; | |
const fetched = []; | |
if (get_option(nodes, "ssr") === false && !(state.prerendering && should_prerender_data)) { | |
return await render_response({ | |
branch: [], | |
fetched, | |
page_config: { | |
ssr: false, | |
csr: get_option(nodes, "csr") ?? true | |
}, | |
status, | |
error: null, | |
event, | |
options: options2, | |
manifest, | |
state, | |
resolve_opts | |
}); | |
} | |
const branch = []; | |
let load_error = null; | |
const server_promises = nodes.map((node, i) => { | |
if (load_error) { | |
throw load_error; | |
} | |
return Promise.resolve().then(async () => { | |
try { | |
if (node === leaf_node && action_result?.type === "error") { | |
throw action_result.error; | |
} | |
return await load_server_data({ | |
event, | |
state, | |
node, | |
parent: async () => { | |
const data = {}; | |
for (let j = 0; j < i; j += 1) { | |
const parent = await server_promises[j]; | |
if (parent) | |
Object.assign(data, await parent.data); | |
} | |
return data; | |
} | |
}); | |
} catch (e) { | |
load_error = /** @type {Error} */ | |
e; | |
throw load_error; | |
} | |
}); | |
}); | |
const csr = get_option(nodes, "csr") ?? true; | |
const load_promises = nodes.map((node, i) => { | |
if (load_error) | |
throw load_error; | |
return Promise.resolve().then(async () => { | |
try { | |
return await load_data({ | |
event, | |
fetched, | |
node, | |
parent: async () => { | |
const data = {}; | |
for (let j = 0; j < i; j += 1) { | |
Object.assign(data, await load_promises[j]); | |
} | |
return data; | |
}, | |
resolve_opts, | |
server_data_promise: server_promises[i], | |
state, | |
csr | |
}); | |
} catch (e) { | |
load_error = /** @type {Error} */ | |
e; | |
throw load_error; | |
} | |
}); | |
}); | |
for (const p of server_promises) | |
p.catch(() => { | |
}); | |
for (const p of load_promises) | |
p.catch(() => { | |
}); | |
for (let i = 0; i < nodes.length; i += 1) { | |
const node = nodes[i]; | |
if (node) { | |
try { | |
const server_data = await server_promises[i]; | |
const data = await load_promises[i]; | |
branch.push({ node, server_data, data }); | |
} catch (e) { | |
const err = normalize_error(e); | |
if (err instanceof Redirect) { | |
if (state.prerendering && should_prerender_data) { | |
const body2 = JSON.stringify({ | |
type: "redirect", | |
location: err.location | |
}); | |
state.prerendering.dependencies.set(data_pathname, { | |
response: text(body2), | |
body: body2 | |
}); | |
} | |
return redirect_response(err.status, err.location); | |
} | |
const status2 = get_status(err); | |
const error = await handle_error_and_jsonify(event, options2, err); | |
while (i--) { | |
if (page.errors[i]) { | |
const index = ( | |
/** @type {number} */ | |
page.errors[i] | |
); | |
const node2 = await manifest._.nodes[index](); | |
let j = i; | |
while (!branch[j]) | |
j -= 1; | |
return await render_response({ | |
event, | |
options: options2, | |
manifest, | |
state, | |
resolve_opts, | |
page_config: { ssr: true, csr: true }, | |
status: status2, | |
error, | |
branch: compact(branch.slice(0, j + 1)).concat({ | |
node: node2, | |
data: null, | |
server_data: null | |
}), | |
fetched | |
}); | |
} | |
} | |
return static_error_page(options2, status2, error.message); | |
} | |
} else { | |
branch.push(null); | |
} | |
} | |
if (state.prerendering && should_prerender_data) { | |
let { data, chunks } = get_data_json( | |
event, | |
options2, | |
branch.map((node) => node?.server_data) | |
); | |
if (chunks) { | |
for await (const chunk of chunks) { | |
data += chunk; | |
} | |
} | |
state.prerendering.dependencies.set(data_pathname, { | |
response: text(data), | |
body: data | |
}); | |
} | |
const ssr = get_option(nodes, "ssr") ?? true; | |
return await render_response({ | |
event, | |
options: options2, | |
manifest, | |
state, | |
resolve_opts, | |
page_config: { | |
csr: get_option(nodes, "csr") ?? true, | |
ssr | |
}, | |
status, | |
error: null, | |
branch: ssr === false ? [] : compact(branch), | |
action_result, | |
fetched | |
}); | |
} catch (e) { | |
return await respond_with_error({ | |
event, | |
options: options2, | |
manifest, | |
state, | |
status: 500, | |
error: e, | |
resolve_opts | |
}); | |
} | |
} | |
function exec(match, params, matchers) { | |
const result = {}; | |
const values = match.slice(1); | |
const values_needing_match = values.filter((value) => value !== void 0); | |
let buffered = 0; | |
for (let i = 0; i < params.length; i += 1) { | |
const param = params[i]; | |
let value = values[i - buffered]; | |
if (param.chained && param.rest && buffered) { | |
value = values.slice(i - buffered, i + 1).filter((s2) => s2).join("/"); | |
buffered = 0; | |
} | |
if (value === void 0) { | |
if (param.rest) | |
result[param.name] = ""; | |
continue; | |
} | |
if (!param.matcher || matchers[param.matcher](value)) { | |
result[param.name] = value; | |
const next_param = params[i + 1]; | |
const next_value = values[i + 1]; | |
if (next_param && !next_param.rest && next_param.optional && next_value && param.chained) { | |
buffered = 0; | |
} | |
if (!next_param && !next_value && Object.keys(result).length === values_needing_match.length) { | |
buffered = 0; | |
} | |
continue; | |
} | |
if (param.optional && param.chained) { | |
buffered++; | |
continue; | |
} | |
return; | |
} | |
if (buffered) | |
return; | |
return result; | |
} | |
function validate_options(options2) { | |
if (options2?.path === void 0) { | |
throw new Error("You must specify a `path` when setting, deleting or serializing cookies"); | |
} | |
} | |
function get_cookies(request, url, trailing_slash) { | |
const header = request.headers.get("cookie") ?? ""; | |
const initial_cookies = cookieExports.parse(header, { decode: (value) => value }); | |
const normalized_url = normalize_path(url.pathname, trailing_slash); | |
const new_cookies = {}; | |
const defaults = { | |
httpOnly: true, | |
sameSite: "lax", | |
secure: url.hostname === "localhost" && url.protocol === "http:" ? false : true | |
}; | |
const cookies = { | |
// The JSDoc param annotations appearing below for get, set and delete | |
// are necessary to expose the `cookie` library types to | |
// typescript users. `@type {import('@sveltejs/kit').Cookies}` above is not | |
// sufficient to do so. | |
/** | |
* @param {string} name | |
* @param {import('cookie').CookieParseOptions} opts | |
*/ | |
get(name, opts) { | |
const c = new_cookies[name]; | |
if (c && domain_matches(url.hostname, c.options.domain) && path_matches(url.pathname, c.options.path)) { | |
return c.value; | |
} | |
const decoder = opts?.decode || decodeURIComponent; | |
const req_cookies = cookieExports.parse(header, { decode: decoder }); | |
const cookie = req_cookies[name]; | |
return cookie; | |
}, | |
/** | |
* @param {import('cookie').CookieParseOptions} opts | |
*/ | |
getAll(opts) { | |
const decoder = opts?.decode || decodeURIComponent; | |
const cookies2 = cookieExports.parse(header, { decode: decoder }); | |
for (const c of Object.values(new_cookies)) { | |
if (domain_matches(url.hostname, c.options.domain) && path_matches(url.pathname, c.options.path)) { | |
cookies2[c.name] = c.value; | |
} | |
} | |
return Object.entries(cookies2).map(([name, value]) => ({ name, value })); | |
}, | |
/** | |
* @param {string} name | |
* @param {string} value | |
* @param {import('./page/types.js').Cookie['options']} options | |
*/ | |
set(name, value, options2) { | |
validate_options(options2); | |
set_internal(name, value, { ...defaults, ...options2 }); | |
}, | |
/** | |
* @param {string} name | |
* @param {import('./page/types.js').Cookie['options']} options | |
*/ | |
delete(name, options2) { | |
validate_options(options2); | |
cookies.set(name, "", { ...options2, maxAge: 0 }); | |
}, | |
/** | |
* @param {string} name | |
* @param {string} value | |
* @param {import('./page/types.js').Cookie['options']} options | |
*/ | |
serialize(name, value, options2) { | |
validate_options(options2); | |
let path = options2.path; | |
if (!options2.domain || options2.domain === url.hostname) { | |
path = resolve(normalized_url, path); | |
} | |
return cookieExports.serialize(name, value, { ...defaults, ...options2, path }); | |
} | |
}; | |
function get_cookie_header(destination, header2) { | |
const combined_cookies = { | |
// cookies sent by the user agent have lowest precedence | |
...initial_cookies | |
}; | |
for (const key2 in new_cookies) { | |
const cookie = new_cookies[key2]; | |
if (!domain_matches(destination.hostname, cookie.options.domain)) | |
continue; | |
if (!path_matches(destination.pathname, cookie.options.path)) | |
continue; | |
const encoder2 = cookie.options.encode || encodeURIComponent; | |
combined_cookies[cookie.name] = encoder2(cookie.value); | |
} | |
if (header2) { | |
const parsed = cookieExports.parse(header2, { decode: (value) => value }); | |
for (const name in parsed) { | |
combined_cookies[name] = parsed[name]; | |
} | |
} | |
return Object.entries(combined_cookies).map(([name, value]) => `${name}=${value}`).join("; "); | |
} | |
function set_internal(name, value, options2) { | |
let path = options2.path; | |
if (!options2.domain || options2.domain === url.hostname) { | |
path = resolve(normalized_url, path); | |
} | |
new_cookies[name] = { name, value, options: { ...options2, path } }; | |
} | |
return { cookies, new_cookies, get_cookie_header, set_internal }; | |
} | |
function domain_matches(hostname, constraint) { | |
if (!constraint) | |
return true; | |
const normalized = constraint[0] === "." ? constraint.slice(1) : constraint; | |
if (hostname === normalized) | |
return true; | |
return hostname.endsWith("." + normalized); | |
} | |
function path_matches(path, constraint) { | |
if (!constraint) | |
return true; | |
const normalized = constraint.endsWith("/") ? constraint.slice(0, -1) : constraint; | |
if (path === normalized) | |
return true; | |
return path.startsWith(normalized + "/"); | |
} | |
function add_cookies_to_headers(headers2, cookies) { | |
for (const new_cookie of cookies) { | |
const { name, value, options: options2 } = new_cookie; | |
headers2.append("set-cookie", cookieExports.serialize(name, value, options2)); | |
if (options2.path.endsWith(".html")) { | |
const path = add_data_suffix(options2.path); | |
headers2.append("set-cookie", cookieExports.serialize(name, value, { ...options2, path })); | |
} | |
} | |
} | |
function create_fetch({ event, options: options2, manifest, state, get_cookie_header, set_internal }) { | |
const server_fetch = async (info, init2) => { | |
const original_request = normalize_fetch_input(info, init2, event.url); | |
let mode = (info instanceof Request ? info.mode : init2?.mode) ?? "cors"; | |
let credentials = (info instanceof Request ? info.credentials : init2?.credentials) ?? "same-origin"; | |
return options2.hooks.handleFetch({ | |
event, | |
request: original_request, | |
fetch: async (info2, init3) => { | |
const request = normalize_fetch_input(info2, init3, event.url); | |
const url = new URL(request.url); | |
if (!request.headers.has("origin")) { | |
request.headers.set("origin", event.url.origin); | |
} | |
if (info2 !== original_request) { | |
mode = (info2 instanceof Request ? info2.mode : init3?.mode) ?? "cors"; | |
credentials = (info2 instanceof Request ? info2.credentials : init3?.credentials) ?? "same-origin"; | |
} | |
if ((request.method === "GET" || request.method === "HEAD") && (mode === "no-cors" && url.origin !== event.url.origin || url.origin === event.url.origin)) { | |
request.headers.delete("origin"); | |
} | |
if (url.origin !== event.url.origin) { | |
if (`.${url.hostname}`.endsWith(`.${event.url.hostname}`) && credentials !== "omit") { | |
const cookie = get_cookie_header(url, request.headers.get("cookie")); | |
if (cookie) | |
request.headers.set("cookie", cookie); | |
} | |
return fetch(request); | |
} | |
const prefix = assets || base; | |
const decoded = decodeURIComponent(url.pathname); | |
const filename = (decoded.startsWith(prefix) ? decoded.slice(prefix.length) : decoded).slice(1); | |
const filename_html = `${filename}/index.html`; | |
const is_asset = manifest.assets.has(filename); | |
const is_asset_html = manifest.assets.has(filename_html); | |
if (is_asset || is_asset_html) { | |
const file = is_asset ? filename : filename_html; | |
if (state.read) { | |
const type = is_asset ? manifest.mimeTypes[filename.slice(filename.lastIndexOf("."))] : "text/html"; | |
return new Response(state.read(file), { | |
headers: type ? { "content-type": type } : {} | |
}); | |
} | |
return await fetch(request); | |
} | |
if (credentials !== "omit") { | |
const cookie = get_cookie_header(url, request.headers.get("cookie")); | |
if (cookie) { | |
request.headers.set("cookie", cookie); | |
} | |
const authorization = event.request.headers.get("authorization"); | |
if (authorization && !request.headers.has("authorization")) { | |
request.headers.set("authorization", authorization); | |
} | |
} | |
if (!request.headers.has("accept")) { | |
request.headers.set("accept", "*/*"); | |
} | |
if (!request.headers.has("accept-language")) { | |
request.headers.set( | |
"accept-language", | |
/** @type {string} */ | |
event.request.headers.get("accept-language") | |
); | |
} | |
const response = await respond(request, options2, manifest, { | |
...state, | |
depth: state.depth + 1 | |
}); | |
const set_cookie = response.headers.get("set-cookie"); | |
if (set_cookie) { | |
for (const str of setCookieExports.splitCookiesString(set_cookie)) { | |
const { name, value, ...options3 } = setCookieExports.parseString(str); | |
const path = options3.path ?? (url.pathname.split("/").slice(0, -1).join("/") || "/"); | |
set_internal(name, value, { | |
path, | |
.../** @type {import('cookie').CookieSerializeOptions} */ | |
options3 | |
}); | |
} | |
} | |
return response; | |
} | |
}); | |
}; | |
return (input, init2) => { | |
const response = server_fetch(input, init2); | |
response.catch(() => { | |
}); | |
return response; | |
}; | |
} | |
function normalize_fetch_input(info, init2, url) { | |
if (info instanceof Request) { | |
return info; | |
} | |
return new Request(typeof info === "string" ? new URL(info, url) : info, init2); | |
} | |
let body; | |
let etag; | |
let headers; | |
function get_public_env(request) { | |
body ??= `export const env=${JSON.stringify(public_env)}`; | |
etag ??= `W/${Date.now()}`; | |
headers ??= new Headers({ | |
"content-type": "application/javascript; charset=utf-8", | |
etag | |
}); | |
if (request.headers.get("if-none-match") === etag) { | |
return new Response(void 0, { status: 304, headers }); | |
} | |
return new Response(body, { headers }); | |
} | |
const default_transform = ({ html }) => html; | |
const default_filter = () => false; | |
const default_preload = ({ type }) => type === "js" || type === "css"; | |
const page_methods = /* @__PURE__ */ new Set(["GET", "HEAD", "POST"]); | |
const allowed_page_methods = /* @__PURE__ */ new Set(["GET", "HEAD", "OPTIONS"]); | |
async function respond(request, options2, manifest, state) { | |
const url = new URL(request.url); | |
if (options2.csrf_check_origin) { | |
const forbidden = is_form_content_type(request) && (request.method === "POST" || request.method === "PUT" || request.method === "PATCH" || request.method === "DELETE") && request.headers.get("origin") !== url.origin; | |
if (forbidden) { | |
const csrf_error = new HttpError( | |
403, | |
`Cross-site ${request.method} form submissions are forbidden` | |
); | |
if (request.headers.get("accept") === "application/json") { | |
return json(csrf_error.body, { status: csrf_error.status }); | |
} | |
return text(csrf_error.body.message, { status: csrf_error.status }); | |
} | |
} | |
let rerouted_path; | |
try { | |
rerouted_path = options2.hooks.reroute({ url: new URL(url) }) ?? url.pathname; | |
} catch (e) { | |
return text("Internal Server Error", { | |
status: 500 | |
}); | |
} | |
let decoded; | |
try { | |
decoded = decode_pathname(rerouted_path); | |
} catch { | |
return text("Malformed URI", { status: 400 }); | |
} | |
let route = null; | |
let params = {}; | |
if (base && !state.prerendering?.fallback) { | |
if (!decoded.startsWith(base)) { | |
return text("Not found", { status: 404 }); | |
} | |
decoded = decoded.slice(base.length) || "/"; | |
} | |
if (decoded === `/${options2.app_dir}/env.js`) { | |
return get_public_env(request); | |
} | |
if (decoded.startsWith(`/${options2.app_dir}`)) { | |
return text("Not found", { status: 404 }); | |
} | |
const is_data_request = has_data_suffix(decoded); | |
let invalidated_data_nodes; | |
if (is_data_request) { | |
decoded = strip_data_suffix(decoded) || "/"; | |
url.pathname = strip_data_suffix(url.pathname) + (url.searchParams.get(TRAILING_SLASH_PARAM) === "1" ? "/" : "") || "/"; | |
url.searchParams.delete(TRAILING_SLASH_PARAM); | |
invalidated_data_nodes = url.searchParams.get(INVALIDATED_PARAM)?.split("").map((node) => node === "1"); | |
url.searchParams.delete(INVALIDATED_PARAM); | |
} | |
if (!state.prerendering?.fallback) { | |
const matchers = await manifest._.matchers(); | |
for (const candidate of manifest._.routes) { | |
const match = candidate.pattern.exec(decoded); | |
if (!match) | |
continue; | |
const matched = exec(match, candidate.params, matchers); | |
if (matched) { | |
route = candidate; | |
params = decode_params(matched); | |
break; | |
} | |
} | |
} | |
let trailing_slash = void 0; | |
const headers2 = {}; | |
let cookies_to_add = {}; | |
const event = { | |
// @ts-expect-error `cookies` and `fetch` need to be created after the `event` itself | |
cookies: null, | |
// @ts-expect-error | |
fetch: null, | |
getClientAddress: state.getClientAddress || (() => { | |
throw new Error( | |
`${"@sveltejs/adapter-node"} does not specify getClientAddress. Please raise an issue` | |
); | |
}), | |
locals: {}, | |
params, | |
platform: state.platform, | |
request, | |
route: { id: route?.id ?? null }, | |
setHeaders: (new_headers) => { | |
for (const key2 in new_headers) { | |
const lower = key2.toLowerCase(); | |
const value = new_headers[key2]; | |
if (lower === "set-cookie") { | |
throw new Error( | |
"Use `event.cookies.set(name, value, options)` instead of `event.setHeaders` to set cookies" | |
); | |
} else if (lower in headers2) { | |
throw new Error(`"${key2}" header is already set`); | |
} else { | |
headers2[lower] = value; | |
if (state.prerendering && lower === "cache-control") { | |
state.prerendering.cache = /** @type {string} */ | |
value; | |
} | |
} | |
} | |
}, | |
url, | |
isDataRequest: is_data_request, | |
isSubRequest: state.depth > 0 | |
}; | |
let resolve_opts = { | |
transformPageChunk: default_transform, | |
filterSerializedResponseHeaders: default_filter, | |
preload: default_preload | |
}; | |
try { | |
if (route) { | |
if (url.pathname === base || url.pathname === base + "/") { | |
trailing_slash = "always"; | |
} else if (route.page) { | |
const nodes = await Promise.all([ | |
// we use == here rather than === because [undefined] serializes as "[null]" | |
...route.page.layouts.map((n) => n == void 0 ? n : manifest._.nodes[n]()), | |
manifest._.nodes[route.page.leaf]() | |
]); | |
if (DEV) | |
; | |
trailing_slash = get_option(nodes, "trailingSlash"); | |
} else if (route.endpoint) { | |
const node = await route.endpoint(); | |
trailing_slash = node.trailingSlash; | |
if (DEV) | |
; | |
} | |
if (!is_data_request) { | |
const normalized = normalize_path(url.pathname, trailing_slash ?? "never"); | |
if (normalized !== url.pathname && !state.prerendering?.fallback) { | |
return new Response(void 0, { | |
status: 308, | |
headers: { | |
"x-sveltekit-normalize": "1", | |
location: ( | |
// ensure paths starting with '//' are not treated as protocol-relative | |
(normalized.startsWith("//") ? url.origin + normalized : normalized) + (url.search === "?" ? "" : url.search) | |
) | |
} | |
}); | |
} | |
} | |
} | |
const { cookies, new_cookies, get_cookie_header, set_internal } = get_cookies( | |
request, | |
url, | |
trailing_slash ?? "never" | |
); | |
cookies_to_add = new_cookies; | |
event.cookies = cookies; | |
event.fetch = create_fetch({ | |
event, | |
options: options2, | |
manifest, | |
state, | |
get_cookie_header, | |
set_internal | |
}); | |
if (state.prerendering && !state.prerendering.fallback) | |
disable_search(url); | |
const response = await options2.hooks.handle({ | |
event, | |
resolve: (event2, opts) => resolve2(event2, opts).then((response2) => { | |
for (const key2 in headers2) { | |
const value = headers2[key2]; | |
response2.headers.set( | |
key2, | |
/** @type {string} */ | |
value | |
); | |
} | |
add_cookies_to_headers(response2.headers, Object.values(cookies_to_add)); | |
if (state.prerendering && event2.route.id !== null) { | |
response2.headers.set("x-sveltekit-routeid", encodeURI(event2.route.id)); | |
} | |
return response2; | |
}) | |
}); | |
if (response.status === 200 && response.headers.has("etag")) { | |
let if_none_match_value = request.headers.get("if-none-match"); | |
if (if_none_match_value?.startsWith('W/"')) { | |
if_none_match_value = if_none_match_value.substring(2); | |
} | |
const etag2 = ( | |
/** @type {string} */ | |
response.headers.get("etag") | |
); | |
if (if_none_match_value === etag2) { | |
const headers22 = new Headers({ etag: etag2 }); | |
for (const key2 of [ | |
"cache-control", | |
"content-location", | |
"date", | |
"expires", | |
"vary", | |
"set-cookie" | |
]) { | |
const value = response.headers.get(key2); | |
if (value) | |
headers22.set(key2, value); | |
} | |
return new Response(void 0, { | |
status: 304, | |
headers: headers22 | |
}); | |
} | |
} | |
if (is_data_request && response.status >= 300 && response.status <= 308) { | |
const location = response.headers.get("location"); | |
if (location) { | |
return redirect_json_response(new Redirect( | |
/** @type {any} */ | |
response.status, | |
location | |
)); | |
} | |
} | |
return response; | |
} catch (e) { | |
if (e instanceof Redirect) { | |
const response = is_data_request ? redirect_json_response(e) : route?.page && is_action_json_request(event) ? action_json_redirect(e) : redirect_response(e.status, e.location); | |
add_cookies_to_headers(response.headers, Object.values(cookies_to_add)); | |
return response; | |
} | |
return await handle_fatal_error(event, options2, e); | |
} | |
async function resolve2(event2, opts) { | |
try { | |
if (opts) { | |
resolve_opts = { | |
transformPageChunk: opts.transformPageChunk || default_transform, | |
filterSerializedResponseHeaders: opts.filterSerializedResponseHeaders || default_filter, | |
preload: opts.preload || default_preload | |
}; | |
} | |
if (state.prerendering?.fallback) { | |
return await render_response({ | |
event: event2, | |
options: options2, | |
manifest, | |
state, | |
page_config: { ssr: false, csr: true }, | |
status: 200, | |
error: null, | |
branch: [], | |
fetched: [], | |
resolve_opts | |
}); | |
} | |
if (route) { | |
const method = ( | |
/** @type {import('types').HttpMethod} */ | |
event2.request.method | |
); | |
let response; | |
if (is_data_request) { | |
response = await render_data( | |
event2, | |
route, | |
options2, | |
manifest, | |
state, | |
invalidated_data_nodes, | |
trailing_slash ?? "never" | |
); | |
} else if (route.endpoint && (!route.page || is_endpoint_request(event2))) { | |
response = await render_endpoint(event2, await route.endpoint(), state); | |
} else if (route.page) { | |
if (page_methods.has(method)) { | |
response = await render_page(event2, route.page, options2, manifest, state, resolve_opts); | |
} else { | |
const allowed_methods2 = new Set(allowed_page_methods); | |
const node = await manifest._.nodes[route.page.leaf](); | |
if (node?.server?.actions) { | |
allowed_methods2.add("POST"); | |
} | |
if (method === "OPTIONS") { | |
response = new Response(null, { | |
status: 204, | |
headers: { | |
allow: Array.from(allowed_methods2.values()).join(", ") | |
} | |
}); | |
} else { | |
const mod = [...allowed_methods2].reduce( | |
(acc, curr) => { | |
acc[curr] = true; | |
return acc; | |
}, | |
/** @type {Record<string, any>} */ | |
{} | |
); | |
response = method_not_allowed(mod, method); | |
} | |
} | |
} else { | |
throw new Error("This should never happen"); | |
} | |
if (request.method === "GET" && route.page && route.endpoint) { | |
const vary = response.headers.get("vary")?.split(",")?.map((v) => v.trim().toLowerCase()); | |
if (!(vary?.includes("accept") || vary?.includes("*"))) { | |
response = new Response(response.body, { | |
status: response.status, | |
statusText: response.statusText, | |
headers: new Headers(response.headers) | |
}); | |
response.headers.append("Vary", "Accept"); | |
} | |
} | |
return response; | |
} | |
if (state.error && event2.isSubRequest) { | |
return await fetch(request, { | |
headers: { | |
"x-sveltekit-error": "true" | |
} | |
}); | |
} | |
if (state.error) { | |
return text("Internal Server Error", { | |
status: 500 | |
}); | |
} | |
if (state.depth === 0) { | |
return await respond_with_error({ | |
event: event2, | |
options: options2, | |
manifest, | |
state, | |
status: 404, | |
error: new SvelteKitError(404, "Not Found", `Not found: ${event2.url.pathname}`), | |
resolve_opts | |
}); | |
} | |
if (state.prerendering) { | |
return text("not found", { status: 404 }); | |
} | |
return await fetch(request); | |
} catch (e) { | |
return await handle_fatal_error(event2, options2, e); | |
} finally { | |
event2.cookies.set = () => { | |
throw new Error("Cannot use `cookies.set(...)` after the response has been generated"); | |
}; | |
event2.setHeaders = () => { | |
throw new Error("Cannot use `setHeaders(...)` after the response has been generated"); | |
}; | |
} | |
} | |
} | |
function filter_private_env(env, { public_prefix, private_prefix }) { | |
return Object.fromEntries( | |
Object.entries(env).filter( | |
([k]) => k.startsWith(private_prefix) && (public_prefix === "" || !k.startsWith(public_prefix)) | |
) | |
); | |
} | |
function filter_public_env(env, { public_prefix, private_prefix }) { | |
return Object.fromEntries( | |
Object.entries(env).filter( | |
([k]) => k.startsWith(public_prefix) && (private_prefix === "" || !k.startsWith(private_prefix)) | |
) | |
); | |
} | |
class Server { | |
/** @type {import('types').SSROptions} */ | |
#options; | |
/** @type {import('@sveltejs/kit').SSRManifest} */ | |
#manifest; | |
/** @param {import('@sveltejs/kit').SSRManifest} manifest */ | |
constructor(manifest) { | |
this.#options = options; | |
this.#manifest = manifest; | |
} | |
/** | |
* @param {{ | |
* env: Record<string, string> | |
* }} opts | |
*/ | |
async init({ env }) { | |
const prefixes = { | |
public_prefix: this.#options.env_public_prefix, | |
private_prefix: this.#options.env_private_prefix | |
}; | |
const private_env = filter_private_env(env, prefixes); | |
const public_env2 = filter_public_env(env, prefixes); | |
set_private_env( | |
private_env | |
); | |
set_public_env( | |
public_env2 | |
); | |
set_safe_public_env(public_env2); | |
if (!this.#options.hooks) { | |
try { | |
const module = await get_hooks(); | |
this.#options.hooks = { | |
handle: module.handle || (({ event, resolve: resolve2 }) => resolve2(event)), | |
handleError: module.handleError || (({ error }) => console.error(error)), | |
handleFetch: module.handleFetch || (({ request, fetch: fetch2 }) => fetch2(request)), | |
reroute: module.reroute || (() => { | |
}) | |
}; | |
} catch (error) { | |
{ | |
throw error; | |
} | |
} | |
} | |
} | |
/** | |
* @param {Request} request | |
* @param {import('types').RequestOptions} options | |
*/ | |
async respond(request, options2) { | |
return respond(request, this.#options, this.#manifest, { | |
...options2, | |
error: false, | |
depth: 0 | |
}); | |
} | |
} | |
export { Server }; | |
//# sourceMappingURL=index.js.map | |