nitro-nightly 4.0.0-20251010-091516-7cafddba → 4.0.0-20251030-121649-98b56dfe

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 (252) hide show
  1. package/dist/_build/common.mjs +763 -0
  2. package/dist/_build/rolldown.mjs +216 -0
  3. package/dist/_build/rollup.mjs +359 -0
  4. package/dist/_build/vite.build.mjs +67 -0
  5. package/dist/_build/vite.plugin.mjs +747 -0
  6. package/dist/_chunks/B-7HiF0V.mjs +1303 -0
  7. package/dist/_chunks/B3asVbT4.mjs +151 -0
  8. package/dist/_chunks/B5ooyfLk.mjs +372 -0
  9. package/dist/_chunks/DXMHy5ca.mjs +43 -0
  10. package/dist/_dev.d.mts +12 -0
  11. package/dist/_dev.mjs +675 -0
  12. package/dist/_libs/acorn.mjs +5034 -0
  13. package/dist/_libs/c12.mjs +2798 -0
  14. package/dist/_libs/chokidar.mjs +1560 -0
  15. package/dist/_libs/citty.mjs +333 -0
  16. package/dist/_libs/commondir.mjs +22 -0
  17. package/dist/_libs/compatx.mjs +46 -0
  18. package/dist/_libs/confbox.mjs +2920 -0
  19. package/dist/_libs/debug.mjs +848 -0
  20. package/dist/_libs/deepmerge.mjs +86 -0
  21. package/dist/_libs/depd.mjs +314 -0
  22. package/dist/_libs/dot-prop.mjs +138 -0
  23. package/dist/_libs/duplexer.mjs +71 -0
  24. package/dist/_libs/ee-first.mjs +68 -0
  25. package/dist/_libs/encodeurl.mjs +49 -0
  26. package/dist/_libs/escape-html.mjs +58 -0
  27. package/dist/_libs/escape-string-regexp.mjs +8 -0
  28. package/dist/_libs/estree-walker.mjs +330 -0
  29. package/dist/_libs/etag.mjs +75 -0
  30. package/dist/_libs/fdir.mjs +514 -0
  31. package/dist/_libs/fresh.mjs +85 -0
  32. package/dist/_libs/function-bind.mjs +63 -0
  33. package/dist/_libs/gen-mapping.mjs +612 -0
  34. package/dist/_libs/giget.mjs +19076 -0
  35. package/dist/_libs/gzip-size.mjs +21 -0
  36. package/dist/_libs/hasown.mjs +14 -0
  37. package/dist/_libs/http-errors.mjs +436 -0
  38. package/dist/_libs/httpxy.mjs +410 -0
  39. package/dist/_libs/is-core-module.mjs +220 -0
  40. package/dist/_libs/is-module.mjs +13 -0
  41. package/dist/_libs/is-reference.mjs +32 -0
  42. package/dist/_libs/js-tokens.mjs +382 -0
  43. package/dist/_libs/klona.mjs +90 -0
  44. package/dist/_libs/knitwork.mjs +124 -0
  45. package/dist/_libs/local-pkg.mjs +1686 -0
  46. package/dist/_libs/magic-string.mjs +939 -0
  47. package/dist/_libs/mime-db.mjs +7642 -0
  48. package/dist/_libs/mime-types.mjs +162 -0
  49. package/dist/_libs/mime.mjs +1391 -0
  50. package/dist/_libs/node-fetch-native.mjs +173 -0
  51. package/dist/_libs/on-finished.mjs +170 -0
  52. package/dist/_libs/parseurl.mjs +103 -0
  53. package/dist/_libs/path-parse.mjs +47 -0
  54. package/dist/_libs/pathe.mjs +41 -0
  55. package/dist/_libs/picomatch.mjs +1673 -0
  56. package/dist/_libs/plugin-alias.mjs +64 -0
  57. package/dist/_libs/plugin-commonjs.mjs +1489 -0
  58. package/dist/_libs/plugin-inject.mjs +135 -0
  59. package/dist/_libs/plugin-json.mjs +36 -0
  60. package/dist/_libs/plugin-node-resolve.mjs +1553 -0
  61. package/dist/_libs/plugin-replace.mjs +104 -0
  62. package/dist/_libs/pretty-bytes.mjs +116 -0
  63. package/dist/_libs/range-parser.mjs +102 -0
  64. package/dist/_libs/remapping.mjs +117 -0
  65. package/dist/_libs/rou3.mjs +311 -0
  66. package/dist/_libs/send.mjs +857 -0
  67. package/dist/_libs/serve-static.mjs +124 -0
  68. package/dist/_libs/std-env.mjs +158 -0
  69. package/dist/_libs/strip-literal.mjs +51 -0
  70. package/dist/_libs/tinyglobby.mjs +292 -0
  71. package/dist/_libs/ultrahtml.mjs +138 -0
  72. package/dist/_libs/unimport.mjs +3520 -0
  73. package/dist/_libs/untyped.mjs +271 -0
  74. package/dist/_libs/unwasm.mjs +5825 -0
  75. package/dist/_libs/vite-plugin-fullstack.mjs +561 -0
  76. package/dist/_presets.mjs +1723 -0
  77. package/dist/builder.d.mts +54 -0
  78. package/dist/builder.mjs +45 -0
  79. package/dist/cli/_chunks/build.mjs +43 -0
  80. package/dist/cli/_chunks/common.mjs +15 -0
  81. package/dist/cli/_chunks/detect-acorn.mjs +12 -0
  82. package/dist/cli/_chunks/dev.mjs +79 -0
  83. package/dist/cli/_chunks/dist.mjs +3 -0
  84. package/dist/cli/_chunks/dist2.mjs +4 -0
  85. package/dist/cli/_chunks/esm.mjs +3 -0
  86. package/dist/cli/_chunks/json5.mjs +4 -0
  87. package/dist/cli/_chunks/jsonc.mjs +4 -0
  88. package/dist/cli/_chunks/list.mjs +27 -0
  89. package/dist/cli/_chunks/multipart-parser.mjs +6 -0
  90. package/dist/cli/_chunks/prepare.mjs +19 -0
  91. package/dist/cli/_chunks/run.mjs +55 -0
  92. package/dist/cli/_chunks/task.mjs +16 -0
  93. package/dist/cli/_chunks/toml.mjs +4 -0
  94. package/dist/cli/_chunks/yaml.mjs +4 -0
  95. package/dist/cli/index.d.mts +1 -1
  96. package/dist/cli/index.mjs +20 -476
  97. package/dist/node_modules/@speed-highlight/core/dist/index.js +1 -1
  98. package/dist/node_modules/@speed-highlight/core/dist/terminal.js +1 -1
  99. package/dist/node_modules/@speed-highlight/core/package.json +7 -3
  100. package/dist/node_modules/cookie-es/dist/index.mjs +262 -0
  101. package/dist/node_modules/cookie-es/package.json +37 -0
  102. package/dist/node_modules/hookable/dist/index.mjs +243 -266
  103. package/dist/node_modules/hookable/package.json +29 -26
  104. package/dist/node_modules/rendu/dist/index.mjs +380 -0
  105. package/dist/node_modules/rendu/package.json +47 -0
  106. package/dist/presets/_nitro/runtime/nitro-dev.d.mts +1 -0
  107. package/dist/presets/_nitro/runtime/nitro-dev.mjs +44 -38
  108. package/dist/presets/_nitro/runtime/nitro-prerenderer.d.mts +2 -2
  109. package/dist/presets/_nitro/runtime/nitro-prerenderer.mjs +9 -12
  110. package/dist/presets/_nitro/runtime/service-worker.d.mts +1 -0
  111. package/dist/presets/_nitro/runtime/service-worker.mjs +14 -11
  112. package/dist/presets/aws-amplify/runtime/aws-amplify.d.mts +1 -0
  113. package/dist/presets/aws-amplify/runtime/aws-amplify.mjs +6 -5
  114. package/dist/presets/aws-lambda/runtime/_utils.d.mts +8 -13
  115. package/dist/presets/aws-lambda/runtime/_utils.mjs +95 -81
  116. package/dist/presets/aws-lambda/runtime/aws-lambda-streaming.d.mts +1 -2
  117. package/dist/presets/aws-lambda/runtime/aws-lambda-streaming.mjs +25 -27
  118. package/dist/presets/aws-lambda/runtime/aws-lambda.mjs +7 -7
  119. package/dist/presets/azure/runtime/_utils.mjs +43 -43
  120. package/dist/presets/azure/runtime/azure-swa.d.mts +2 -2
  121. package/dist/presets/azure/runtime/azure-swa.mjs +22 -21
  122. package/dist/presets/bun/runtime/bun.d.mts +1 -0
  123. package/dist/presets/bun/runtime/bun.mjs +18 -17
  124. package/dist/presets/cloudflare/runtime/_module-handler.d.mts +10 -7
  125. package/dist/presets/cloudflare/runtime/_module-handler.mjs +79 -87
  126. package/dist/presets/cloudflare/runtime/cloudflare-durable.d.mts +6 -14
  127. package/dist/presets/cloudflare/runtime/cloudflare-durable.mjs +55 -61
  128. package/dist/presets/cloudflare/runtime/cloudflare-module.d.mts +1 -7
  129. package/dist/presets/cloudflare/runtime/cloudflare-module.mjs +12 -14
  130. package/dist/presets/cloudflare/runtime/cloudflare-pages.d.mts +12 -12
  131. package/dist/presets/cloudflare/runtime/cloudflare-pages.mjs +35 -39
  132. package/dist/presets/cloudflare/runtime/plugin.dev.d.mts +2 -2
  133. package/dist/presets/cloudflare/runtime/plugin.dev.mjs +89 -82
  134. package/dist/presets/cloudflare/runtime/shims/workers.dev.mjs +27 -0
  135. package/dist/presets/deno/runtime/deno-deploy.d.mts +1 -0
  136. package/dist/presets/deno/runtime/deno-deploy.mjs +12 -11
  137. package/dist/presets/deno/runtime/deno-server.d.mts +2 -1
  138. package/dist/presets/deno/runtime/deno-server.mjs +31 -39
  139. package/dist/presets/netlify/runtime/netlify-edge.d.mts +2 -1
  140. package/dist/presets/netlify/runtime/netlify-edge.mjs +14 -11
  141. package/dist/presets/netlify/runtime/netlify.mjs +11 -17
  142. package/dist/presets/node/runtime/node-cluster.mjs +49 -48
  143. package/dist/presets/node/runtime/node-middleware.d.mts +3 -2
  144. package/dist/presets/node/runtime/node-middleware.mjs +7 -6
  145. package/dist/presets/node/runtime/node-server.mjs +34 -24
  146. package/dist/presets/standard/runtime/server.d.mts +1 -3
  147. package/dist/presets/standard/runtime/server.mjs +1 -3
  148. package/dist/presets/stormkit/runtime/stormkit.d.mts +13 -13
  149. package/dist/presets/stormkit/runtime/stormkit.mjs +20 -20
  150. package/dist/presets/vercel/runtime/vercel.d.mts +3 -3
  151. package/dist/presets/vercel/runtime/vercel.mjs +15 -14
  152. package/dist/presets/winterjs/runtime/winterjs.d.mts +2 -0
  153. package/dist/presets/winterjs/runtime/winterjs.mjs +72 -68
  154. package/dist/presets/zeabur/runtime/zeabur.d.mts +1 -1
  155. package/dist/runtime/index.d.mts +13 -2
  156. package/dist/runtime/index.mjs +13 -8
  157. package/dist/runtime/internal/app.d.mts +3 -1
  158. package/dist/runtime/internal/app.mjs +179 -140
  159. package/dist/runtime/internal/cache.d.mts +10 -4
  160. package/dist/runtime/internal/cache.mjs +247 -216
  161. package/dist/runtime/internal/context.d.mts +9 -9
  162. package/dist/runtime/internal/context.mjs +17 -10
  163. package/dist/runtime/internal/database.mjs +8 -10
  164. package/dist/runtime/internal/error/dev.d.mts +5 -4
  165. package/dist/runtime/internal/error/dev.mjs +106 -105
  166. package/dist/runtime/internal/error/prod.d.mts +5 -4
  167. package/dist/runtime/internal/error/prod.mjs +51 -59
  168. package/dist/runtime/internal/error/utils.d.mts +4 -4
  169. package/dist/runtime/internal/error/utils.mjs +1 -1
  170. package/dist/runtime/internal/index.d.mts +2 -0
  171. package/dist/runtime/internal/index.mjs +2 -0
  172. package/dist/runtime/internal/lib/http-graceful-shutdown.d.mts +14 -14
  173. package/dist/runtime/internal/lib/http-graceful-shutdown.mjs +248 -209
  174. package/dist/runtime/internal/meta.d.mts +1 -1
  175. package/dist/runtime/internal/meta.mjs +1 -1
  176. package/dist/runtime/internal/plugin.d.mts +2 -2
  177. package/dist/runtime/internal/plugin.mjs +1 -1
  178. package/dist/runtime/internal/renderer.mjs +40 -32
  179. package/dist/runtime/internal/route-rules.d.mts +8 -8
  180. package/dist/runtime/internal/route-rules.mjs +53 -51
  181. package/dist/runtime/internal/routes/dev-tasks.d.mts +1 -2
  182. package/dist/runtime/internal/routes/dev-tasks.mjs +15 -17
  183. package/dist/runtime/internal/routes/openapi.d.mts +1 -0
  184. package/dist/runtime/internal/routes/openapi.mjs +81 -87
  185. package/dist/runtime/internal/routes/renderer-template.d.mts +1 -1
  186. package/dist/runtime/internal/routes/renderer-template.dev.d.mts +1 -2
  187. package/dist/runtime/internal/routes/renderer-template.dev.mjs +10 -17
  188. package/dist/runtime/internal/routes/renderer-template.mjs +1 -1
  189. package/dist/runtime/internal/routes/scalar.d.mts +1 -0
  190. package/dist/runtime/internal/routes/scalar.mjs +22 -23
  191. package/dist/runtime/internal/routes/swagger.d.mts +1 -0
  192. package/dist/runtime/internal/routes/swagger.mjs +9 -11
  193. package/dist/runtime/internal/runtime-config.d.mts +4 -4
  194. package/dist/runtime/internal/runtime-config.mjs +40 -35
  195. package/dist/runtime/internal/shutdown.d.mts +2 -8
  196. package/dist/runtime/internal/shutdown.mjs +32 -30
  197. package/dist/runtime/internal/static.mjs +63 -74
  198. package/dist/runtime/internal/storage.mjs +2 -2
  199. package/dist/runtime/internal/task.d.mts +6 -6
  200. package/dist/runtime/internal/task.mjs +56 -60
  201. package/dist/runtime/internal/utils.d.mts +1 -1
  202. package/dist/runtime/internal/utils.mjs +4 -10
  203. package/dist/types/index.d.mts +2594 -2286
  204. package/dist/types/index.mjs +1 -1
  205. package/dist/vite.d.mts +56 -47
  206. package/dist/vite.mjs +55 -68
  207. package/lib/index.d.mts +1 -0
  208. package/lib/index.mjs +1 -0
  209. package/lib/runtime/meta.mjs +35 -0
  210. package/package.json +49 -50
  211. package/dist/_chunks/app.mjs +0 -19797
  212. package/dist/_chunks/build.mjs +0 -84
  213. package/dist/_chunks/build2.mjs +0 -318
  214. package/dist/_chunks/build3.mjs +0 -6452
  215. package/dist/_chunks/detect-acorn.mjs +0 -503
  216. package/dist/_chunks/index.mjs +0 -22242
  217. package/dist/_chunks/index2.mjs +0 -297
  218. package/dist/_chunks/index3.mjs +0 -1058
  219. package/dist/_chunks/index4.mjs +0 -1206
  220. package/dist/_chunks/info.mjs +0 -11157
  221. package/dist/_chunks/json5.mjs +0 -68
  222. package/dist/_chunks/jsonc.mjs +0 -51
  223. package/dist/_chunks/pathe.M-eThtNZ.mjs +0 -204
  224. package/dist/_chunks/plugin.mjs +0 -960
  225. package/dist/_chunks/server.mjs +0 -254
  226. package/dist/_chunks/snapshot.mjs +0 -284
  227. package/dist/_chunks/toml.mjs +0 -259
  228. package/dist/_chunks/yaml.mjs +0 -86
  229. package/dist/cli/build.mjs +0 -54
  230. package/dist/cli/common.mjs +0 -13
  231. package/dist/cli/dev.mjs +0 -95
  232. package/dist/cli/index2.mjs +0 -17
  233. package/dist/cli/list.mjs +0 -34
  234. package/dist/cli/prepare.mjs +0 -24
  235. package/dist/cli/run.mjs +0 -60
  236. package/dist/index.d.mts +0 -52
  237. package/dist/index.mjs +0 -55
  238. package/dist/node_modules/klona/dist/index.mjs +0 -81
  239. package/dist/node_modules/klona/full/index.mjs +0 -53
  240. package/dist/node_modules/klona/package.json +0 -74
  241. package/dist/node_modules/std-env/dist/index.mjs +0 -1
  242. package/dist/node_modules/std-env/package.json +0 -46
  243. package/dist/presets.mjs +0 -2460
  244. package/dist/runtime/internal/debug.d.mts +0 -2
  245. package/dist/runtime/internal/debug.mjs +0 -5
  246. package/dist/runtime/internal/empty.d.mts +0 -0
  247. package/dist/runtime/internal/vite/dev-entry.d.mts +0 -1
  248. package/dist/runtime/internal/vite/dev-worker.d.mts +0 -1
  249. package/dist/runtime/internal/vite/prod-setup.d.mts +0 -4
  250. package/dist/runtime/internal/vite/ssr-renderer.d.mts +0 -4
  251. package/lib/runtime-meta.mjs +0 -38
  252. /package/lib/{runtime-meta.d.mts → runtime/meta.d.mts} +0 -0
@@ -0,0 +1,380 @@
1
+ import { FastResponse } from "srvx";
2
+
3
+ //#region src/parser.ts
4
+ function parseTemplate(template) {
5
+ if (!template) return [];
6
+ template = template.replace(/<script\s+server\s*>([\s\S]*?)<\/script>/gi, (_m, code) => `<?js${code}?>`);
7
+ template = template.replace(/{{\s*(.+)\s*}}|{{{\s*(.+)\s*}}}/g, (_m, code) => {
8
+ if (code[0] === "{") return `<?=${code.slice(1, -1).trim()}?>`;
9
+ return `<?=htmlspecialchars(${code.trim()})?>`;
10
+ });
11
+ const tokens = [];
12
+ const re = /<\?(?:js)?(?<equals>=)?(?<value>[\s\S]*?)\?>/g;
13
+ let cursor = 0;
14
+ let match;
15
+ while (match = re.exec(template)) {
16
+ const { equals, value } = match.groups || {};
17
+ const matchStart = match.index;
18
+ const matchEnd = matchStart + match[0].length;
19
+ if (matchStart > cursor) {
20
+ const textContent = template.slice(cursor, matchStart);
21
+ if (textContent) tokens.push({
22
+ type: "text",
23
+ contents: textContent
24
+ });
25
+ }
26
+ if (equals) tokens.push({
27
+ type: "expr",
28
+ contents: value || ""
29
+ });
30
+ else tokens.push({
31
+ type: "code",
32
+ contents: value || ""
33
+ });
34
+ cursor = matchEnd;
35
+ }
36
+ if (cursor < template.length) {
37
+ const remainingText = template.slice(cursor);
38
+ if (remainingText) tokens.push({
39
+ type: "text",
40
+ contents: remainingText
41
+ });
42
+ }
43
+ return tokens;
44
+ }
45
+ /**
46
+ * Check if a template string contains template syntax.
47
+ */
48
+ function hasTemplateSyntax(template) {
49
+ return /(?:<script\s+server\s*>[\s\S]*?<\/script>)|(?:<\?(?:js)?=?[\s\S]*?\?>)|(?:\{\{[\s\S]*?\}\})/i.test(template);
50
+ }
51
+
52
+ //#endregion
53
+ //#region src/_runtime.ts
54
+ function runtimeStream(body) {
55
+ return `const __chunks__ = [];const echo = (chunk) => { __chunks__.push(chunk); };${body};
56
+ function concatStreams(chunks) {
57
+ const encoder = new TextEncoder();
58
+ return new ReadableStream({
59
+ async pull(controller) {
60
+ for (let chunk of chunks) {
61
+ if (typeof chunk === 'function'){
62
+ chunk = chunk();
63
+ }
64
+ if (chunk instanceof Promise) {
65
+ chunk = await chunk;
66
+ if (!chunk) continue;
67
+ if (chunk instanceof Response){
68
+ chunk = chunk.body;
69
+ }
70
+ if (!(chunk instanceof ReadableStream)) {
71
+ controller.enqueue(chunk instanceof Uint8Array ? chunk : encoder.encode(chunk));
72
+ continue;
73
+ }
74
+ }
75
+ if (chunk instanceof ReadableStream) {
76
+ const reader = chunk.getReader();
77
+ while (true) {
78
+ const { value, done } = await reader.read();
79
+ if (done) break;
80
+ controller.enqueue(value);
81
+ }
82
+ reader.releaseLock();
83
+ } else {
84
+ controller.enqueue(chunk instanceof Uint8Array ? chunk : encoder.encode(chunk));
85
+ }
86
+ }
87
+ controller.close();
88
+ },
89
+ });
90
+ }
91
+ return concatStreams(__chunks__);
92
+ `;
93
+ }
94
+ function runtimeText(body) {
95
+ return `const __chunks__ = [];const echo = (chunk) => { __chunks__.push(chunk); };${body};
96
+ let __out__ = "";
97
+ for(let chunk of __chunks__){
98
+ if (typeof chunk === 'function'){
99
+ chunk = chunk();
100
+ }
101
+ if (chunk instanceof Promise){
102
+ chunk = await chunk;
103
+ }
104
+ if (chunk instanceof Response){
105
+ chunk = chunk.body;
106
+ }
107
+ if (chunk instanceof ReadableStream){
108
+ const reader = chunk.getReader();
109
+ while(true){
110
+ const {value, done} = await reader.read();
111
+ if(done) break;
112
+ __out__ += typeof value === "string" ? value : new TextDecoder().decode(value);
113
+ }
114
+ reader.releaseLock();
115
+ } else {
116
+ __out__ += typeof chunk === "string" ? chunk : new TextDecoder().decode(chunk);
117
+ }
118
+ }
119
+ return __out__;
120
+ `;
121
+ }
122
+
123
+ //#endregion
124
+ //#region src/compiler.ts
125
+ /**
126
+ * Compile a template string into a render function.
127
+ *
128
+ * @example
129
+ * ```ts
130
+ * import { compileTemplate } from "rendu";
131
+ *
132
+ * const template = `
133
+ * <h1>{{ title }}</h1>
134
+ * <ul>
135
+ * <? for (const item of items) { ?>
136
+ * <li>{{ item }}</li>
137
+ * <? } ?>
138
+ * </ul>
139
+ * `;
140
+ *
141
+ * const render = compileTemplate(template, { stream: false });
142
+ *
143
+ * const html = await render({ title: "My List", items: ["Item 1", "Item 2", "Item 3"] });
144
+ * console.log(html);
145
+ * // Output:
146
+ * // <h1>My List</h1>
147
+ * // <ul>
148
+ * // <li>Item 1</li>
149
+ * // <li>Item 2</li>
150
+ * // <li>Item 3</li>
151
+ * // </ul>
152
+ * ```
153
+ */
154
+ function compileTemplate(template, opts = {}) {
155
+ const body = compileTemplateToString(template, opts, false);
156
+ const sourcemaps = opts.filename ? `\n//# sourceURL=${opts.filename}` : "";
157
+ try {
158
+ const AsyncFunction = Object.getPrototypeOf(async () => {}).constructor;
159
+ return new AsyncFunction("__context__", body + sourcemaps);
160
+ } catch (error) {
161
+ throw new SyntaxError(`Template syntax error: ${error.message}`, {});
162
+ }
163
+ }
164
+ /**
165
+ * Compile a template string into a render function code string.
166
+ *
167
+ * **Note:** This function is for advanced use cases where you need the generated code as a string.
168
+ */
169
+ function compileTemplateToString(template, opts, asyncWrapper) {
170
+ const parts = [];
171
+ const tokens = parseTemplate(template);
172
+ for (const token of tokens) switch (token.type) {
173
+ case "text":
174
+ if (opts.preserveLines) for (const line of token.contents.split("\n")) parts.push(`echo(${JSON.stringify(line + "\n")})\n`);
175
+ else parts.push(`echo(${JSON.stringify(token.contents)})`);
176
+ break;
177
+ case "expr":
178
+ parts.push(`echo(${token.contents})`);
179
+ break;
180
+ case "code":
181
+ parts.push(token.contents);
182
+ break;
183
+ }
184
+ let body = parts.join(opts.preserveLines ? ";" : "\n");
185
+ body = opts.contextKeys ? `const {${opts.contextKeys.join(",")}}=__context__;${body}` : `with(__context__){${body}}`;
186
+ body = opts.stream === false ? runtimeText(body) : runtimeStream(body);
187
+ return asyncWrapper === false ? body : `(async (__context__) => {${body}})`;
188
+ }
189
+
190
+ //#endregion
191
+ //#region node_modules/.pnpm/cookie-es@2.0.0/node_modules/cookie-es/dist/index.mjs
192
+ function parse(str, options) {
193
+ if (typeof str !== "string") throw new TypeError("argument str must be a string");
194
+ const obj = {};
195
+ const opt = options || {};
196
+ const dec = opt.decode || decode;
197
+ let index = 0;
198
+ while (index < str.length) {
199
+ const eqIdx = str.indexOf("=", index);
200
+ if (eqIdx === -1) break;
201
+ let endIdx = str.indexOf(";", index);
202
+ if (endIdx === -1) endIdx = str.length;
203
+ else if (endIdx < eqIdx) {
204
+ index = str.lastIndexOf(";", eqIdx - 1) + 1;
205
+ continue;
206
+ }
207
+ const key = str.slice(index, eqIdx).trim();
208
+ if (opt?.filter && !opt?.filter(key)) {
209
+ index = endIdx + 1;
210
+ continue;
211
+ }
212
+ if (void 0 === obj[key]) {
213
+ let val = str.slice(eqIdx + 1, endIdx).trim();
214
+ if (val.codePointAt(0) === 34) val = val.slice(1, -1);
215
+ obj[key] = tryDecode(val, dec);
216
+ }
217
+ index = endIdx + 1;
218
+ }
219
+ return obj;
220
+ }
221
+ function decode(str) {
222
+ return str.includes("%") ? decodeURIComponent(str) : str;
223
+ }
224
+ function tryDecode(str, decode2) {
225
+ try {
226
+ return decode2(str);
227
+ } catch {
228
+ return str;
229
+ }
230
+ }
231
+ const fieldContentRegExp = /^[\u0009\u0020-\u007E\u0080-\u00FF]+$/;
232
+ function serialize(name, value, options) {
233
+ const opt = options || {};
234
+ const enc = opt.encode || encodeURIComponent;
235
+ if (typeof enc !== "function") throw new TypeError("option encode is invalid");
236
+ if (!fieldContentRegExp.test(name)) throw new TypeError("argument name is invalid");
237
+ const encodedValue = enc(value);
238
+ if (encodedValue && !fieldContentRegExp.test(encodedValue)) throw new TypeError("argument val is invalid");
239
+ let str = name + "=" + encodedValue;
240
+ if (void 0 !== opt.maxAge && opt.maxAge !== null) {
241
+ const maxAge = opt.maxAge - 0;
242
+ if (Number.isNaN(maxAge) || !Number.isFinite(maxAge)) throw new TypeError("option maxAge is invalid");
243
+ str += "; Max-Age=" + Math.floor(maxAge);
244
+ }
245
+ if (opt.domain) {
246
+ if (!fieldContentRegExp.test(opt.domain)) throw new TypeError("option domain is invalid");
247
+ str += "; Domain=" + opt.domain;
248
+ }
249
+ if (opt.path) {
250
+ if (!fieldContentRegExp.test(opt.path)) throw new TypeError("option path is invalid");
251
+ str += "; Path=" + opt.path;
252
+ }
253
+ if (opt.expires) {
254
+ if (!isDate(opt.expires) || Number.isNaN(opt.expires.valueOf())) throw new TypeError("option expires is invalid");
255
+ str += "; Expires=" + opt.expires.toUTCString();
256
+ }
257
+ if (opt.httpOnly) str += "; HttpOnly";
258
+ if (opt.secure) str += "; Secure";
259
+ if (opt.priority) switch (typeof opt.priority === "string" ? opt.priority.toLowerCase() : opt.priority) {
260
+ case "low":
261
+ str += "; Priority=Low";
262
+ break;
263
+ case "medium":
264
+ str += "; Priority=Medium";
265
+ break;
266
+ case "high":
267
+ str += "; Priority=High";
268
+ break;
269
+ default: throw new TypeError("option priority is invalid");
270
+ }
271
+ if (opt.sameSite) switch (typeof opt.sameSite === "string" ? opt.sameSite.toLowerCase() : opt.sameSite) {
272
+ case true:
273
+ str += "; SameSite=Strict";
274
+ break;
275
+ case "lax":
276
+ str += "; SameSite=Lax";
277
+ break;
278
+ case "strict":
279
+ str += "; SameSite=Strict";
280
+ break;
281
+ case "none":
282
+ str += "; SameSite=None";
283
+ break;
284
+ default: throw new TypeError("option sameSite is invalid");
285
+ }
286
+ if (opt.partitioned) str += "; Partitioned";
287
+ return str;
288
+ }
289
+ function isDate(val) {
290
+ return Object.prototype.toString.call(val) === "[object Date]" || val instanceof Date;
291
+ }
292
+
293
+ //#endregion
294
+ //#region src/render.ts
295
+ /**
296
+ * Renders an HTML template to a Response object.
297
+ *
298
+ * @example
299
+ * ```ts
300
+ * import { compileTemplate, renderToResponse } from "rendu";
301
+ *
302
+ * const render = compileTemplate(template, { stream: true });
303
+ *
304
+ * const response = await renderToResponse(render, { request });
305
+ * ```
306
+ * @param htmlTemplate The compiled HTML template.
307
+ * @param opts Options for rendering.
308
+ * @returns A Response object.
309
+ */
310
+ async function renderToResponse(htmlTemplate, opts) {
311
+ const ctx = createRenderContext(opts);
312
+ const body = await htmlTemplate(ctx);
313
+ if (body instanceof Response) return body;
314
+ return new FastResponse(body, {
315
+ status: ctx.$RESPONSE.status,
316
+ statusText: ctx.$RESPONSE.statusText,
317
+ headers: ctx.$RESPONSE.headers
318
+ });
319
+ }
320
+ const RENDER_CONTEXT_KEYS = [
321
+ "htmlspecialchars",
322
+ "setCookie",
323
+ "redirect",
324
+ "$REQUEST",
325
+ "$METHOD",
326
+ "$URL",
327
+ "$HEADERS",
328
+ "$COOKIES",
329
+ "$RESPONSE"
330
+ ];
331
+ function createRenderContext(options) {
332
+ const url = new URL(options.request?.url || "http://_");
333
+ const response = {
334
+ status: 200,
335
+ statusText: "OK",
336
+ headers: new Headers({ "Content-Type": "text/html ; charset=utf-8" })
337
+ };
338
+ const $COOKIES = lazyCookies(options.request);
339
+ const setCookie = (name, value, sOpts = {}) => {
340
+ response.headers.append("Set-Cookie", serialize(name, value, sOpts));
341
+ };
342
+ const redirect = (to, status = 302) => {
343
+ response.status = status;
344
+ response.headers.set("Location", to);
345
+ };
346
+ return {
347
+ ...options.context,
348
+ htmlspecialchars,
349
+ setCookie,
350
+ redirect,
351
+ $REQUEST: options.request,
352
+ $METHOD: options.request?.method,
353
+ $URL: url,
354
+ $HEADERS: options.request?.headers,
355
+ $COOKIES,
356
+ $RESPONSE: response
357
+ };
358
+ }
359
+ function lazyCookies(req) {
360
+ if (!req) return {};
361
+ let parsed;
362
+ return new Proxy(Object.freeze(Object.create(null)), { get(_, prop) {
363
+ if (typeof prop !== "string") return void 0;
364
+ parsed ??= parse(req.headers.get("cookie") || "");
365
+ return parsed[prop];
366
+ } });
367
+ }
368
+ function htmlspecialchars(s) {
369
+ const htmlSpecialCharsMap = {
370
+ "&": "&amp;",
371
+ "<": "&lt;",
372
+ ">": "&gt;",
373
+ "\"": "&quot;",
374
+ "'": "&#39;"
375
+ };
376
+ return String(s).replace(/[&<>"']/g, (c) => htmlSpecialCharsMap[c] || c);
377
+ }
378
+
379
+ //#endregion
380
+ export { RENDER_CONTEXT_KEYS, compileTemplate, compileTemplateToString, createRenderContext, hasTemplateSyntax, renderToResponse };
@@ -0,0 +1,47 @@
1
+ {
2
+ "name": "rendu",
3
+ "version": "0.0.7",
4
+ "description": "",
5
+ "repository": "h3js/rendu",
6
+ "license": "MIT",
7
+ "sideEffects": false,
8
+ "type": "module",
9
+ "exports": {
10
+ ".": "./dist/index.mjs"
11
+ },
12
+ "types": "./dist/index.d.mts",
13
+ "bin": "./dist/cli.mjs",
14
+ "files": [
15
+ "dist"
16
+ ],
17
+ "scripts": {
18
+ "build": "obuild",
19
+ "dev": "vitest dev",
20
+ "lint": "eslint . && prettier -c .",
21
+ "lint:fix": "automd && eslint . --fix && prettier -w .",
22
+ "prepack": "pnpm build",
23
+ "play": "pnpm rendu playground",
24
+ "release": "pnpm test && pnpm build && changelogen --release && npm publish && git push --follow-tags",
25
+ "rendu": "node ./src/cli.ts",
26
+ "test": "pnpm lint && pnpm test:types && vitest run --coverage",
27
+ "test:types": "tsc --noEmit --skipLibCheck"
28
+ },
29
+ "dependencies": {
30
+ "srvx": "^0.9.1"
31
+ },
32
+ "devDependencies": {
33
+ "@types/node": "^24.9.1",
34
+ "@vitest/coverage-v8": "^4.0.4",
35
+ "automd": "^0.4.2",
36
+ "changelogen": "^0.6.2",
37
+ "cookie-es": "^2.0.0",
38
+ "eslint": "^9.38.0",
39
+ "eslint-config-unjs": "^0.5.0",
40
+ "obuild": "^0.3.0",
41
+ "prettier": "^3.6.2",
42
+ "rendu": "^0.0.6",
43
+ "typescript": "^5.9.3",
44
+ "vitest": "^4.0.4"
45
+ },
46
+ "packageManager": "pnpm@10.19.0"
47
+ }
@@ -1 +1,2 @@
1
1
  import "#nitro-internal-pollyfills";
2
+ export {};
@@ -1,62 +1,68 @@
1
1
  import "#nitro-internal-pollyfills";
2
- import { useNitroApp } from "nitro/runtime";
2
+ import { useNitroApp, useNitroHooks } from "nitro/runtime";
3
3
  import { trapUnhandledNodeErrors } from "nitro/runtime/internal";
4
4
  import { startScheduleRunner } from "nitro/runtime/internal";
5
5
  import { Server } from "node:http";
6
- import nodeCrypto from "node:crypto";
7
6
  import { parentPort, threadId } from "node:worker_threads";
8
7
  import wsAdapter from "crossws/adapters/node";
9
8
  import { toNodeHandler } from "srvx/node";
10
9
  import { getSocketAddress, isSocketSupported } from "get-port-please";
11
- if (!globalThis.crypto) {
12
- globalThis.crypto = nodeCrypto;
13
- }
10
+ // Trap unhandled errors
14
11
  trapUnhandledNodeErrors();
12
+ // Listen for shutdown signal from runner
15
13
  parentPort?.on("message", (msg) => {
16
- if (msg && msg.event === "shutdown") {
17
- shutdown();
18
- }
14
+ if (msg && msg.event === "shutdown") {
15
+ shutdown();
16
+ }
19
17
  });
20
18
  const nitroApp = useNitroApp();
19
+ const nitroHooks = useNitroHooks();
21
20
  const server = new Server(toNodeHandler(nitroApp.fetch));
22
21
  let listener;
23
22
  listen().catch((error) => {
24
- console.error("Dev worker failed to listen:", error);
25
- return shutdown();
23
+ console.error("Dev worker failed to listen:", error);
24
+ return shutdown();
26
25
  });
26
+ // https://crossws.unjs.io/adapters/node
27
27
  if (import.meta._websocket) {
28
- const { handleUpgrade } = wsAdapter(nitroApp.h3App.websocket);
29
- server.on("upgrade", handleUpgrade);
28
+ // @ts-expect-error
29
+ const { handleUpgrade } = wsAdapter(nitroApp.h3App.websocket);
30
+ server.on("upgrade", handleUpgrade);
30
31
  }
32
+ // Scheduled tasks
31
33
  if (import.meta._tasks) {
32
- startScheduleRunner();
34
+ startScheduleRunner();
33
35
  }
36
+ // --- utils ---
34
37
  async function listen() {
35
- const listenAddr = await isSocketSupported() ? getSocketAddress({
36
- name: `nitro-dev-${threadId}`,
37
- pid: true,
38
- random: true
39
- }) : { port: 0, host: "localhost" };
40
- return new Promise((resolve, reject) => {
41
- try {
42
- listener = server.listen(listenAddr, () => {
43
- const address = server.address();
44
- parentPort?.postMessage({
45
- event: "listen",
46
- address: typeof address === "string" ? { socketPath: address } : { host: "localhost", port: address?.port }
47
- });
48
- resolve();
49
- });
50
- } catch (error) {
51
- reject(error);
52
- }
53
- });
38
+ const listenAddr = await isSocketSupported() ? getSocketAddress({
39
+ name: `nitro-dev-${threadId}`,
40
+ pid: true,
41
+ random: true
42
+ }) : {
43
+ port: 0,
44
+ host: "localhost"
45
+ };
46
+ return new Promise((resolve, reject) => {
47
+ try {
48
+ listener = server.listen(listenAddr, () => {
49
+ const address = server.address();
50
+ parentPort?.postMessage({
51
+ event: "listen",
52
+ address: typeof address === "string" ? { socketPath: address } : {
53
+ host: "localhost",
54
+ port: address?.port
55
+ }
56
+ });
57
+ resolve();
58
+ });
59
+ } catch (error) {
60
+ reject(error);
61
+ }
62
+ });
54
63
  }
55
64
  async function shutdown() {
56
- server.closeAllConnections?.();
57
- await Promise.all([
58
- new Promise((resolve) => listener?.close(resolve)),
59
- nitroApp.hooks.callHook("close").catch(console.error)
60
- ]);
61
- parentPort?.postMessage({ event: "exit" });
65
+ server.closeAllConnections?.();
66
+ await Promise.all([new Promise((resolve) => listener?.close(resolve)), nitroHooks.callHook("close")]).catch(console.error);
67
+ parentPort?.postMessage({ event: "exit" });
62
68
  }
@@ -1,3 +1,3 @@
1
1
  import "#nitro-internal-pollyfills";
2
- export declare const appFetch: any;
3
- export declare const closePrerenderer: () => any;
2
+ export declare const appFetch: unknown;
3
+ export declare const closePrerenderer: unknown;
@@ -1,18 +1,15 @@
1
1
  import "#nitro-internal-pollyfills";
2
2
  import consola from "consola";
3
- import { useNitroApp } from "nitro/runtime";
3
+ import { useNitroApp, useNitroHooks } from "nitro/runtime";
4
4
  import { trapUnhandledNodeErrors } from "nitro/runtime/internal";
5
5
  const nitroApp = useNitroApp();
6
- export const appFetch = nitroApp.fetch;
7
- export const closePrerenderer = () => nitroApp.hooks.callHook("close");
8
- nitroApp.hooks.hook("error", (error, context) => {
9
- if (!error.unhandled && error.status >= 500 && context.event?.req?.headers instanceof Headers && context.event.req.headers.get("x-nitro-prerender")) {
10
- consola.error(
11
- `[prerender error]`,
12
- `[${context.event.req.method}]`,
13
- `[${context.event.req.url}]`,
14
- error
15
- );
16
- }
6
+ const nitroHooks = useNitroHooks();
7
+ export const appFetch = nitroApp.request;
8
+ export const closePrerenderer = () => nitroHooks.callHook("close");
9
+ nitroHooks.hook("error", (error, context) => {
10
+ if (!error.unhandled && error.status >= 500 && context.event?.req?.headers instanceof Headers && context.event.req.headers.get("x-nitro-prerender")) {
11
+ consola.error(`[prerender error]`, `[${context.event.req.method}]`, `[${context.event.req.url}]`, error);
12
+ }
17
13
  });
14
+ // Trap unhandled errors
18
15
  trapUnhandledNodeErrors();
@@ -1 +1,2 @@
1
1
  import "#nitro-internal-pollyfills";
2
+ export {};
@@ -2,20 +2,23 @@ import "#nitro-internal-pollyfills";
2
2
  import { useNitroApp } from "nitro/runtime";
3
3
  import { isPublicAssetURL } from "#nitro-internal-virtual/public-assets";
4
4
  const nitroApp = useNitroApp();
5
+ // @ts-expect-error
5
6
  addEventListener("fetch", (event) => {
6
- const url = new URL(event.request.url);
7
- if (isPublicAssetURL(url.pathname) || url.pathname.includes("/_server/")) {
8
- return;
9
- }
10
- const req = event.request;
11
- req.runtime ??= { name: "service-worker" };
12
- req.runtime.serviceWorker ??= { event };
13
- req.waitUntil = event.waitUntil.bind(event);
14
- event.respondWith(nitroApp.fetch(req));
7
+ const url = new URL(event.request.url);
8
+ if (isPublicAssetURL(url.pathname) || url.pathname.includes("/_server/")) {
9
+ return;
10
+ }
11
+ // srvx compatibility
12
+ const req = event.request;
13
+ req.runtime ??= { name: "service-worker" };
14
+ // @ts-expect-error (add to srvx types)
15
+ req.runtime.serviceWorker ??= { event };
16
+ req.waitUntil = event.waitUntil.bind(event);
17
+ event.respondWith(nitroApp.fetch(req));
15
18
  });
16
19
  self.addEventListener("install", () => {
17
- self.skipWaiting();
20
+ self.skipWaiting();
18
21
  });
19
22
  self.addEventListener("activate", (event) => {
20
- event.waitUntil(self.clients.claim());
23
+ event.waitUntil(self.clients.claim());
21
24
  });
@@ -1 +1,2 @@
1
1
  import "#nitro-internal-pollyfills";
2
+ export {};
@@ -4,10 +4,11 @@ import { Server } from "node:http";
4
4
  import { toNodeHandler } from "srvx/node";
5
5
  const nitroApp = useNitroApp();
6
6
  const server = new Server(toNodeHandler(nitroApp.fetch));
7
+ // @ts-ignore
7
8
  server.listen(3e3, (err) => {
8
- if (err) {
9
- console.error(err);
10
- } else {
11
- console.log(`Listening on http://localhost:3000 (AWS Amplify Hosting)`);
12
- }
9
+ if (err) {
10
+ console.error(err);
11
+ } else {
12
+ console.log(`Listening on http://localhost:3000 (AWS Amplify Hosting)`);
13
+ }
13
14
  });
@@ -1,18 +1,13 @@
1
1
  import type { APIGatewayProxyEvent, APIGatewayProxyEventV2 } from "aws-lambda";
2
2
  import type { ServerRequest } from "srvx";
3
+ // Incoming (AWS => Web)
3
4
  export declare function awsRequest(event: APIGatewayProxyEvent | APIGatewayProxyEventV2, context: unknown): ServerRequest;
4
- export declare function awsResponseHeaders(response: Response): {
5
- headers: any;
6
- cookies: string[];
7
- multiValueHeaders: {
8
- "set-cookie": string[];
9
- };
10
- } | {
11
- headers: any;
12
- cookies?: undefined;
13
- multiValueHeaders?: undefined;
14
- };
5
+ // Outgoing (Web => AWS)
6
+ export declare function awsResponseHeaders(response: Response);
7
+ // AWS Lambda proxy integrations requires base64 encoded buffers
8
+ // binaryMediaTypes should be */*
9
+ // see https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-payload-encodings.html
15
10
  export declare function awsResponseBody(response: Response): Promise<{
16
- body: string;
17
- isBase64Encoded?: boolean;
11
+ body: string;
12
+ isBase64Encoded?: boolean;
18
13
  }>;