Hooks
hooks是整个应用级别的函数, 让你对框架行为有细粒度的控制。
有两个用来放hooks的文件,都是可选的:
src/hooks.server.js
— your app's server hookssrc/hooks.client.js
— your app's client hooks
当应用启动时,这俩文件的代码会运行,他们在初始化数据库clients等方面很有用。
你可以使用
config.kit.files.hooks
配置这些文件的位置
Server hooks
下面这些hooks可以添加到 src/hooks.server.js
:
handle
每当SvelteKit服务端收到一个请求 ,这个函数都会运行,然后决定 response. 它收到两个参数,一个是event
对象,该对象包裹了请求信息。 另一个是resolve
,他可以渲染路由和生成响应。 这个hook让你修改响应头或响应体,or bypass SvelteKit entirely (for implementing routes programmatically, for example).
/// file: src/hooks.server.js
/** @type {import('@sveltejs/kit').Handle} */
export async function handle({ event, resolve }) {
if (event.url.pathname.startsWith('/custom')) {
return new Response('custom response');
}
const response = await resolve(event);
return response;
}
请求静态资源 — 包括已经prerendered的页面 — 是不会被SvelteKit处理的。
如果未实现该hook, 该hook默认等于 ({ event, resolve }) => resolve(event)
。 想要给请求添加自定义数据,使用event.locals
对象。
/// file: src/hooks.server.js
// @filename: ambient.d.ts
type User = {
name: string;
}
declare namespace App {
interface Locals {
user: User;
}
}
const getUserInformation: (cookie: string | void) => Promise<User>;
// @filename: index.js
// ---cut---
/** @type {import('@sveltejs/kit').Handle} */
export async function handle({ event, resolve }) {
event.locals.user = await getUserInformation(event.cookies.get('sessionid'));
const response = await resolve(event);
response.headers.set('x-custom-header', 'potato');
return response;
}
你可以通过使用 the sequence
helper function 添加多个 handle
函数。
resolve
有两个参数,第二个参数是可选的。第二个参数让你对 怎么返回响应 有更多控制。 第二个参数是个对象,有俩字段:
transformPageChunk(opts: { html: string, done: boolean }): MaybePromise<string | undefined>
— applies custom transforms to HTML. Ifdone
is true, it's the final chunk. Chunks are not guaranteed to be well-formed HTML (they could include an element's opening tag but not its closing tag, for example) but they will always be split at sensible boundaries such as%sveltekit.head%
or layout/page components.filterSerializedResponseHeaders(name: string, value: string): boolean
— 决定响应里要包含哪些headers。 默认什么都没有。
/// file: src/hooks.server.js
/** @type {import('@sveltejs/kit').Handle} */
export async function handle({ event, resolve }) {
const response = await resolve(event, {
transformPageChunk: ({ html }) => html.replace('old', 'new'),
filterSerializedResponseHeaders: (name) => name.startsWith('x-')
});
return response;
}
handleFetch
你可以用这个函数修改或替换load
函数里的fetch
请求。
比如你在某个页面导航时,可能请求是使用的公共URL,比如https://api.yourapp.com
。 在SSR时,把该请求转化成API可能很有意义。(能绕过一堆代理 且 load balancers sit between it and the public internet)
/** @type {import('@sveltejs/kit').HandleFetch} */
export async function handleFetch({ request, fetch }) {
if (request.url.startsWith('https://api.yourapp.com/')) {
// clone the original request, but change the URL
request = new Request(
request.url.replace('https://api.yourapp.com/', 'http://localhost:9999/'),
request
);
}
return fetch(request);
}
Credentials
For same-origin requests, SvelteKit's fetch
implementation will forward cookie
and authorization
headers unless the credentials
option is set to "omit"
.
For cross-origin requests, cookie
will be included if the request URL belongs to a subdomain of the app — for example if your app is on my-domain.com
, and your API is on api.my-domain.com
, cookies will be included in the request.
If your app and your API are on sibling subdomains — www.my-domain.com
and api.my-domain.com
for example — then a cookie belonging to a common parent domain like my-domain.com
will not be included, because SvelteKit has no way to know which domain the cookie belongs to. In these cases you will need to manually include the cookie using handleFetch
:
// @errors: 2345
/** @type {import('@sveltejs/kit').HandleFetch} */
export async function handleFetch({ event, request, fetch }) {
if (request.url.startsWith('https://api.my-domain.com/')) {
request.headers.set('cookie', event.request.headers.get('cookie'));
}
return fetch(request);
}
Shared hooks
指的是既可以添加到src/hooks.server.js
又可以添加到src/hooks.client.js
的hooks。
handleError
如果在loading或rendering期间有未预期的错误抛出,就会调用该函数,该函数的参数,有俩字段:error
和 event
。
你可以用该函数做下面两件事:
- 你可以打印error
- 你可以构造一个自定义的error对象, 包含一些细节信息和错误堆栈信息。返回值就是
$page.error
的值,假如是404默认是{ message: 'Not Found' }
。
下面代码展示了一个例子,错误对象被构造为{ message: string; code: string }
,从handleError
函数返回。
/// file: src/app.d.ts
declare namespace App {
interface Error {
message: string;
code: string;
}
}
/// file: src/hooks.server.js
// @errors: 2322 2571
// @filename: ambient.d.ts
const Sentry: any;
// @filename: index.js
// ---cut---
/** @type {import('@sveltejs/kit').HandleServerError} */
export function handleError({ error, event }) {
// example integration with https://sentry.io/
Sentry.captureException(error, { event });
return {
message: 'Whoops!',
code: error.code ?? 'UNKNOWN'
};
}
/// file: src/hooks.client.js
// @errors: 2322 2571
// @filename: ambient.d.ts
const Sentry: any;
// @filename: index.js
// ---cut---
/** @type {import('@sveltejs/kit').HandleClientError} */
export function handleError({ error, event }) {
// example integration with https://sentry.io/
Sentry.captureException(error, { event });
return {
message: 'Whoops!',
code: error.code ?? 'UNKNOWN'
};
}
如果是
src/hooks.client.js
,handleError
的类型是HandleClientError
而非HandleServerError
, 且event
是NavigationEvent
类型 而非RequestEvent
类型。
请注意,如果是可预取的错误,该hook是不会调用的。 (可预期的错误指的是从@sveltejs/kit
包的error
函数抛出的错误)
在开发模式下,如果是因为你的svelte代码有语法错误抛出的错误,error对象会有一个frame
属性,显示错误发生的位置。