|
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 { n as noop, s as safe_not_equal, c as create_ssr_component, a as setContext, v as validate_component, m as missing_component } from './chunks/ssr-jUwReIMa.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'; |
|
|
|
|
|
const escaped = { |
|
'<': '\\u003C', |
|
'\\': '\\\\', |
|
'\b': '\\b', |
|
'\f': '\\f', |
|
'\n': '\\n', |
|
'\r': '\\r', |
|
'\t': '\\t', |
|
'\u2028': '\\u2028', |
|
'\u2029': '\\u2029' |
|
}; |
|
|
|
class DevalueError extends Error { |
|
|
|
|
|
|
|
|
|
constructor(message, keys) { |
|
super(message); |
|
this.name = 'DevalueError'; |
|
this.path = keys.join(''); |
|
} |
|
} |
|
|
|
|
|
function is_primitive(thing) { |
|
return Object(thing) !== thing; |
|
} |
|
|
|
const object_proto_names = Object.getOwnPropertyNames( |
|
Object.prototype |
|
) |
|
.sort() |
|
.join('\0'); |
|
|
|
|
|
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 |
|
); |
|
} |
|
|
|
|
|
function get_type(thing) { |
|
return Object.prototype.toString.call(thing).slice(8, -1); |
|
} |
|
|
|
|
|
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')}` |
|
: ''; |
|
} |
|
} |
|
|
|
|
|
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)$/; |
|
|
|
|
|
|
|
|
|
|
|
|
|
function uneval(value, replacer) { |
|
const counts = new Map(); |
|
|
|
|
|
const keys = []; |
|
|
|
const custom = new Map(); |
|
|
|
|
|
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': |
|
(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)); |
|
}); |
|
|
|
|
|
|
|
|
|
|
|
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 = (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) { |
|
|
|
const params = []; |
|
|
|
|
|
const statements = []; |
|
|
|
|
|
const values = []; |
|
|
|
names.forEach((name, thing) => { |
|
params.push(name); |
|
|
|
if (custom.has(thing)) { |
|
values.push( (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})`); |
|
(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; |
|
} |
|
} |
|
|
|
|
|
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; |
|
} |
|
|
|
|
|
function escape_unsafe_char(c) { |
|
return escaped[c] || c; |
|
} |
|
|
|
|
|
function escape_unsafe_chars(str) { |
|
return str.replace(unsafe_chars, escape_unsafe_char); |
|
} |
|
|
|
|
|
function safe_key(key) { |
|
return /^[_$a-zA-Z][_$a-zA-Z0-9]*$/.test(key) |
|
? key |
|
: escape_unsafe_chars(JSON.stringify(key)); |
|
} |
|
|
|
|
|
function safe_prop(key) { |
|
return /^[_$a-zA-Z][_$a-zA-Z0-9]*$/.test(key) |
|
? `.${key}` |
|
: `[${escape_unsafe_chars(JSON.stringify(key))}]`; |
|
} |
|
|
|
|
|
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; |
|
|
|
|
|
|
|
|
|
|
|
|
|
function stringify(value, reducers) { |
|
|
|
const stringified = []; |
|
|
|
|
|
const indexes = new Map(); |
|
|
|
|
|
const custom = []; |
|
for (const key in reducers) { |
|
custom.push({ key, fn: reducers[key] }); |
|
} |
|
|
|
|
|
const keys = []; |
|
|
|
let p = 0; |
|
|
|
|
|
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); |
|
|
|
|
|
if (index < 0) return `${index}`; |
|
|
|
return `[${stringified.join(',')}]`; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
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); |
|
} |
|
|
|
const subscriber_queue = []; |
|
function readable(value, start) { |
|
return { |
|
subscribe: writable(value, start).subscribe |
|
}; |
|
} |
|
function writable(value, start = noop) { |
|
let stop; |
|
const subscribers = new Set(); |
|
function set(new_value) { |
|
if (safe_not_equal(value, new_value)) { |
|
value = new_value; |
|
if (stop) { |
|
const run_queue = !subscriber_queue.length; |
|
for (const subscriber of subscribers) { |
|
subscriber[1](); |
|
subscriber_queue.push(subscriber, value); |
|
} |
|
if (run_queue) { |
|
for (let i = 0; i < subscriber_queue.length; i += 2) { |
|
subscriber_queue[i][0](subscriber_queue[i + 1]); |
|
} |
|
subscriber_queue.length = 0; |
|
} |
|
} |
|
} |
|
} |
|
function update(fn) { |
|
set(fn(value)); |
|
} |
|
function subscribe(run, invalidate = noop) { |
|
const subscriber = [run, invalidate]; |
|
subscribers.add(subscriber); |
|
if (subscribers.size === 1) { |
|
stop = start(set, update) || noop; |
|
} |
|
run(value); |
|
return () => { |
|
subscribers.delete(subscriber); |
|
if (subscribers.size === 0 && stop) { |
|
stop(); |
|
stop = null; |
|
} |
|
}; |
|
} |
|
return { set, update, subscribe }; |
|
} |
|
|
|
var cookie = {}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var hasRequiredCookie; |
|
|
|
function requireCookie () { |
|
if (hasRequiredCookie) return cookie; |
|
hasRequiredCookie = 1; |
|
|
|
|
|
|
|
|
|
|
|
|
|
cookie.parse = parse; |
|
cookie.serialize = serialize; |
|
|
|
|
|
|
|
|
|
|
|
|
|
var __toString = Object.prototype.toString; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var fieldContentRegExp = /^[\u0009\u0020-\u007e\u0080-\u00ff]+$/; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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); |
|
|
|
|
|
if (eqIdx === -1) { |
|
break |
|
} |
|
|
|
var endIdx = str.indexOf(';', index); |
|
|
|
if (endIdx === -1) { |
|
endIdx = str.length; |
|
} else if (endIdx < eqIdx) { |
|
|
|
index = str.lastIndexOf(';', eqIdx - 1) + 1; |
|
continue |
|
} |
|
|
|
var key = str.slice(index, eqIdx).trim(); |
|
|
|
|
|
if (undefined === obj[key]) { |
|
var val = str.slice(eqIdx + 1, endIdx).trim(); |
|
|
|
|
|
if (val.charCodeAt(0) === 0x22) { |
|
val = val.slice(1, -1); |
|
} |
|
|
|
obj[key] = tryDecode(val, dec); |
|
} |
|
|
|
index = endIdx + 1; |
|
} |
|
|
|
return obj; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function decode (str) { |
|
return str.indexOf('%') !== -1 |
|
? decodeURIComponent(str) |
|
: str |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function encode (val) { |
|
return encodeURIComponent(val) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function isDate (val) { |
|
return __toString.call(val) === '[object Date]' || |
|
val instanceof Date |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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; |
|
} 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) { |
|
|
|
|
|
var name = ""; |
|
var value = ""; |
|
var nameValueArr = nameValuePairStr.split("="); |
|
if (nameValueArr.length > 1) { |
|
name = nameValueArr.shift(); |
|
value = nameValueArr.join("="); |
|
} 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") { |
|
|
|
|
|
input = input.headers.getSetCookie(); |
|
} else if (input.headers["set-cookie"]) { |
|
|
|
input = input.headers["set-cookie"]; |
|
} else { |
|
|
|
var sch = |
|
input.headers[ |
|
Object.keys(input.headers).find(function (key) { |
|
return key.toLowerCase() === "set-cookie"; |
|
}) |
|
]; |
|
|
|
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); |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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 === ",") { |
|
|
|
lastComma = pos; |
|
pos += 1; |
|
|
|
skipWhitespace(); |
|
nextStart = pos; |
|
|
|
while (pos < cookiesString.length && notSpecialChar()) { |
|
pos += 1; |
|
} |
|
|
|
|
|
if (pos < cookiesString.length && cookiesString.charAt(pos) === "=") { |
|
|
|
cookiesSeparatorFound = true; |
|
|
|
pos = nextStart; |
|
cookiesStrings.push(cookiesString.substring(start, lastComma)); |
|
start = pos; |
|
} else { |
|
|
|
|
|
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, |
|
|
|
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">\r\n <div style="width: 50px; height: 50px;background-color: red;position:absolute; opacity: 0.6; top:10px; left: 10px"></div>\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: "qoeoop" |
|
}; |
|
async function get_hooks() { |
|
return {}; |
|
} |
|
|
|
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 && |
|
err.name && |
|
err.message ? ( |
|
|
|
err |
|
) : new Error(JSON.stringify(err)); |
|
} |
|
function normalize_error(error) { |
|
return ( |
|
|
|
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: { |
|
|
|
|
|
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 = ( |
|
|
|
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( |
|
|
|
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( |
|
|
|
(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: { |
|
|
|
|
|
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, |
|
|
|
|
|
|
|
data: stringify_action_response( |
|
data.data, |
|
|
|
event.route.id |
|
) |
|
}); |
|
} else { |
|
return action_json({ |
|
type: "success", |
|
status: data ? 200 : 204, |
|
|
|
data: stringify_action_response( |
|
data, |
|
|
|
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({ |
|
|
|
|
|
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, |
|
|
|
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 = ( |
|
|
|
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: new Set(), |
|
params: new Set(), |
|
parent: false, |
|
route: false, |
|
url: false, |
|
search_params: 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); |
|
}, |
|
|
|
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[ |
|
|
|
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[ |
|
|
|
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: ( |
|
|
|
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( |
|
|
|
`[${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 = 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 { |
|
|
|
#use_hashes; |
|
|
|
#script_needs_csp; |
|
|
|
#style_needs_csp; |
|
|
|
#directives; |
|
|
|
#script_src; |
|
|
|
#script_src_elem; |
|
|
|
#style_src; |
|
|
|
#style_src_attr; |
|
|
|
#style_src_elem; |
|
|
|
#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; |
|
} |
|
|
|
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}`); |
|
} |
|
} |
|
} |
|
} |
|
|
|
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}`); |
|
} |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
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 = ( |
|
|
|
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 { |
|
|
|
|
|
|
|
|
|
|
|
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 { |
|
|
|
nonce = generate_nonce(); |
|
|
|
csp_provider; |
|
|
|
report_only_provider; |
|
|
|
|
|
|
|
|
|
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; |
|
} |
|
|
|
add_script(content) { |
|
this.csp_provider.add_script(content); |
|
this.report_only_provider.add_script(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 = new Set(); |
|
const inline_styles = 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: ( |
|
|
|
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, |
|
|
|
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: ( |
|
|
|
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( |
|
|
|
(data) => ({ data }) |
|
).catch( |
|
|
|
async (error) => ({ |
|
error: await handle_error_and_jsonify(event, options2, error) |
|
}) |
|
).then( |
|
|
|
|
|
|
|
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, |
|
|
|
e |
|
)); |
|
} |
|
} |
|
function get_option(nodes, option) { |
|
return nodes.reduce( |
|
(value, node) => { |
|
return ( |
|
|
|
node?.universal?.[option] ?? node?.server?.[option] ?? value |
|
); |
|
}, |
|
|
|
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, |
|
|
|
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](), |
|
|
|
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: "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 = ( |
|
|
|
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: "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: "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: { |
|
|
|
|
|
"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 = { |
|
|
|
Promise: (thing) => { |
|
if (typeof thing?.then === "function") { |
|
const id = promise_id++; |
|
count += 1; |
|
let key2 = "data"; |
|
thing.catch( |
|
|
|
async (e) => { |
|
key2 = "error"; |
|
return handle_error_and_jsonify( |
|
event, |
|
options2, |
|
|
|
e |
|
); |
|
} |
|
).then( |
|
|
|
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, |
|
|
|
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 |
|
|
|
}); |
|
} |
|
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([ |
|
|
|
...page.layouts.map((n) => n == void 0 ? n : manifest._.nodes[n]()), |
|
manifest._.nodes[page.leaf]() |
|
]); |
|
const leaf_node = ( |
|
|
|
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 = |
|
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 = |
|
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 = ( |
|
|
|
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 = { |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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; |
|
}, |
|
|
|
|
|
|
|
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 })); |
|
}, |
|
|
|
|
|
|
|
|
|
|
|
set(name, value, options2) { |
|
validate_options(options2); |
|
set_internal(name, value, { ...defaults, ...options2 }); |
|
}, |
|
|
|
|
|
|
|
|
|
delete(name, options2) { |
|
validate_options(options2); |
|
cookies.set(name, "", { ...options2, maxAge: 0 }); |
|
}, |
|
|
|
|
|
|
|
|
|
|
|
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 = { |
|
|
|
...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", |
|
|
|
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, |
|
... |
|
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 = new Set(["GET", "HEAD", "POST"]); |
|
const allowed_page_methods = 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 = { |
|
|
|
cookies: null, |
|
|
|
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 = |
|
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([ |
|
|
|
...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: ( |
|
|
|
(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, |
|
|
|
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 = ( |
|
|
|
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( |
|
|
|
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 = ( |
|
|
|
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; |
|
}, |
|
|
|
{} |
|
); |
|
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 { |
|
|
|
#options; |
|
|
|
#manifest; |
|
|
|
constructor(manifest) { |
|
this.#options = options; |
|
this.#manifest = manifest; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
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; |
|
} |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
async respond(request, options2) { |
|
return respond(request, this.#options, this.#manifest, { |
|
...options2, |
|
error: false, |
|
depth: 0 |
|
}); |
|
} |
|
} |
|
|
|
export { Server }; |
|
|
|
|