@sveltejs/kit
Version:
SvelteKit is the fastest way to build Svelte apps
322 lines (270 loc) • 8.26 kB
JavaScript
/** @import { Transport } from '@sveltejs/kit' */
import * as devalue from 'devalue';
import { base64_decode, base64_encode, text_encoder } from './utils.js';
import * as svelte from 'svelte';
/**
* @param {string} route_id
* @param {string} dep
*/
export function validate_depends(route_id, dep) {
const match = /^(moz-icon|view-source|jar):/.exec(dep);
if (match) {
console.warn(
`${route_id}: Calling \`depends('${dep}')\` will throw an error in Firefox because \`${match[1]}\` is a special URI scheme`
);
}
}
export const INVALIDATED_PARAM = 'x-sveltekit-invalidated';
export const TRAILING_SLASH_PARAM = 'x-sveltekit-trailing-slash';
/**
* @param {any} data
* @param {string} [location_description]
*/
export function validate_load_response(data, location_description) {
if (data != null && Object.getPrototypeOf(data) !== Object.prototype) {
throw new Error(
`a load function ${location_description} returned ${
typeof data !== 'object'
? `a ${typeof data}`
: data instanceof Response
? 'a Response object'
: Array.isArray(data)
? 'an array'
: 'a non-plain object'
}, but must return a plain object at the top level (i.e. \`return {...}\`)`
);
}
}
/**
* Try to `devalue.stringify` the data object using the provided transport encoders.
* @param {any} data
* @param {Transport} transport
*/
export function stringify(data, transport) {
const encoders = Object.fromEntries(Object.entries(transport).map(([k, v]) => [k, v.encode]));
return devalue.stringify(data, encoders);
}
const object_proto_names = /* @__PURE__ */ Object.getOwnPropertyNames(Object.prototype)
.sort()
.join('\0');
/**
* @param {unknown} thing
* @returns {thing is Record<PropertyKey, unknown>}
*/
function is_plain_object(thing) {
if (typeof thing !== 'object' || thing === null) return false;
const proto = Object.getPrototypeOf(thing);
return (
proto === Object.prototype ||
proto === null ||
Object.getPrototypeOf(proto) === null ||
Object.getOwnPropertyNames(proto).sort().join('\0') === object_proto_names
);
}
/**
* @param {Record<string, any>} value
* @param {Map<object, any>} clones
*/
function to_sorted(value, clones) {
const clone = Object.getPrototypeOf(value) === null ? Object.create(null) : {};
clones.set(value, clone);
Object.defineProperty(clone, remote_arg_marker, { value: true });
for (const key of Object.keys(value).sort()) {
const property = value[key];
Object.defineProperty(clone, key, {
value: clones.get(property) ?? property,
enumerable: true,
configurable: true,
writable: true
});
}
return clone;
}
// "sveltekit remote arg"
const remote_object = '__skrao';
const remote_map = '__skram';
const remote_set = '__skras';
const remote_regex_guard = '__skrag';
const remote_arg_marker = Symbol(remote_object);
/**
* @param {Transport} transport
* @param {boolean} sort
* @param {Map<any, any>} remote_arg_clones
*/
function create_remote_arg_reducers(transport, sort, remote_arg_clones) {
/** @type {Record<string, (value: unknown) => unknown>} */
const remote_fns_reducers = {
[remote_regex_guard]:
/** @type {(value: unknown) => void} */
(value) => {
if (value instanceof RegExp) {
throw new Error('Regular expressions are not valid remote function arguments');
}
}
};
if (sort) {
/** @type {(value: unknown) => Array<[unknown, unknown]> | undefined} */
remote_fns_reducers[remote_map] = (value) => {
if (!(value instanceof Map)) {
return;
}
/** @type {Array<[string, string]>} */
const entries = [];
for (const [key, val] of value) {
entries.push([stringify(key), stringify(val)]);
}
return entries.sort(([a1, a2], [b1, b2]) => {
if (a1 < b1) return -1;
if (a1 > b1) return 1;
if (a2 < b2) return -1;
if (a2 > b2) return 1;
return 0;
});
};
/** @type {(value: unknown) => unknown[] | undefined} */
remote_fns_reducers[remote_set] = (value) => {
if (!(value instanceof Set)) {
return;
}
/** @type {string[]} */
const items = [];
for (const item of value) {
items.push(stringify(item));
}
items.sort();
return items;
};
/** @type {(value: unknown) => Record<PropertyKey, unknown> | undefined} */
remote_fns_reducers[remote_object] = (value) => {
if (!is_plain_object(value)) {
return;
}
if (Object.hasOwn(value, remote_arg_marker)) {
return;
}
if (remote_arg_clones.has(value)) {
return remote_arg_clones.get(value);
}
return to_sorted(value, remote_arg_clones);
};
}
const user_reducers = Object.fromEntries(
Object.entries(transport).map(([k, v]) => [k, v.encode])
);
const all_reducers = { ...user_reducers, ...remote_fns_reducers };
/** @type {(value: unknown) => string} */
const stringify = (value) => devalue.stringify(value, all_reducers);
return all_reducers;
}
/** @param {Transport} transport */
function create_remote_arg_revivers(transport) {
const remote_fns_revivers = {
/** @type {(value: unknown) => unknown} */
[remote_object]: (value) => value,
/** @type {(value: unknown) => Map<unknown, unknown>} */
[remote_map]: (value) => {
if (!Array.isArray(value)) {
throw new Error('Invalid data for Map reviver');
}
const map = new Map();
for (const item of value) {
if (
!Array.isArray(item) ||
item.length !== 2 ||
typeof item[0] !== 'string' ||
typeof item[1] !== 'string'
) {
throw new Error('Invalid data for Map reviver');
}
const [key, val] = item;
map.set(parse(key), parse(val));
}
return map;
},
/** @type {(value: unknown) => Set<unknown>} */
[remote_set]: (value) => {
if (!Array.isArray(value)) {
throw new Error('Invalid data for Set reviver');
}
const set = new Set();
for (const item of value) {
if (typeof item !== 'string') {
throw new Error('Invalid data for Set reviver');
}
set.add(parse(item));
}
return set;
}
};
const user_revivers = Object.fromEntries(
Object.entries(transport).map(([k, v]) => [k, v.decode])
);
const all_revivers = { ...user_revivers, ...remote_fns_revivers };
/** @type {(data: string) => unknown} */
const parse = (data) => devalue.parse(data, all_revivers);
return all_revivers;
}
/**
* Stringifies the argument (if any) for a remote function in such a way that
* it is both a valid URL and a valid file name (necessary for prerendering).
* @param {any} value
* @param {Transport} transport
* @param {boolean} [sort]
*/
export function stringify_remote_arg(value, transport, sort = true) {
if (value === undefined) return '';
// If people hit file/url size limits, we can look into using something like compress_and_encode_text from svelte.dev beyond a certain size
const json_string = devalue.stringify(
value,
create_remote_arg_reducers(transport, sort, new Map())
);
const bytes = text_encoder.encode(json_string);
return base64_encode(bytes).replaceAll('=', '').replaceAll('+', '-').replaceAll('/', '_');
}
/**
* Parses the argument (if any) for a remote function
* @param {string} string
* @param {Transport} transport
*/
export function parse_remote_arg(string, transport) {
if (!string) return undefined;
const json_string = new TextDecoder().decode(
// no need to add back `=` characters, atob can handle it
base64_decode(string.replaceAll('-', '+').replaceAll('_', '/'))
);
return devalue.parse(json_string, create_remote_arg_revivers(transport));
}
/**
* @param {string} id
* @param {string} payload
*/
export function create_remote_key(id, payload) {
return id + '/' + payload;
}
/**
* @param {string} key
* @returns {{ id: string; payload: string }}
*/
export function split_remote_key(key) {
const i = key.lastIndexOf('/');
if (i === -1) {
throw new Error(`Invalid remote key: ${key}`);
}
return {
id: key.slice(0, i),
payload: key.slice(i + 1)
};
}
/**
* @template T
* @param {string} key
* @param {() => T} fn
* @returns {T}
* @deprecated TODO remove in SvelteKit 3.0
*/
export function unfriendly_hydratable(key, fn) {
if (!svelte.hydratable) {
throw new Error('Remote functions require Svelte 5.44.0 or later');
}
return svelte.hydratable(key, fn);
}