@sigil-dev/grimoire 0.4.0 → 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/index.ts +29 -23
- package/package.json +1 -1
- package/public/__grimoire__/client.js +5 -36
- package/public/__grimoire__/hydrate.js +7 -45
- package/src/build.ts +88 -0
- package/src/cookie-utils.ts +66 -66
- package/src/plugins.ts +19 -5
- package/src/renderer.ts +12 -1
- package/src/server.ts +81 -81
- package/src/ssrPlugin.ts +7 -2
- package/src/transform-routes.ts +6 -1
- package/src/typegen.ts +77 -3
- package/src/types.ts +45 -1
- package/test/fail.test.ts +46 -46
- package/test/headers.test.ts +96 -96
- package/test/middleware.test.ts +217 -217
- package/test/redirect-error.test.ts +112 -112
- package/test/rendering.test.ts +310 -310
- package/test/server.test.ts +120 -0
- package/test/streaming.test.ts +132 -132
- package/test/typegen.test.ts +6 -6
- package/.grimoire/_routes.dom.js +0 -4
- package/.grimoire/_routes.hydrate.js +0 -4
- package/.grimoire/compiled/0-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503001657_protected__page.dom.js +0 -9
- package/.grimoire/compiled/0-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503001657_protected__page.hydrate.js +0 -11
- package/.grimoire/compiled/0-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503161592_protected__page.dom.js +0 -4
- package/.grimoire/compiled/0-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503161592_protected__page.hydrate.js +0 -4
- package/.grimoire/compiled/0-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503395872_login.dom.js +0 -4
- package/.grimoire/compiled/0-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503395872_login.hydrate.js +0 -4
- package/.grimoire/compiled/0-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503614350_login.dom.js +0 -4
- package/.grimoire/compiled/0-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503614350_login.hydrate.js +0 -4
- package/.grimoire/compiled/1-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503001657_login__page.dom.js +0 -8
- package/.grimoire/compiled/1-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503001657_login__page.hydrate.js +0 -9
- package/.grimoire/compiled/1-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503161592_login__page.dom.js +0 -4
- package/.grimoire/compiled/1-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503161592_login__page.hydrate.js +0 -4
- package/.grimoire/compiled/1-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503395872_protected.dom.js +0 -4
- package/.grimoire/compiled/1-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503395872_protected.hydrate.js +0 -4
- package/.grimoire/compiled/1-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503614350_protected.dom.js +0 -4
- package/.grimoire/compiled/1-C__Users_Cane1712_AppData_Local_Temp_grimoire-actions-1780503614350_protected.hydrate.js +0 -4
- package/.grimoire/tsconfig.generated.json +0 -11
- package/.grimoire/types/ambient.d.ts +0 -6
- package/.grimoire/types/api/hello/$types.d.ts +0 -29
- package/.grimoire/types/api/items/$types.d.ts +0 -29
package/test/headers.test.ts
CHANGED
|
@@ -1,96 +1,96 @@
|
|
|
1
|
-
import { describe, expect, test } from "bun:test";
|
|
2
|
-
import { securityHeaders } from "../src/headers";
|
|
3
|
-
|
|
4
|
-
function fakeRequest(path = "/") {
|
|
5
|
-
return new Request(`http://localhost${path}`);
|
|
6
|
-
}
|
|
7
|
-
|
|
8
|
-
async function runPlugin(
|
|
9
|
-
config: Parameters<typeof securityHeaders>[0],
|
|
10
|
-
req?: Request,
|
|
11
|
-
): Promise<Headers> {
|
|
12
|
-
const plugin = securityHeaders(config);
|
|
13
|
-
const res = new Response("<html></html>", {
|
|
14
|
-
headers: { "Content-Type": "text/html" },
|
|
15
|
-
});
|
|
16
|
-
const result = await plugin.onRequest!(req ?? fakeRequest(), async () => res);
|
|
17
|
-
return result.headers;
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
describe("securityHeaders()", () => {
|
|
21
|
-
test("applies default headers", async () => {
|
|
22
|
-
const headers = await runPlugin({});
|
|
23
|
-
expect(headers.get("X-Content-Type-Options")).toBe("nosniff");
|
|
24
|
-
expect(headers.get("X-Frame-Options")).toBe("DENY");
|
|
25
|
-
expect(headers.get("Referrer-Policy")).toBe("strict-origin-when-cross-origin");
|
|
26
|
-
expect(headers.get("Permissions-Policy")).toBe(
|
|
27
|
-
"camera=(), microphone=(), geolocation=()",
|
|
28
|
-
);
|
|
29
|
-
expect(headers.get("X-XSS-Protection")).toBe("0");
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
test("applies CSP by default", async () => {
|
|
33
|
-
const headers = await runPlugin({});
|
|
34
|
-
expect(headers.get("Content-Security-Policy")).toContain("default-src 'self'");
|
|
35
|
-
});
|
|
36
|
-
|
|
37
|
-
test("allows overriding individual headers", async () => {
|
|
38
|
-
const headers = await runPlugin({
|
|
39
|
-
frameOptions: "SAMEORIGIN",
|
|
40
|
-
contentTypeOptions: false,
|
|
41
|
-
});
|
|
42
|
-
expect(headers.get("X-Frame-Options")).toBe("SAMEORIGIN");
|
|
43
|
-
expect(headers.has("X-Content-Type-Options")).toBe(false);
|
|
44
|
-
});
|
|
45
|
-
|
|
46
|
-
test("disables headers when set to false", async () => {
|
|
47
|
-
const headers = await runPlugin({
|
|
48
|
-
contentSecurityPolicy: false,
|
|
49
|
-
strictTransportSecurity: false,
|
|
50
|
-
permissionsPolicy: false,
|
|
51
|
-
});
|
|
52
|
-
expect(headers.has("Content-Security-Policy")).toBe(false);
|
|
53
|
-
expect(headers.has("Strict-Transport-Security")).toBe(false);
|
|
54
|
-
expect(headers.has("Permissions-Policy")).toBe(false);
|
|
55
|
-
});
|
|
56
|
-
|
|
57
|
-
test("applies route overrides", async () => {
|
|
58
|
-
const headers = await runPlugin(
|
|
59
|
-
{
|
|
60
|
-
frameOptions: "DENY",
|
|
61
|
-
routes: {
|
|
62
|
-
"/admin": { frameOptions: "SAMEORIGIN" },
|
|
63
|
-
},
|
|
64
|
-
},
|
|
65
|
-
fakeRequest("/admin/settings"),
|
|
66
|
-
);
|
|
67
|
-
expect(headers.get("X-Frame-Options")).toBe("SAMEORIGIN");
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
test("route override only applies to matching prefix", async () => {
|
|
71
|
-
const headers = await runPlugin(
|
|
72
|
-
{
|
|
73
|
-
frameOptions: "DENY",
|
|
74
|
-
routes: {
|
|
75
|
-
"/admin": { frameOptions: "SAMEORIGIN" },
|
|
76
|
-
},
|
|
77
|
-
},
|
|
78
|
-
fakeRequest("/dashboard"),
|
|
79
|
-
);
|
|
80
|
-
expect(headers.get("X-Frame-Options")).toBe("DENY");
|
|
81
|
-
});
|
|
82
|
-
|
|
83
|
-
test("preserves existing response headers", async () => {
|
|
84
|
-
const plugin = securityHeaders({});
|
|
85
|
-
const res = new Response("<html></html>", {
|
|
86
|
-
headers: {
|
|
87
|
-
"Content-Type": "text/html",
|
|
88
|
-
"Set-Cookie": "session=abc",
|
|
89
|
-
},
|
|
90
|
-
});
|
|
91
|
-
const result = await plugin.onRequest!(fakeRequest(), async () => res);
|
|
92
|
-
expect(result.headers.get("Content-Type")).toBe("text/html");
|
|
93
|
-
expect(result.headers.get("Set-Cookie")).toBe("session=abc");
|
|
94
|
-
expect(result.headers.get("X-Frame-Options")).toBe("DENY");
|
|
95
|
-
});
|
|
96
|
-
});
|
|
1
|
+
import { describe, expect, test } from "bun:test";
|
|
2
|
+
import { securityHeaders } from "../src/headers";
|
|
3
|
+
|
|
4
|
+
function fakeRequest(path = "/") {
|
|
5
|
+
return new Request(`http://localhost${path}`);
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
async function runPlugin(
|
|
9
|
+
config: Parameters<typeof securityHeaders>[0],
|
|
10
|
+
req?: Request,
|
|
11
|
+
): Promise<Headers> {
|
|
12
|
+
const plugin = securityHeaders(config);
|
|
13
|
+
const res = new Response("<html></html>", {
|
|
14
|
+
headers: { "Content-Type": "text/html" },
|
|
15
|
+
});
|
|
16
|
+
const result = await plugin.onRequest!(req ?? fakeRequest(), async () => res);
|
|
17
|
+
return result.headers;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
describe("securityHeaders()", () => {
|
|
21
|
+
test("applies default headers", async () => {
|
|
22
|
+
const headers = await runPlugin({});
|
|
23
|
+
expect(headers.get("X-Content-Type-Options")).toBe("nosniff");
|
|
24
|
+
expect(headers.get("X-Frame-Options")).toBe("DENY");
|
|
25
|
+
expect(headers.get("Referrer-Policy")).toBe("strict-origin-when-cross-origin");
|
|
26
|
+
expect(headers.get("Permissions-Policy")).toBe(
|
|
27
|
+
"camera=(), microphone=(), geolocation=()",
|
|
28
|
+
);
|
|
29
|
+
expect(headers.get("X-XSS-Protection")).toBe("0");
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
test("applies CSP by default", async () => {
|
|
33
|
+
const headers = await runPlugin({});
|
|
34
|
+
expect(headers.get("Content-Security-Policy")).toContain("default-src 'self'");
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
test("allows overriding individual headers", async () => {
|
|
38
|
+
const headers = await runPlugin({
|
|
39
|
+
frameOptions: "SAMEORIGIN",
|
|
40
|
+
contentTypeOptions: false,
|
|
41
|
+
});
|
|
42
|
+
expect(headers.get("X-Frame-Options")).toBe("SAMEORIGIN");
|
|
43
|
+
expect(headers.has("X-Content-Type-Options")).toBe(false);
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
test("disables headers when set to false", async () => {
|
|
47
|
+
const headers = await runPlugin({
|
|
48
|
+
contentSecurityPolicy: false,
|
|
49
|
+
strictTransportSecurity: false,
|
|
50
|
+
permissionsPolicy: false,
|
|
51
|
+
});
|
|
52
|
+
expect(headers.has("Content-Security-Policy")).toBe(false);
|
|
53
|
+
expect(headers.has("Strict-Transport-Security")).toBe(false);
|
|
54
|
+
expect(headers.has("Permissions-Policy")).toBe(false);
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
test("applies route overrides", async () => {
|
|
58
|
+
const headers = await runPlugin(
|
|
59
|
+
{
|
|
60
|
+
frameOptions: "DENY",
|
|
61
|
+
routes: {
|
|
62
|
+
"/admin": { frameOptions: "SAMEORIGIN" },
|
|
63
|
+
},
|
|
64
|
+
},
|
|
65
|
+
fakeRequest("/admin/settings"),
|
|
66
|
+
);
|
|
67
|
+
expect(headers.get("X-Frame-Options")).toBe("SAMEORIGIN");
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
test("route override only applies to matching prefix", async () => {
|
|
71
|
+
const headers = await runPlugin(
|
|
72
|
+
{
|
|
73
|
+
frameOptions: "DENY",
|
|
74
|
+
routes: {
|
|
75
|
+
"/admin": { frameOptions: "SAMEORIGIN" },
|
|
76
|
+
},
|
|
77
|
+
},
|
|
78
|
+
fakeRequest("/dashboard"),
|
|
79
|
+
);
|
|
80
|
+
expect(headers.get("X-Frame-Options")).toBe("DENY");
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
test("preserves existing response headers", async () => {
|
|
84
|
+
const plugin = securityHeaders({});
|
|
85
|
+
const res = new Response("<html></html>", {
|
|
86
|
+
headers: {
|
|
87
|
+
"Content-Type": "text/html",
|
|
88
|
+
"Set-Cookie": "session=abc",
|
|
89
|
+
},
|
|
90
|
+
});
|
|
91
|
+
const result = await plugin.onRequest!(fakeRequest(), async () => res);
|
|
92
|
+
expect(result.headers.get("Content-Type")).toBe("text/html");
|
|
93
|
+
expect(result.headers.get("Set-Cookie")).toBe("session=abc");
|
|
94
|
+
expect(result.headers.get("X-Frame-Options")).toBe("DENY");
|
|
95
|
+
});
|
|
96
|
+
});
|
package/test/middleware.test.ts
CHANGED
|
@@ -1,217 +1,217 @@
|
|
|
1
|
-
import { describe, expect, test } from "bun:test";
|
|
2
|
-
import { sequence, createHooks } from "../src/hooks";
|
|
3
|
-
import { createCookies } from "../src/cookie-utils";
|
|
4
|
-
import type { Handle, RequestEvent } from "../src/hooks";
|
|
5
|
-
|
|
6
|
-
function fakeEvent(overrides?: Partial<RequestEvent>): RequestEvent {
|
|
7
|
-
return {
|
|
8
|
-
request: new Request("http://localhost/"),
|
|
9
|
-
url: new URL("http://localhost/"),
|
|
10
|
-
params: {},
|
|
11
|
-
locals: {},
|
|
12
|
-
cookies: {
|
|
13
|
-
get: () => undefined,
|
|
14
|
-
set: () => {},
|
|
15
|
-
delete: () => {},
|
|
16
|
-
},
|
|
17
|
-
setHeaders: () => {},
|
|
18
|
-
...overrides,
|
|
19
|
-
};
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
describe("sequence()", () => {
|
|
23
|
-
test("chains two handlers in order", async () => {
|
|
24
|
-
const order: string[] = [];
|
|
25
|
-
|
|
26
|
-
const a: Handle = async ({ event, resolve }) => {
|
|
27
|
-
order.push("a-before");
|
|
28
|
-
const res = await resolve(event);
|
|
29
|
-
order.push("a-after");
|
|
30
|
-
return res;
|
|
31
|
-
};
|
|
32
|
-
|
|
33
|
-
const b: Handle = async ({ event, resolve }) => {
|
|
34
|
-
order.push("b-before");
|
|
35
|
-
const res = await resolve(event);
|
|
36
|
-
order.push("b-after");
|
|
37
|
-
return res;
|
|
38
|
-
};
|
|
39
|
-
|
|
40
|
-
const handler = sequence(a, b);
|
|
41
|
-
const res = await handler({
|
|
42
|
-
event: fakeEvent(),
|
|
43
|
-
resolve: async (evt) => new Response("OK"),
|
|
44
|
-
});
|
|
45
|
-
|
|
46
|
-
expect(order).toEqual(["a-before", "b-before", "b-after", "a-after"]);
|
|
47
|
-
expect(await res.text()).toBe("OK");
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
test("can short-circuit", async () => {
|
|
51
|
-
let reached = false;
|
|
52
|
-
|
|
53
|
-
const a: Handle = async () => {
|
|
54
|
-
return new Response("Blocked", { status: 403 });
|
|
55
|
-
};
|
|
56
|
-
|
|
57
|
-
const b: Handle = async ({ resolve, event }) => {
|
|
58
|
-
reached = true;
|
|
59
|
-
return resolve(event);
|
|
60
|
-
};
|
|
61
|
-
|
|
62
|
-
const handler = sequence(a, b);
|
|
63
|
-
const res = await handler({
|
|
64
|
-
event: fakeEvent(),
|
|
65
|
-
resolve: async () => new Response("OK"),
|
|
66
|
-
});
|
|
67
|
-
|
|
68
|
-
expect(reached).toBe(false);
|
|
69
|
-
expect(res.status).toBe(403);
|
|
70
|
-
});
|
|
71
|
-
|
|
72
|
-
test("can modify event.locals", async () => {
|
|
73
|
-
const a: Handle = async ({ event, resolve }) => {
|
|
74
|
-
event.locals.user = "alice";
|
|
75
|
-
return resolve(event);
|
|
76
|
-
};
|
|
77
|
-
|
|
78
|
-
const b: Handle = async ({ event, resolve }) => {
|
|
79
|
-
event.locals.greet = `hello ${event.locals.user}`;
|
|
80
|
-
return resolve(event);
|
|
81
|
-
};
|
|
82
|
-
|
|
83
|
-
const handler = sequence(a, b);
|
|
84
|
-
const event = fakeEvent();
|
|
85
|
-
await handler({
|
|
86
|
-
event,
|
|
87
|
-
resolve: async (evt) => {
|
|
88
|
-
return new Response(JSON.stringify(evt.locals));
|
|
89
|
-
},
|
|
90
|
-
});
|
|
91
|
-
|
|
92
|
-
expect(event.locals.user).toBe("alice");
|
|
93
|
-
expect(event.locals.greet).toBe("hello alice");
|
|
94
|
-
});
|
|
95
|
-
|
|
96
|
-
test("can modify response", async () => {
|
|
97
|
-
const a: Handle = async ({ event, resolve }) => {
|
|
98
|
-
const res = await resolve(event);
|
|
99
|
-
const headers = new Headers(res.headers);
|
|
100
|
-
headers.set("X-First", "1");
|
|
101
|
-
return new Response(res.body, { status: res.status, headers });
|
|
102
|
-
};
|
|
103
|
-
|
|
104
|
-
const b: Handle = async ({ event, resolve }) => {
|
|
105
|
-
const res = await resolve(event);
|
|
106
|
-
const headers = new Headers(res.headers);
|
|
107
|
-
headers.set("X-Second", "2");
|
|
108
|
-
return new Response(res.body, { status: res.status, headers });
|
|
109
|
-
};
|
|
110
|
-
|
|
111
|
-
const handler = sequence(a, b);
|
|
112
|
-
const res = await handler({
|
|
113
|
-
event: fakeEvent(),
|
|
114
|
-
resolve: async () => new Response("OK"),
|
|
115
|
-
});
|
|
116
|
-
|
|
117
|
-
expect(res.headers.get("X-First")).toBe("1");
|
|
118
|
-
expect(res.headers.get("X-Second")).toBe("2");
|
|
119
|
-
});
|
|
120
|
-
|
|
121
|
-
test("single handler works", async () => {
|
|
122
|
-
const handler: Handle = async ({ event, resolve }) => {
|
|
123
|
-
return resolve(event);
|
|
124
|
-
};
|
|
125
|
-
|
|
126
|
-
const res = await sequence(handler)({
|
|
127
|
-
event: fakeEvent(),
|
|
128
|
-
resolve: async () => new Response("Single"),
|
|
129
|
-
});
|
|
130
|
-
|
|
131
|
-
expect(await res.text()).toBe("Single");
|
|
132
|
-
});
|
|
133
|
-
|
|
134
|
-
test("three handlers chain correctly", async () => {
|
|
135
|
-
const order: string[] = [];
|
|
136
|
-
|
|
137
|
-
const a: Handle = async ({ event, resolve }) => {
|
|
138
|
-
order.push("a");
|
|
139
|
-
return resolve(event);
|
|
140
|
-
};
|
|
141
|
-
const b: Handle = async ({ event, resolve }) => {
|
|
142
|
-
order.push("b");
|
|
143
|
-
return resolve(event);
|
|
144
|
-
};
|
|
145
|
-
const c: Handle = async ({ event, resolve }) => {
|
|
146
|
-
order.push("c");
|
|
147
|
-
return resolve(event);
|
|
148
|
-
};
|
|
149
|
-
|
|
150
|
-
await sequence(a, b, c)({
|
|
151
|
-
event: fakeEvent(),
|
|
152
|
-
resolve: async () => new Response("OK"),
|
|
153
|
-
});
|
|
154
|
-
|
|
155
|
-
expect(order).toEqual(["a", "b", "c"]);
|
|
156
|
-
});
|
|
157
|
-
});
|
|
158
|
-
|
|
159
|
-
describe("Cookies", () => {
|
|
160
|
-
function makeEvent(cookieHeader = ""): RequestEvent {
|
|
161
|
-
return fakeEvent({
|
|
162
|
-
cookies: createCookies(cookieHeader),
|
|
163
|
-
});
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
test("cookies.get reads from header", async () => {
|
|
167
|
-
let value: string | undefined;
|
|
168
|
-
const handler: Handle = async ({ event, resolve }) => {
|
|
169
|
-
value = event.cookies.get("session");
|
|
170
|
-
return resolve(event);
|
|
171
|
-
};
|
|
172
|
-
|
|
173
|
-
await handler({
|
|
174
|
-
event: makeEvent("session=abc123; theme=dark"),
|
|
175
|
-
resolve: async () => new Response("OK"),
|
|
176
|
-
});
|
|
177
|
-
|
|
178
|
-
expect(value).toBe("abc123");
|
|
179
|
-
});
|
|
180
|
-
|
|
181
|
-
test("cookies.set adds Set-Cookie header", async () => {
|
|
182
|
-
const event = makeEvent();
|
|
183
|
-
const handler: Handle = async ({ event: evt, resolve }) => {
|
|
184
|
-
evt.cookies.set("token", "xyz", { httpOnly: true, path: "/" });
|
|
185
|
-
return resolve(evt);
|
|
186
|
-
};
|
|
187
|
-
|
|
188
|
-
await handler({
|
|
189
|
-
event,
|
|
190
|
-
resolve: async () => new Response("OK"),
|
|
191
|
-
});
|
|
192
|
-
|
|
193
|
-
expect(event.cookies.get("token")).toBe("xyz");
|
|
194
|
-
const cookieUtils = event.cookies as ReturnType<typeof createCookies>;
|
|
195
|
-
expect(cookieUtils.toHeaders().length).toBe(1);
|
|
196
|
-
expect(cookieUtils.toHeaders()[0]).toContain("token=xyz");
|
|
197
|
-
expect(cookieUtils.toHeaders()[0]).toContain("HttpOnly");
|
|
198
|
-
expect(cookieUtils.toHeaders()[0]).toContain("Path=/");
|
|
199
|
-
});
|
|
200
|
-
});
|
|
201
|
-
|
|
202
|
-
describe("createHooks()", () => {
|
|
203
|
-
test("returns handle and init", async () => {
|
|
204
|
-
const handle: Handle = async ({ resolve, event }) => resolve(event);
|
|
205
|
-
const init = () => {};
|
|
206
|
-
const hooks = createHooks(handle, init);
|
|
207
|
-
expect(hooks.handle).toBe(handle);
|
|
208
|
-
expect(hooks.init).toBe(init);
|
|
209
|
-
});
|
|
210
|
-
|
|
211
|
-
test("returns handle without init", () => {
|
|
212
|
-
const handle: Handle = async ({ resolve, event }) => resolve(event);
|
|
213
|
-
const hooks = createHooks(handle);
|
|
214
|
-
expect(hooks.handle).toBe(handle);
|
|
215
|
-
expect(hooks.init).toBeUndefined();
|
|
216
|
-
});
|
|
217
|
-
});
|
|
1
|
+
import { describe, expect, test } from "bun:test";
|
|
2
|
+
import { sequence, createHooks } from "../src/hooks";
|
|
3
|
+
import { createCookies } from "../src/cookie-utils";
|
|
4
|
+
import type { Handle, RequestEvent } from "../src/hooks";
|
|
5
|
+
|
|
6
|
+
function fakeEvent(overrides?: Partial<RequestEvent>): RequestEvent {
|
|
7
|
+
return {
|
|
8
|
+
request: new Request("http://localhost/"),
|
|
9
|
+
url: new URL("http://localhost/"),
|
|
10
|
+
params: {},
|
|
11
|
+
locals: {},
|
|
12
|
+
cookies: {
|
|
13
|
+
get: () => undefined,
|
|
14
|
+
set: () => {},
|
|
15
|
+
delete: () => {},
|
|
16
|
+
},
|
|
17
|
+
setHeaders: () => {},
|
|
18
|
+
...overrides,
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
describe("sequence()", () => {
|
|
23
|
+
test("chains two handlers in order", async () => {
|
|
24
|
+
const order: string[] = [];
|
|
25
|
+
|
|
26
|
+
const a: Handle = async ({ event, resolve }) => {
|
|
27
|
+
order.push("a-before");
|
|
28
|
+
const res = await resolve(event);
|
|
29
|
+
order.push("a-after");
|
|
30
|
+
return res;
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
const b: Handle = async ({ event, resolve }) => {
|
|
34
|
+
order.push("b-before");
|
|
35
|
+
const res = await resolve(event);
|
|
36
|
+
order.push("b-after");
|
|
37
|
+
return res;
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
const handler = sequence(a, b);
|
|
41
|
+
const res = await handler({
|
|
42
|
+
event: fakeEvent(),
|
|
43
|
+
resolve: async (evt) => new Response("OK"),
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
expect(order).toEqual(["a-before", "b-before", "b-after", "a-after"]);
|
|
47
|
+
expect(await res.text()).toBe("OK");
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
test("can short-circuit", async () => {
|
|
51
|
+
let reached = false;
|
|
52
|
+
|
|
53
|
+
const a: Handle = async () => {
|
|
54
|
+
return new Response("Blocked", { status: 403 });
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
const b: Handle = async ({ resolve, event }) => {
|
|
58
|
+
reached = true;
|
|
59
|
+
return resolve(event);
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
const handler = sequence(a, b);
|
|
63
|
+
const res = await handler({
|
|
64
|
+
event: fakeEvent(),
|
|
65
|
+
resolve: async () => new Response("OK"),
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
expect(reached).toBe(false);
|
|
69
|
+
expect(res.status).toBe(403);
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
test("can modify event.locals", async () => {
|
|
73
|
+
const a: Handle = async ({ event, resolve }) => {
|
|
74
|
+
event.locals.user = "alice";
|
|
75
|
+
return resolve(event);
|
|
76
|
+
};
|
|
77
|
+
|
|
78
|
+
const b: Handle = async ({ event, resolve }) => {
|
|
79
|
+
event.locals.greet = `hello ${event.locals.user}`;
|
|
80
|
+
return resolve(event);
|
|
81
|
+
};
|
|
82
|
+
|
|
83
|
+
const handler = sequence(a, b);
|
|
84
|
+
const event = fakeEvent();
|
|
85
|
+
await handler({
|
|
86
|
+
event,
|
|
87
|
+
resolve: async (evt) => {
|
|
88
|
+
return new Response(JSON.stringify(evt.locals));
|
|
89
|
+
},
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
expect(event.locals.user).toBe("alice");
|
|
93
|
+
expect(event.locals.greet).toBe("hello alice");
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
test("can modify response", async () => {
|
|
97
|
+
const a: Handle = async ({ event, resolve }) => {
|
|
98
|
+
const res = await resolve(event);
|
|
99
|
+
const headers = new Headers(res.headers);
|
|
100
|
+
headers.set("X-First", "1");
|
|
101
|
+
return new Response(res.body, { status: res.status, headers });
|
|
102
|
+
};
|
|
103
|
+
|
|
104
|
+
const b: Handle = async ({ event, resolve }) => {
|
|
105
|
+
const res = await resolve(event);
|
|
106
|
+
const headers = new Headers(res.headers);
|
|
107
|
+
headers.set("X-Second", "2");
|
|
108
|
+
return new Response(res.body, { status: res.status, headers });
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
const handler = sequence(a, b);
|
|
112
|
+
const res = await handler({
|
|
113
|
+
event: fakeEvent(),
|
|
114
|
+
resolve: async () => new Response("OK"),
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
expect(res.headers.get("X-First")).toBe("1");
|
|
118
|
+
expect(res.headers.get("X-Second")).toBe("2");
|
|
119
|
+
});
|
|
120
|
+
|
|
121
|
+
test("single handler works", async () => {
|
|
122
|
+
const handler: Handle = async ({ event, resolve }) => {
|
|
123
|
+
return resolve(event);
|
|
124
|
+
};
|
|
125
|
+
|
|
126
|
+
const res = await sequence(handler)({
|
|
127
|
+
event: fakeEvent(),
|
|
128
|
+
resolve: async () => new Response("Single"),
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
expect(await res.text()).toBe("Single");
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
test("three handlers chain correctly", async () => {
|
|
135
|
+
const order: string[] = [];
|
|
136
|
+
|
|
137
|
+
const a: Handle = async ({ event, resolve }) => {
|
|
138
|
+
order.push("a");
|
|
139
|
+
return resolve(event);
|
|
140
|
+
};
|
|
141
|
+
const b: Handle = async ({ event, resolve }) => {
|
|
142
|
+
order.push("b");
|
|
143
|
+
return resolve(event);
|
|
144
|
+
};
|
|
145
|
+
const c: Handle = async ({ event, resolve }) => {
|
|
146
|
+
order.push("c");
|
|
147
|
+
return resolve(event);
|
|
148
|
+
};
|
|
149
|
+
|
|
150
|
+
await sequence(a, b, c)({
|
|
151
|
+
event: fakeEvent(),
|
|
152
|
+
resolve: async () => new Response("OK"),
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
expect(order).toEqual(["a", "b", "c"]);
|
|
156
|
+
});
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
describe("Cookies", () => {
|
|
160
|
+
function makeEvent(cookieHeader = ""): RequestEvent {
|
|
161
|
+
return fakeEvent({
|
|
162
|
+
cookies: createCookies(cookieHeader),
|
|
163
|
+
});
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
test("cookies.get reads from header", async () => {
|
|
167
|
+
let value: string | undefined;
|
|
168
|
+
const handler: Handle = async ({ event, resolve }) => {
|
|
169
|
+
value = event.cookies.get("session");
|
|
170
|
+
return resolve(event);
|
|
171
|
+
};
|
|
172
|
+
|
|
173
|
+
await handler({
|
|
174
|
+
event: makeEvent("session=abc123; theme=dark"),
|
|
175
|
+
resolve: async () => new Response("OK"),
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
expect(value).toBe("abc123");
|
|
179
|
+
});
|
|
180
|
+
|
|
181
|
+
test("cookies.set adds Set-Cookie header", async () => {
|
|
182
|
+
const event = makeEvent();
|
|
183
|
+
const handler: Handle = async ({ event: evt, resolve }) => {
|
|
184
|
+
evt.cookies.set("token", "xyz", { httpOnly: true, path: "/" });
|
|
185
|
+
return resolve(evt);
|
|
186
|
+
};
|
|
187
|
+
|
|
188
|
+
await handler({
|
|
189
|
+
event,
|
|
190
|
+
resolve: async () => new Response("OK"),
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
expect(event.cookies.get("token")).toBe("xyz");
|
|
194
|
+
const cookieUtils = event.cookies as ReturnType<typeof createCookies>;
|
|
195
|
+
expect(cookieUtils.toHeaders().length).toBe(1);
|
|
196
|
+
expect(cookieUtils.toHeaders()[0]).toContain("token=xyz");
|
|
197
|
+
expect(cookieUtils.toHeaders()[0]).toContain("HttpOnly");
|
|
198
|
+
expect(cookieUtils.toHeaders()[0]).toContain("Path=/");
|
|
199
|
+
});
|
|
200
|
+
});
|
|
201
|
+
|
|
202
|
+
describe("createHooks()", () => {
|
|
203
|
+
test("returns handle and init", async () => {
|
|
204
|
+
const handle: Handle = async ({ resolve, event }) => resolve(event);
|
|
205
|
+
const init = () => {};
|
|
206
|
+
const hooks = createHooks(handle, init);
|
|
207
|
+
expect(hooks.handle).toBe(handle);
|
|
208
|
+
expect(hooks.init).toBe(init);
|
|
209
|
+
});
|
|
210
|
+
|
|
211
|
+
test("returns handle without init", () => {
|
|
212
|
+
const handle: Handle = async ({ resolve, event }) => resolve(event);
|
|
213
|
+
const hooks = createHooks(handle);
|
|
214
|
+
expect(hooks.handle).toBe(handle);
|
|
215
|
+
expect(hooks.init).toBeUndefined();
|
|
216
|
+
});
|
|
217
|
+
});
|