@qwik.dev/router 0.0.0 → 2.0.0-alpha.1

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.
Files changed (98) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +15 -0
  3. package/adapters/azure-swa/vite.d.ts +2 -0
  4. package/adapters/bun-server/vite.d.ts +2 -0
  5. package/adapters/cloud-run/vite.d.ts +2 -0
  6. package/adapters/cloudflare-pages/vite.d.ts +2 -0
  7. package/adapters/deno-server/vite.d.ts +2 -0
  8. package/adapters/netlify-edge/vite.d.ts +2 -0
  9. package/adapters/node-server/vite.d.ts +2 -0
  10. package/adapters/shared/vite.d.ts +2 -0
  11. package/adapters/static/vite.d.ts +2 -0
  12. package/adapters/vercel-edge/vite.d.ts +2 -0
  13. package/index.d.ts +2 -0
  14. package/lib/adapters/azure-swa/vite/index.cjs +96 -0
  15. package/lib/adapters/azure-swa/vite/index.d.ts +13 -0
  16. package/lib/adapters/azure-swa/vite/index.mjs +61 -0
  17. package/lib/adapters/bun-server/vite/index.cjs +50 -0
  18. package/lib/adapters/bun-server/vite/index.d.ts +14 -0
  19. package/lib/adapters/bun-server/vite/index.mjs +25 -0
  20. package/lib/adapters/cloud-run/vite/index.cjs +47 -0
  21. package/lib/adapters/cloud-run/vite/index.d.ts +13 -0
  22. package/lib/adapters/cloud-run/vite/index.mjs +22 -0
  23. package/lib/adapters/cloudflare-pages/vite/index.cjs +115 -0
  24. package/lib/adapters/cloudflare-pages/vite/index.d.ts +27 -0
  25. package/lib/adapters/cloudflare-pages/vite/index.mjs +80 -0
  26. package/lib/adapters/deno-server/vite/index.cjs +62 -0
  27. package/lib/adapters/deno-server/vite/index.d.ts +14 -0
  28. package/lib/adapters/deno-server/vite/index.mjs +37 -0
  29. package/lib/adapters/netlify-edge/vite/index.cjs +129 -0
  30. package/lib/adapters/netlify-edge/vite/index.d.ts +44 -0
  31. package/lib/adapters/netlify-edge/vite/index.mjs +92 -0
  32. package/lib/adapters/node-server/vite/index.cjs +50 -0
  33. package/lib/adapters/node-server/vite/index.d.ts +14 -0
  34. package/lib/adapters/node-server/vite/index.mjs +25 -0
  35. package/lib/adapters/shared/vite/index.cjs +371 -0
  36. package/lib/adapters/shared/vite/index.d.ts +114 -0
  37. package/lib/adapters/shared/vite/index.mjs +331 -0
  38. package/lib/adapters/static/vite/index.cjs +361 -0
  39. package/lib/adapters/static/vite/index.d.ts +10 -0
  40. package/lib/adapters/static/vite/index.mjs +324 -0
  41. package/lib/adapters/vercel-edge/vite/index.cjs +118 -0
  42. package/lib/adapters/vercel-edge/vite/index.d.ts +45 -0
  43. package/lib/adapters/vercel-edge/vite/index.mjs +83 -0
  44. package/lib/index.d.ts +803 -0
  45. package/lib/index.qwik.cjs +1876 -0
  46. package/lib/index.qwik.mjs +1857 -0
  47. package/lib/middleware/aws-lambda/index.d.ts +50 -0
  48. package/lib/middleware/aws-lambda/index.mjs +55 -0
  49. package/lib/middleware/azure-swa/index.d.ts +28 -0
  50. package/lib/middleware/azure-swa/index.mjs +297 -0
  51. package/lib/middleware/bun/index.d.ts +35 -0
  52. package/lib/middleware/bun/index.mjs +208 -0
  53. package/lib/middleware/cloudflare-pages/index.d.ts +35 -0
  54. package/lib/middleware/cloudflare-pages/index.mjs +107 -0
  55. package/lib/middleware/deno/index.d.ts +47 -0
  56. package/lib/middleware/deno/index.mjs +195 -0
  57. package/lib/middleware/firebase/index.d.ts +26 -0
  58. package/lib/middleware/firebase/index.mjs +36 -0
  59. package/lib/middleware/netlify-edge/index.d.ts +27 -0
  60. package/lib/middleware/netlify-edge/index.mjs +83 -0
  61. package/lib/middleware/node/index.cjs +314 -0
  62. package/lib/middleware/node/index.d.ts +66 -0
  63. package/lib/middleware/node/index.mjs +287 -0
  64. package/lib/middleware/request-handler/index.cjs +1544 -0
  65. package/lib/middleware/request-handler/index.d.ts +681 -0
  66. package/lib/middleware/request-handler/index.mjs +1501 -0
  67. package/lib/middleware/vercel-edge/index.d.ts +26 -0
  68. package/lib/middleware/vercel-edge/index.mjs +109 -0
  69. package/lib/modules.d.ts +38 -0
  70. package/lib/service-worker.cjs +267 -0
  71. package/lib/service-worker.d.ts +4 -0
  72. package/lib/service-worker.mjs +265 -0
  73. package/lib/static/deno.mjs +8 -0
  74. package/lib/static/index.cjs +67 -0
  75. package/lib/static/index.d.ts +98 -0
  76. package/lib/static/index.mjs +48 -0
  77. package/lib/static/node.cjs +1124 -0
  78. package/lib/static/node.mjs +1086 -0
  79. package/lib/vite/index.cjs +27496 -0
  80. package/lib/vite/index.d.ts +154 -0
  81. package/lib/vite/index.mjs +27484 -0
  82. package/middleware/aws-lambda.d.ts +2 -0
  83. package/middleware/azure-swa.d.ts +2 -0
  84. package/middleware/bun.d.ts +2 -0
  85. package/middleware/cloudflare-pages.d.ts +2 -0
  86. package/middleware/deno.d.ts +2 -0
  87. package/middleware/firebase.d.ts +2 -0
  88. package/middleware/netlify-edge.d.ts +2 -0
  89. package/middleware/node.d.ts +2 -0
  90. package/middleware/request-handler/generated/not-found-paths.ts +7 -0
  91. package/middleware/request-handler/generated/static-paths.ts +35 -0
  92. package/middleware/request-handler.d.ts +2 -0
  93. package/middleware/vercel-edge.d.ts +2 -0
  94. package/modules.d.ts +38 -0
  95. package/package.json +171 -165
  96. package/service-worker.d.ts +2 -0
  97. package/static.d.ts +2 -0
  98. package/vite.d.ts +2 -0
@@ -0,0 +1,314 @@
1
+ "use strict";
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __export = (target, all) => {
7
+ for (var name in all)
8
+ __defProp(target, name, { get: all[name], enumerable: true });
9
+ };
10
+ var __copyProps = (to, from, except, desc) => {
11
+ if (from && typeof from === "object" || typeof from === "function") {
12
+ for (let key of __getOwnPropNames(from))
13
+ if (!__hasOwnProp.call(to, key) && key !== except)
14
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
+ }
16
+ return to;
17
+ };
18
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
+
20
+ // packages/qwik-router/src/middleware/node/index.ts
21
+ var node_exports = {};
22
+ __export(node_exports, {
23
+ createQwikCity: () => createQwikCity,
24
+ createQwikRouter: () => createQwikRouter
25
+ });
26
+ module.exports = __toCommonJS(node_exports);
27
+ var import_qwik_router_not_found_paths = require("@qwik-router-not-found-paths");
28
+ var import_qwik_router_static_paths = require("@qwik-router-static-paths");
29
+ var import_internal = require("@qwik.dev/core/internal");
30
+ var import_server = require("@qwik.dev/core/server");
31
+ var import_request_handler = require("../request-handler/index.cjs");
32
+ var import_node_fs = require("node:fs");
33
+ var import_node_path = require("node:path");
34
+ var import_node_url = require("node:url");
35
+
36
+ // packages/qwik-router/src/middleware/request-handler/mime-types.ts
37
+ var MIME_TYPES = {
38
+ "3gp": "video/3gpp",
39
+ "3gpp": "video/3gpp",
40
+ asf: "video/x-ms-asf",
41
+ asx: "video/x-ms-asf",
42
+ avi: "video/x-msvideo",
43
+ avif: "image/avif",
44
+ bmp: "image/x-ms-bmp",
45
+ css: "text/css",
46
+ flv: "video/x-flv",
47
+ gif: "image/gif",
48
+ htm: "text/html",
49
+ html: "text/html",
50
+ ico: "image/x-icon",
51
+ jng: "image/x-jng",
52
+ jpeg: "image/jpeg",
53
+ jpg: "image/jpeg",
54
+ js: "application/javascript",
55
+ json: "application/json",
56
+ kar: "audio/midi",
57
+ m4a: "audio/x-m4a",
58
+ m4v: "video/x-m4v",
59
+ mid: "audio/midi",
60
+ midi: "audio/midi",
61
+ mng: "video/x-mng",
62
+ mov: "video/quicktime",
63
+ mp3: "audio/mpeg",
64
+ mp4: "video/mp4",
65
+ mpeg: "video/mpeg",
66
+ mpg: "video/mpeg",
67
+ ogg: "audio/ogg",
68
+ pdf: "application/pdf",
69
+ png: "image/png",
70
+ rar: "application/x-rar-compressed",
71
+ shtml: "text/html",
72
+ svg: "image/svg+xml",
73
+ svgz: "image/svg+xml",
74
+ tif: "image/tiff",
75
+ tiff: "image/tiff",
76
+ ts: "video/mp2t",
77
+ txt: "text/plain",
78
+ wbmp: "image/vnd.wap.wbmp",
79
+ webm: "video/webm",
80
+ webp: "image/webp",
81
+ wmv: "video/x-ms-wmv",
82
+ woff: "font/woff",
83
+ woff2: "font/woff2",
84
+ xml: "text/xml",
85
+ zip: "application/zip"
86
+ };
87
+
88
+ // packages/qwik-router/src/middleware/node/http.ts
89
+ var import_node_http2 = require("node:http2");
90
+ function computeOrigin(req, opts) {
91
+ var _a;
92
+ return ((_a = opts == null ? void 0 : opts.getOrigin) == null ? void 0 : _a.call(opts, req)) ?? (opts == null ? void 0 : opts.origin) ?? process.env.ORIGIN ?? fallbackOrigin(req);
93
+ }
94
+ function fallbackOrigin(req) {
95
+ const { PROTOCOL_HEADER, HOST_HEADER } = process.env;
96
+ const headers = req.headers;
97
+ const protocol = PROTOCOL_HEADER && headers[PROTOCOL_HEADER] || (req.socket.encrypted || req.connection.encrypted ? "https" : "http");
98
+ const hostHeader = HOST_HEADER ?? (req instanceof import_node_http2.Http2ServerRequest ? ":authority" : "host");
99
+ const host = headers[hostHeader];
100
+ return `${protocol}://${host}`;
101
+ }
102
+ function getUrl(req, origin) {
103
+ return normalizeUrl(req.originalUrl || req.url || "/", origin);
104
+ }
105
+ function isIgnoredError(message = "") {
106
+ const ignoredErrors = ["The stream has been destroyed", "write after end"];
107
+ return ignoredErrors.some((ignored) => message.includes(ignored));
108
+ }
109
+ var invalidHeadersPattern = /^:(method|scheme|authority|path)$/i;
110
+ function normalizeUrl(url, base) {
111
+ const DOUBLE_SLASH_REG = /\/\/|\\\\/g;
112
+ return new URL(url.replace(DOUBLE_SLASH_REG, "/"), base);
113
+ }
114
+ async function fromNodeHttp(url, req, res, mode, getClientConn) {
115
+ const requestHeaders = new Headers();
116
+ const nodeRequestHeaders = req.headers;
117
+ try {
118
+ for (const [key, value] of Object.entries(nodeRequestHeaders)) {
119
+ if (invalidHeadersPattern.test(key)) {
120
+ continue;
121
+ }
122
+ if (typeof value === "string") {
123
+ requestHeaders.set(key, value);
124
+ } else if (Array.isArray(value)) {
125
+ for (const v of value) {
126
+ requestHeaders.append(key, v);
127
+ }
128
+ }
129
+ }
130
+ } catch (err) {
131
+ console.error(err);
132
+ }
133
+ const getRequestBody = async function* () {
134
+ for await (const chunk of req) {
135
+ yield chunk;
136
+ }
137
+ };
138
+ const body = req.method === "HEAD" || req.method === "GET" ? void 0 : getRequestBody();
139
+ const controller = new AbortController();
140
+ const options = {
141
+ method: req.method,
142
+ headers: requestHeaders,
143
+ body,
144
+ signal: controller.signal,
145
+ duplex: "half"
146
+ };
147
+ res.on("close", () => {
148
+ controller.abort();
149
+ });
150
+ const serverRequestEv = {
151
+ mode,
152
+ url,
153
+ request: new Request(url.href, options),
154
+ env: {
155
+ get(key) {
156
+ return process.env[key];
157
+ }
158
+ },
159
+ getWritableStream: (status, headers, cookies) => {
160
+ res.statusCode = status;
161
+ try {
162
+ for (const [key, value] of headers) {
163
+ if (invalidHeadersPattern.test(key)) {
164
+ continue;
165
+ }
166
+ res.setHeader(key, value);
167
+ }
168
+ const cookieHeaders = cookies.headers();
169
+ if (cookieHeaders.length > 0) {
170
+ res.setHeader("Set-Cookie", cookieHeaders);
171
+ }
172
+ } catch (err) {
173
+ console.error(err);
174
+ }
175
+ return new WritableStream({
176
+ write(chunk) {
177
+ if (res.closed || res.destroyed) {
178
+ return;
179
+ }
180
+ res.write(chunk, (error) => {
181
+ if (error && !isIgnoredError(error.message)) {
182
+ console.error(error);
183
+ }
184
+ });
185
+ },
186
+ close() {
187
+ res.end();
188
+ }
189
+ });
190
+ },
191
+ getClientConn: () => {
192
+ return getClientConn ? getClientConn(req) : {
193
+ ip: req.socket.remoteAddress
194
+ };
195
+ },
196
+ platform: {
197
+ ssr: true,
198
+ incomingMessage: req,
199
+ node: process.versions.node
200
+ // Weirdly needed to make typecheck of insights happy
201
+ },
202
+ locale: void 0
203
+ };
204
+ return serverRequestEv;
205
+ }
206
+
207
+ // packages/qwik-router/src/middleware/node/index.ts
208
+ var import_meta = {};
209
+ function createQwikRouter(opts) {
210
+ var _a;
211
+ if (opts.qwikCityPlan && !opts.qwikRouterConfig) {
212
+ console.warn("qwikCityPlan is deprecated. Use qwikRouterConfig instead.");
213
+ opts.qwikRouterConfig = opts.qwikCityPlan;
214
+ } else if (!opts.qwikRouterConfig) {
215
+ throw new Error("qwikRouterConfig is required.");
216
+ }
217
+ const qwikSerializer = {
218
+ _deserialize: import_internal._deserialize,
219
+ _serialize: import_internal._serialize,
220
+ _verifySerializable: import_internal._verifySerializable
221
+ };
222
+ if (opts.manifest) {
223
+ (0, import_server.setServerPlatform)(opts.manifest);
224
+ }
225
+ const staticFolder = ((_a = opts.static) == null ? void 0 : _a.root) ?? (0, import_node_path.join)((0, import_node_url.fileURLToPath)(import_meta.url), "..", "..", "dist");
226
+ const router = async (req, res, next) => {
227
+ try {
228
+ const origin = computeOrigin(req, opts);
229
+ const serverRequestEv = await fromNodeHttp(
230
+ getUrl(req, origin),
231
+ req,
232
+ res,
233
+ "server",
234
+ opts.getClientConn
235
+ );
236
+ const handled = await (0, import_request_handler.requestHandler)(serverRequestEv, opts, qwikSerializer);
237
+ if (handled) {
238
+ const err = await handled.completion;
239
+ if (err) {
240
+ throw err;
241
+ }
242
+ if (handled.requestEv.headersSent) {
243
+ return;
244
+ }
245
+ }
246
+ next();
247
+ } catch (e) {
248
+ console.error(e);
249
+ next(e);
250
+ }
251
+ };
252
+ const notFound = async (req, res, next) => {
253
+ try {
254
+ if (!res.headersSent) {
255
+ const origin = computeOrigin(req, opts);
256
+ const url = getUrl(req, origin);
257
+ const notFoundHtml = (0, import_qwik_router_static_paths.isStaticPath)(req.method || "GET", url) ? "Not Found" : (0, import_qwik_router_not_found_paths.getNotFound)(url.pathname);
258
+ res.writeHead(404, {
259
+ "Content-Type": "text/html; charset=utf-8",
260
+ "X-Not-Found": url.pathname
261
+ });
262
+ res.end(notFoundHtml);
263
+ }
264
+ } catch (e) {
265
+ console.error(e);
266
+ next(e);
267
+ }
268
+ };
269
+ const staticFile = async (req, res, next) => {
270
+ var _a2;
271
+ try {
272
+ const origin = computeOrigin(req, opts);
273
+ const url = getUrl(req, origin);
274
+ if ((0, import_qwik_router_static_paths.isStaticPath)(req.method || "GET", url)) {
275
+ const pathname = url.pathname;
276
+ let filePath;
277
+ if ((0, import_node_path.basename)(pathname).includes(".")) {
278
+ filePath = (0, import_node_path.join)(staticFolder, pathname);
279
+ } else if (opts.qwikRouterConfig.trailingSlash) {
280
+ filePath = (0, import_node_path.join)(staticFolder, pathname + "index.html");
281
+ } else {
282
+ filePath = (0, import_node_path.join)(staticFolder, pathname, "index.html");
283
+ }
284
+ const ext = (0, import_node_path.extname)(filePath).replace(/^\./, "");
285
+ const stream = (0, import_node_fs.createReadStream)(filePath);
286
+ stream.on("error", next);
287
+ const contentType = MIME_TYPES[ext];
288
+ if (contentType) {
289
+ res.setHeader("Content-Type", contentType);
290
+ }
291
+ if ((_a2 = opts.static) == null ? void 0 : _a2.cacheControl) {
292
+ res.setHeader("Cache-Control", opts.static.cacheControl);
293
+ }
294
+ stream.pipe(res);
295
+ return;
296
+ }
297
+ return next();
298
+ } catch (e) {
299
+ console.error(e);
300
+ next(e);
301
+ }
302
+ };
303
+ return {
304
+ router,
305
+ notFound,
306
+ staticFile
307
+ };
308
+ }
309
+ var createQwikCity = createQwikRouter;
310
+ // Annotate the CommonJS export names for ESM import in node:
311
+ 0 && (module.exports = {
312
+ createQwikCity,
313
+ createQwikRouter
314
+ });
@@ -0,0 +1,66 @@
1
+ /// <reference types="node" />
2
+
3
+ import type { ClientConn } from '@qwik.dev/router/middleware/request-handler';
4
+ import type { Http2ServerRequest } from 'node:http2';
5
+ import type { IncomingMessage } from 'node:http';
6
+ import type { ServerRenderOptions } from '@qwik.dev/router/middleware/request-handler';
7
+ import type { ServerResponse } from 'node:http';
8
+
9
+ /**
10
+ * @deprecated Use `createQwikRouter` instead. Will be removed in V3
11
+ * @public
12
+ */
13
+ export declare const createQwikCity: typeof createQwikRouter;
14
+
15
+ /** @public */
16
+ export declare function createQwikRouter(opts: QwikRouterNodeRequestOptions | QwikCityNodeRequestOptions): {
17
+ router: (req: IncomingMessage | Http2ServerRequest, res: ServerResponse, next: NodeRequestNextFunction) => Promise<void>;
18
+ notFound: (req: IncomingMessage | Http2ServerRequest, res: ServerResponse, next: (e: any) => void) => Promise<void>;
19
+ staticFile: (req: IncomingMessage | Http2ServerRequest, res: ServerResponse, next: (e?: any) => void) => Promise<void>;
20
+ };
21
+
22
+ /** @public */
23
+ export declare interface NodeRequestNextFunction {
24
+ (err?: any): void;
25
+ }
26
+
27
+ /** @public */
28
+ export declare interface PlatformNode {
29
+ ssr?: true;
30
+ incomingMessage?: IncomingMessage | Http2ServerRequest;
31
+ node?: string;
32
+ }
33
+
34
+ /**
35
+ * @deprecated Use `QwikRouterNodeRequestOptions` instead. Will be removed in V3
36
+ * @public
37
+ */
38
+ export declare type QwikCityNodeRequestOptions = QwikRouterNodeRequestOptions;
39
+
40
+ /** @public */
41
+ export declare interface QwikRouterNodeRequestOptions extends ServerRenderOptions {
42
+ /** Options for serving static files */
43
+ static?: {
44
+ /** The root folder for statics files. Defaults to /dist */
45
+ root?: string;
46
+ /** Set the Cache-Control header for all static files */
47
+ cacheControl?: string;
48
+ };
49
+ /**
50
+ * Provide a function that computes the origin of the server, used to resolve relative URLs and
51
+ * validate the request origin against CSRF attacks.
52
+ *
53
+ * When not specified, it defaults to the `ORIGIN` environment variable (if set).
54
+ *
55
+ * If `ORIGIN` is not set, it's derived from the incoming request, which is not recommended for
56
+ * production use. You can specify the `PROTOCOL_HEADER`, `HOST_HEADER` to `X-Forwarded-Proto` and
57
+ * `X-Forwarded-Host` respectively to override the default behavior.
58
+ */
59
+ getOrigin?: (req: IncomingMessage | Http2ServerRequest) => string | null;
60
+ /** Provide a function that returns a `ClientConn` for the given request. */
61
+ getClientConn?: (req: IncomingMessage | Http2ServerRequest) => ClientConn;
62
+ /** @deprecated Use `getOrigin` instead. Will be removed in V3 */
63
+ origin?: string;
64
+ }
65
+
66
+ export { }
@@ -0,0 +1,287 @@
1
+ // packages/qwik-router/src/middleware/node/index.ts
2
+ import { getNotFound } from "@qwik-router-not-found-paths";
3
+ import { isStaticPath } from "@qwik-router-static-paths";
4
+ import { _deserialize, _serialize, _verifySerializable } from "@qwik.dev/core/internal";
5
+ import { setServerPlatform } from "@qwik.dev/core/server";
6
+ import { requestHandler } from "../request-handler/index.mjs";
7
+ import { createReadStream } from "node:fs";
8
+ import { basename, extname, join } from "node:path";
9
+ import { fileURLToPath } from "node:url";
10
+
11
+ // packages/qwik-router/src/middleware/request-handler/mime-types.ts
12
+ var MIME_TYPES = {
13
+ "3gp": "video/3gpp",
14
+ "3gpp": "video/3gpp",
15
+ asf: "video/x-ms-asf",
16
+ asx: "video/x-ms-asf",
17
+ avi: "video/x-msvideo",
18
+ avif: "image/avif",
19
+ bmp: "image/x-ms-bmp",
20
+ css: "text/css",
21
+ flv: "video/x-flv",
22
+ gif: "image/gif",
23
+ htm: "text/html",
24
+ html: "text/html",
25
+ ico: "image/x-icon",
26
+ jng: "image/x-jng",
27
+ jpeg: "image/jpeg",
28
+ jpg: "image/jpeg",
29
+ js: "application/javascript",
30
+ json: "application/json",
31
+ kar: "audio/midi",
32
+ m4a: "audio/x-m4a",
33
+ m4v: "video/x-m4v",
34
+ mid: "audio/midi",
35
+ midi: "audio/midi",
36
+ mng: "video/x-mng",
37
+ mov: "video/quicktime",
38
+ mp3: "audio/mpeg",
39
+ mp4: "video/mp4",
40
+ mpeg: "video/mpeg",
41
+ mpg: "video/mpeg",
42
+ ogg: "audio/ogg",
43
+ pdf: "application/pdf",
44
+ png: "image/png",
45
+ rar: "application/x-rar-compressed",
46
+ shtml: "text/html",
47
+ svg: "image/svg+xml",
48
+ svgz: "image/svg+xml",
49
+ tif: "image/tiff",
50
+ tiff: "image/tiff",
51
+ ts: "video/mp2t",
52
+ txt: "text/plain",
53
+ wbmp: "image/vnd.wap.wbmp",
54
+ webm: "video/webm",
55
+ webp: "image/webp",
56
+ wmv: "video/x-ms-wmv",
57
+ woff: "font/woff",
58
+ woff2: "font/woff2",
59
+ xml: "text/xml",
60
+ zip: "application/zip"
61
+ };
62
+
63
+ // packages/qwik-router/src/middleware/node/http.ts
64
+ import { Http2ServerRequest } from "node:http2";
65
+ function computeOrigin(req, opts) {
66
+ var _a;
67
+ return ((_a = opts == null ? void 0 : opts.getOrigin) == null ? void 0 : _a.call(opts, req)) ?? (opts == null ? void 0 : opts.origin) ?? process.env.ORIGIN ?? fallbackOrigin(req);
68
+ }
69
+ function fallbackOrigin(req) {
70
+ const { PROTOCOL_HEADER, HOST_HEADER } = process.env;
71
+ const headers = req.headers;
72
+ const protocol = PROTOCOL_HEADER && headers[PROTOCOL_HEADER] || (req.socket.encrypted || req.connection.encrypted ? "https" : "http");
73
+ const hostHeader = HOST_HEADER ?? (req instanceof Http2ServerRequest ? ":authority" : "host");
74
+ const host = headers[hostHeader];
75
+ return `${protocol}://${host}`;
76
+ }
77
+ function getUrl(req, origin) {
78
+ return normalizeUrl(req.originalUrl || req.url || "/", origin);
79
+ }
80
+ function isIgnoredError(message = "") {
81
+ const ignoredErrors = ["The stream has been destroyed", "write after end"];
82
+ return ignoredErrors.some((ignored) => message.includes(ignored));
83
+ }
84
+ var invalidHeadersPattern = /^:(method|scheme|authority|path)$/i;
85
+ function normalizeUrl(url, base) {
86
+ const DOUBLE_SLASH_REG = /\/\/|\\\\/g;
87
+ return new URL(url.replace(DOUBLE_SLASH_REG, "/"), base);
88
+ }
89
+ async function fromNodeHttp(url, req, res, mode, getClientConn) {
90
+ const requestHeaders = new Headers();
91
+ const nodeRequestHeaders = req.headers;
92
+ try {
93
+ for (const [key, value] of Object.entries(nodeRequestHeaders)) {
94
+ if (invalidHeadersPattern.test(key)) {
95
+ continue;
96
+ }
97
+ if (typeof value === "string") {
98
+ requestHeaders.set(key, value);
99
+ } else if (Array.isArray(value)) {
100
+ for (const v of value) {
101
+ requestHeaders.append(key, v);
102
+ }
103
+ }
104
+ }
105
+ } catch (err) {
106
+ console.error(err);
107
+ }
108
+ const getRequestBody = async function* () {
109
+ for await (const chunk of req) {
110
+ yield chunk;
111
+ }
112
+ };
113
+ const body = req.method === "HEAD" || req.method === "GET" ? void 0 : getRequestBody();
114
+ const controller = new AbortController();
115
+ const options = {
116
+ method: req.method,
117
+ headers: requestHeaders,
118
+ body,
119
+ signal: controller.signal,
120
+ duplex: "half"
121
+ };
122
+ res.on("close", () => {
123
+ controller.abort();
124
+ });
125
+ const serverRequestEv = {
126
+ mode,
127
+ url,
128
+ request: new Request(url.href, options),
129
+ env: {
130
+ get(key) {
131
+ return process.env[key];
132
+ }
133
+ },
134
+ getWritableStream: (status, headers, cookies) => {
135
+ res.statusCode = status;
136
+ try {
137
+ for (const [key, value] of headers) {
138
+ if (invalidHeadersPattern.test(key)) {
139
+ continue;
140
+ }
141
+ res.setHeader(key, value);
142
+ }
143
+ const cookieHeaders = cookies.headers();
144
+ if (cookieHeaders.length > 0) {
145
+ res.setHeader("Set-Cookie", cookieHeaders);
146
+ }
147
+ } catch (err) {
148
+ console.error(err);
149
+ }
150
+ return new WritableStream({
151
+ write(chunk) {
152
+ if (res.closed || res.destroyed) {
153
+ return;
154
+ }
155
+ res.write(chunk, (error) => {
156
+ if (error && !isIgnoredError(error.message)) {
157
+ console.error(error);
158
+ }
159
+ });
160
+ },
161
+ close() {
162
+ res.end();
163
+ }
164
+ });
165
+ },
166
+ getClientConn: () => {
167
+ return getClientConn ? getClientConn(req) : {
168
+ ip: req.socket.remoteAddress
169
+ };
170
+ },
171
+ platform: {
172
+ ssr: true,
173
+ incomingMessage: req,
174
+ node: process.versions.node
175
+ // Weirdly needed to make typecheck of insights happy
176
+ },
177
+ locale: void 0
178
+ };
179
+ return serverRequestEv;
180
+ }
181
+
182
+ // packages/qwik-router/src/middleware/node/index.ts
183
+ function createQwikRouter(opts) {
184
+ var _a;
185
+ if (opts.qwikCityPlan && !opts.qwikRouterConfig) {
186
+ console.warn("qwikCityPlan is deprecated. Use qwikRouterConfig instead.");
187
+ opts.qwikRouterConfig = opts.qwikCityPlan;
188
+ } else if (!opts.qwikRouterConfig) {
189
+ throw new Error("qwikRouterConfig is required.");
190
+ }
191
+ const qwikSerializer = {
192
+ _deserialize,
193
+ _serialize,
194
+ _verifySerializable
195
+ };
196
+ if (opts.manifest) {
197
+ setServerPlatform(opts.manifest);
198
+ }
199
+ const staticFolder = ((_a = opts.static) == null ? void 0 : _a.root) ?? join(fileURLToPath(import.meta.url), "..", "..", "dist");
200
+ const router = async (req, res, next) => {
201
+ try {
202
+ const origin = computeOrigin(req, opts);
203
+ const serverRequestEv = await fromNodeHttp(
204
+ getUrl(req, origin),
205
+ req,
206
+ res,
207
+ "server",
208
+ opts.getClientConn
209
+ );
210
+ const handled = await requestHandler(serverRequestEv, opts, qwikSerializer);
211
+ if (handled) {
212
+ const err = await handled.completion;
213
+ if (err) {
214
+ throw err;
215
+ }
216
+ if (handled.requestEv.headersSent) {
217
+ return;
218
+ }
219
+ }
220
+ next();
221
+ } catch (e) {
222
+ console.error(e);
223
+ next(e);
224
+ }
225
+ };
226
+ const notFound = async (req, res, next) => {
227
+ try {
228
+ if (!res.headersSent) {
229
+ const origin = computeOrigin(req, opts);
230
+ const url = getUrl(req, origin);
231
+ const notFoundHtml = isStaticPath(req.method || "GET", url) ? "Not Found" : getNotFound(url.pathname);
232
+ res.writeHead(404, {
233
+ "Content-Type": "text/html; charset=utf-8",
234
+ "X-Not-Found": url.pathname
235
+ });
236
+ res.end(notFoundHtml);
237
+ }
238
+ } catch (e) {
239
+ console.error(e);
240
+ next(e);
241
+ }
242
+ };
243
+ const staticFile = async (req, res, next) => {
244
+ var _a2;
245
+ try {
246
+ const origin = computeOrigin(req, opts);
247
+ const url = getUrl(req, origin);
248
+ if (isStaticPath(req.method || "GET", url)) {
249
+ const pathname = url.pathname;
250
+ let filePath;
251
+ if (basename(pathname).includes(".")) {
252
+ filePath = join(staticFolder, pathname);
253
+ } else if (opts.qwikRouterConfig.trailingSlash) {
254
+ filePath = join(staticFolder, pathname + "index.html");
255
+ } else {
256
+ filePath = join(staticFolder, pathname, "index.html");
257
+ }
258
+ const ext = extname(filePath).replace(/^\./, "");
259
+ const stream = createReadStream(filePath);
260
+ stream.on("error", next);
261
+ const contentType = MIME_TYPES[ext];
262
+ if (contentType) {
263
+ res.setHeader("Content-Type", contentType);
264
+ }
265
+ if ((_a2 = opts.static) == null ? void 0 : _a2.cacheControl) {
266
+ res.setHeader("Cache-Control", opts.static.cacheControl);
267
+ }
268
+ stream.pipe(res);
269
+ return;
270
+ }
271
+ return next();
272
+ } catch (e) {
273
+ console.error(e);
274
+ next(e);
275
+ }
276
+ };
277
+ return {
278
+ router,
279
+ notFound,
280
+ staticFile
281
+ };
282
+ }
283
+ var createQwikCity = createQwikRouter;
284
+ export {
285
+ createQwikCity,
286
+ createQwikRouter
287
+ };