reroute-js 0.5.0 → 0.7.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.
Files changed (121) hide show
  1. package/README.md +142 -0
  2. package/_/basic/package.json +4 -4
  3. package/_/basic/src/client/index.html +1 -1
  4. package/_/basic/src/index.ts +1 -3
  5. package/_/blog/package.json +4 -4
  6. package/_/blog/src/client/App.tsx +1 -0
  7. package/_/blog/src/client/index.html +1 -1
  8. package/_/blog/src/client/routes/blog/content/1-hello-world.tsx +1 -1
  9. package/_/blog/src/client/routes/blog/content/3-markdown-with-syntax-highlighting.mdx +213 -0
  10. package/_/blog/src/client/routes/blog/content/4-content-in-markdown.md +143 -0
  11. package/_/blog/src/client/routes/blog/content/5-mdx-with-components.mdx +267 -0
  12. package/_/blog/src/client/routes/blog/index.tsx +59 -0
  13. package/_/blog/src/client/routes/client.tsx +564 -0
  14. package/_/blog/src/client/routes/docs.tsx +670 -0
  15. package/_/blog/src/client/routes/index.tsx +37 -0
  16. package/_/blog/src/client/routes/markdown-demo.md +169 -0
  17. package/_/blog/src/client/routes/markdown.tsx +160 -0
  18. package/_/blog/src/index.ts +5 -2
  19. package/_/store/package.json +4 -4
  20. package/_/store/src/client/index.html +1 -1
  21. package/_/store/src/index.ts +1 -1
  22. package/cli/bin.d.ts +1 -1
  23. package/cli/bin.js +1313 -74
  24. package/cli/bin.js.map +13 -7
  25. package/cli/index.d.ts +1 -1
  26. package/cli/index.js +428 -17
  27. package/cli/index.js.map +6 -3
  28. package/cli/src/cli.d.ts +1 -1
  29. package/cli/src/commands/boot.d.ts +1 -1
  30. package/cli/src/commands/build.d.ts +19 -0
  31. package/cli/src/commands/build.d.ts.map +1 -0
  32. package/cli/src/commands/dev.d.ts +18 -0
  33. package/cli/src/commands/dev.d.ts.map +1 -0
  34. package/cli/src/commands/gen.d.ts +1 -1
  35. package/cli/src/commands/gen.d.ts.map +1 -1
  36. package/cli/src/commands/init.d.ts +1 -1
  37. package/cli/src/commands/start.d.ts +19 -0
  38. package/cli/src/commands/start.d.ts.map +1 -0
  39. package/cli/src/libs/index.d.ts +4 -1
  40. package/cli/src/libs/index.d.ts.map +1 -1
  41. package/cli/src/libs/log.d.ts +46 -0
  42. package/cli/src/libs/log.d.ts.map +1 -0
  43. package/cli/src/libs/markdown-processor.d.ts +59 -0
  44. package/cli/src/libs/markdown-processor.d.ts.map +1 -0
  45. package/cli/src/libs/markdown.d.ts +33 -0
  46. package/cli/src/libs/markdown.d.ts.map +1 -0
  47. package/cli/src/libs/tailwind.d.ts +1 -1
  48. package/cli/src/libs/version.d.ts +1 -1
  49. package/core/index.d.ts +1 -1
  50. package/core/index.js +4 -4
  51. package/core/index.js.map +2 -2
  52. package/core/src/bundler/hash.d.ts +1 -1
  53. package/core/src/bundler/index.d.ts +1 -1
  54. package/core/src/bundler/transpile.d.ts +1 -1
  55. package/core/src/content/discovery.d.ts +1 -1
  56. package/core/src/content/index.d.ts +1 -1
  57. package/core/src/content/metadata.d.ts +1 -1
  58. package/core/src/content/registry.d.ts +1 -1
  59. package/core/src/index.d.ts +1 -1
  60. package/core/src/ssr/data.d.ts +1 -1
  61. package/core/src/ssr/index.d.ts +1 -1
  62. package/core/src/ssr/modules.d.ts +1 -1
  63. package/core/src/ssr/render.d.ts +1 -1
  64. package/core/src/ssr/seed.d.ts +1 -1
  65. package/core/src/template/html.d.ts +1 -1
  66. package/core/src/template/index.d.ts +1 -1
  67. package/core/src/types.d.ts +1 -1
  68. package/core/src/utils/cache.d.ts +1 -1
  69. package/core/src/utils/compression.d.ts +1 -1
  70. package/core/src/utils/index.d.ts +1 -1
  71. package/core/src/utils/mime.d.ts +1 -1
  72. package/core/src/utils/path.d.ts +1 -1
  73. package/elysia/index.d.ts +1 -1
  74. package/elysia/index.js +4 -4
  75. package/elysia/index.js.map +2 -2
  76. package/elysia/src/index.d.ts +1 -1
  77. package/elysia/src/libs/http.d.ts +1 -1
  78. package/elysia/src/libs/image.d.ts +1 -1
  79. package/elysia/src/plugin.d.ts +1 -1
  80. package/elysia/src/routes/artifacts.d.ts +1 -1
  81. package/elysia/src/routes/content.d.ts +1 -1
  82. package/elysia/src/routes/dev.d.ts +1 -1
  83. package/elysia/src/routes/image.d.ts +1 -1
  84. package/elysia/src/routes/ssr.d.ts +1 -1
  85. package/elysia/src/routes/static.d.ts +1 -1
  86. package/elysia/src/types.d.ts +1 -1
  87. package/package.json +22 -9
  88. package/react/index.d.ts +1 -1
  89. package/react/index.js +141 -36
  90. package/react/index.js.map +7 -6
  91. package/react/src/components/ClientOnly.d.ts +1 -1
  92. package/react/src/components/ContentRoute.d.ts +1 -1
  93. package/react/src/components/ContentRoute.d.ts.map +1 -1
  94. package/react/src/components/Image.d.ts +1 -1
  95. package/react/src/components/Link.d.ts +1 -1
  96. package/react/src/components/Link.d.ts.map +1 -1
  97. package/react/src/components/Markdown.d.ts +89 -0
  98. package/react/src/components/Markdown.d.ts.map +1 -0
  99. package/react/src/components/Outlet.d.ts +1 -1
  100. package/react/src/components/index.d.ts +2 -1
  101. package/react/src/components/index.d.ts.map +1 -1
  102. package/react/src/hooks/index.d.ts +1 -1
  103. package/react/src/hooks/useContent.d.ts +1 -1
  104. package/react/src/hooks/useData.d.ts +1 -1
  105. package/react/src/hooks/useNavigate.d.ts +1 -1
  106. package/react/src/hooks/useParams.d.ts +1 -1
  107. package/react/src/hooks/useRouter.d.ts +1 -1
  108. package/react/src/hooks/useSearchParams.d.ts +1 -1
  109. package/react/src/index.d.ts +1 -1
  110. package/react/src/providers/ContentProvider.d.ts +1 -1
  111. package/react/src/providers/RerouteProvider.d.ts +1 -1
  112. package/react/src/providers/RouterProvider.d.ts +1 -1
  113. package/react/src/providers/RouterProvider.d.ts.map +1 -1
  114. package/react/src/providers/index.d.ts +1 -1
  115. package/react/src/types/any.d.ts +1 -1
  116. package/react/src/types/index.d.ts +1 -1
  117. package/react/src/types/router.d.ts +1 -1
  118. package/react/src/utils/content.d.ts +1 -1
  119. package/react/src/utils/head.d.ts +1 -1
  120. package/react/src/utils/index.d.ts +1 -1
  121. package/_/blog/src/client/components/Counter.tsx +0 -14
package/cli/index.js CHANGED
@@ -1,5 +1,5 @@
1
1
  /**
2
- * reroute-js v0.5.0
2
+ * reroute-js v0.7.0
3
3
  *
4
4
  * @license MIT
5
5
  * @copyright 2025 stewones <hi@stewan.io>
@@ -7,18 +7,417 @@
7
7
  *
8
8
  * Built with Bun <3
9
9
  */
10
+ import { createRequire } from "node:module";
11
+ var __require = /* @__PURE__ */ createRequire(import.meta.url);
10
12
 
13
+ // packages/cli/src/libs/log.ts
14
+ var PREFIX_COLORS = {
15
+ "[reroute]": "{cyan-fg}",
16
+ "[reroute/gen]": "{bright-blue-fg}",
17
+ "[reroute/content]": "{magenta-fg}",
18
+ "[reroute/markdown]": "{yellow-fg}",
19
+ "[reroute/tailwind]": "{red-fg}",
20
+ "[reroute/mdx]": "{bright-red-fg}",
21
+ "[reroute/boot]": "{bright-yellow-fg}",
22
+ "[reroute/dev]": "{green-fg}",
23
+ "[reroute/start]": "{bright-cyan-fg}",
24
+ "[reroute/build]": "{bright-magenta-fg}",
25
+ "[api]": "{blue-fg}",
26
+ "[app]": "{bright-green-fg}"
27
+ };
28
+ function colorizeLogPrefix(text) {
29
+ let colored = text;
30
+ const sortedPrefixes = Object.entries(PREFIX_COLORS).sort(([a], [b]) => b.length - a.length);
31
+ for (const [prefix, colorTag] of sortedPrefixes) {
32
+ colored = colored.replace(new RegExp(`(${prefix.replace(/[.*+?^${}()|[\]\\]/g, "\\$&")})`, "g"), `${colorTag}$1{/}`);
33
+ }
34
+ return colored;
35
+ }
36
+ function blessedToAnsi(text) {
37
+ return text.replace(/\{cyan-fg\}/g, "\x1B[36m").replace(/\{blue-fg\}/g, "\x1B[34m").replace(/\{magenta-fg\}/g, "\x1B[35m").replace(/\{yellow-fg\}/g, "\x1B[33m").replace(/\{green-fg\}/g, "\x1B[32m").replace(/\{red-fg\}/g, "\x1B[31m").replace(/\{bright-cyan-fg\}/g, "\x1B[96m").replace(/\{bright-blue-fg\}/g, "\x1B[94m").replace(/\{bright-magenta-fg\}/g, "\x1B[95m").replace(/\{bright-yellow-fg\}/g, "\x1B[93m").replace(/\{bright-green-fg\}/g, "\x1B[92m").replace(/\{bright-red-fg\}/g, "\x1B[91m").replace(/\{\/\}/g, "\x1B[0m");
38
+ }
39
+ function colorizeLogPrefixAnsi(text) {
40
+ return blessedToAnsi(colorizeLogPrefix(text));
41
+ }
42
+ function createServerOutputHandler(buffer) {
43
+ return (data) => {
44
+ const text = buffer.current + data.toString();
45
+ const lines = text.split(`
46
+ `);
47
+ buffer.current = lines.pop() || "";
48
+ for (const line of lines) {
49
+ const trimmedLine = line.trim();
50
+ if (!trimmedLine) {
51
+ continue;
52
+ }
53
+ const ansiColored = colorizeLogPrefixAnsi(line);
54
+ console.log(ansiColored);
55
+ }
56
+ };
57
+ }
58
+ // packages/cli/src/libs/markdown.ts
59
+ import { existsSync, readFileSync } from "node:fs";
60
+ import { dirname, join } from "node:path";
61
+ function isPackageAvailable(cwd, packageName) {
62
+ let currentDir = cwd;
63
+ for (let i = 0;i < 5; i++) {
64
+ const packageJsonPath = join(currentDir, "package.json");
65
+ if (existsSync(packageJsonPath)) {
66
+ try {
67
+ const packageJson = JSON.parse(readFileSync(packageJsonPath, "utf-8"));
68
+ const deps = {
69
+ ...packageJson.dependencies,
70
+ ...packageJson.devDependencies,
71
+ ...packageJson.peerDependencies
72
+ };
73
+ if (packageName in deps) {
74
+ return true;
75
+ }
76
+ } catch {}
77
+ }
78
+ const parentDir = dirname(currentDir);
79
+ if (parentDir === currentDir) {
80
+ break;
81
+ }
82
+ currentDir = parentDir;
83
+ }
84
+ return false;
85
+ }
86
+ function isReactMarkdownAvailable(cwd) {
87
+ return isPackageAvailable(cwd, "react-markdown");
88
+ }
89
+ function isShikiAvailable(cwd) {
90
+ return isPackageAvailable(cwd, "@shikijs/rehype");
91
+ }
92
+ function isRemarkGfmAvailable(cwd) {
93
+ return isPackageAvailable(cwd, "remark-gfm");
94
+ }
95
+ function isMdxAvailable(cwd) {
96
+ return isPackageAvailable(cwd, "@mdx-js/mdx");
97
+ }
98
+ function getMarkdownConfig(cwd) {
99
+ const hasReactMarkdown = isReactMarkdownAvailable(cwd);
100
+ const hasShiki = isShikiAvailable(cwd);
101
+ const hasRemarkGfm = isRemarkGfmAvailable(cwd);
102
+ const hasMdx = isMdxAvailable(cwd);
103
+ return {
104
+ hasReactMarkdown,
105
+ hasShiki,
106
+ hasRemarkGfm,
107
+ hasMdx,
108
+ isConfigured: hasReactMarkdown
109
+ };
110
+ }
111
+ function logMarkdownStatus(cwd) {
112
+ const config = getMarkdownConfig(cwd);
113
+ if (config.isConfigured) {
114
+ console.log("[reroute/markdown] Markdown rendering available");
115
+ if (config.hasShiki) {
116
+ console.log("[reroute/markdown] ✓ Shiki syntax highlighting enabled");
117
+ }
118
+ if (config.hasRemarkGfm) {
119
+ console.log("[reroute/markdown] ✓ GitHub Flavored Markdown enabled");
120
+ }
121
+ if (config.hasMdx) {
122
+ console.log("[reroute/markdown] ✓ MDX compiler enabled (JSX in markdown)");
123
+ }
124
+ }
125
+ }
126
+ function checkMarkdownIfConfigured(cwd) {
127
+ try {
128
+ const config = getMarkdownConfig(cwd);
129
+ if (config.isConfigured) {
130
+ logMarkdownStatus(cwd);
131
+ }
132
+ return config.isConfigured;
133
+ } catch (e) {
134
+ console.warn("[reroute/markdown] check failed:", e);
135
+ return false;
136
+ }
137
+ }
138
+ // packages/cli/src/libs/markdown-processor.ts
139
+ import { existsSync as existsSync2, readFileSync as readFileSync2 } from "node:fs";
140
+ import { basename, dirname as dirname2, extname, join as join2 } from "node:path";
141
+ function parseFrontmatter(content) {
142
+ const frontmatterRegex = /^---\s*\n([\s\S]*?)\n---\s*\n([\s\S]*)$/;
143
+ const match = content.match(frontmatterRegex);
144
+ if (!match) {
145
+ return { data: {}, content };
146
+ }
147
+ const [, frontmatterText, markdownContent] = match;
148
+ const data = {};
149
+ const lines = frontmatterText.split(`
150
+ `);
151
+ for (const line of lines) {
152
+ const colonIndex = line.indexOf(":");
153
+ if (colonIndex === -1)
154
+ continue;
155
+ const key = line.slice(0, colonIndex).trim();
156
+ let value = line.slice(colonIndex + 1).trim();
157
+ if (value.startsWith('"') && value.endsWith('"') || value.startsWith("'") && value.endsWith("'")) {
158
+ value = value.slice(1, -1);
159
+ }
160
+ if (value.startsWith("[") && value.endsWith("]")) {
161
+ data[key] = value.slice(1, -1).split(",").map((v) => v.trim().replace(/^["']|["']$/g, ""));
162
+ } else if (value === "true" || value === "false") {
163
+ data[key] = value === "true";
164
+ } else if (!Number.isNaN(Number(value))) {
165
+ data[key] = Number(value);
166
+ } else {
167
+ data[key] = value;
168
+ }
169
+ }
170
+ return { data, content: markdownContent };
171
+ }
172
+ function hasJSXSyntax(content) {
173
+ const jsxPatterns = [
174
+ /<[A-Z][a-zA-Z0-9]*/,
175
+ /import\s+{[^}]+}\s+from/,
176
+ /import\s+\w+\s+from/,
177
+ /export\s+(default|const|function)/,
178
+ /<\w+\s+\w+={/
179
+ ];
180
+ return jsxPatterns.some((pattern) => pattern.test(content));
181
+ }
182
+ function processMarkdownFile(filePath) {
183
+ const rawContent = readFileSync2(filePath, "utf-8");
184
+ const { data, content } = parseFrontmatter(rawContent);
185
+ const hasJSX = hasJSXSyntax(content);
186
+ return {
187
+ frontmatter: data,
188
+ content,
189
+ rawContent,
190
+ hasJSX
191
+ };
192
+ }
193
+ function isMdxAvailable2(cwd) {
194
+ let currentDir = cwd;
195
+ for (let i = 0;i < 5; i++) {
196
+ const packageJsonPath = join2(currentDir, "package.json");
197
+ if (existsSync2(packageJsonPath)) {
198
+ try {
199
+ const packageJson = JSON.parse(readFileSync2(packageJsonPath, "utf-8"));
200
+ const deps = {
201
+ ...packageJson.dependencies,
202
+ ...packageJson.devDependencies,
203
+ ...packageJson.peerDependencies
204
+ };
205
+ if ("@mdx-js/mdx" in deps) {
206
+ return true;
207
+ }
208
+ } catch {}
209
+ }
210
+ const parentDir = dirname2(currentDir);
211
+ if (parentDir === currentDir) {
212
+ break;
213
+ }
214
+ currentDir = parentDir;
215
+ }
216
+ return false;
217
+ }
218
+ async function compileMdx(content, cwd, options = {}) {
219
+ if (!isMdxAvailable2(cwd)) {
220
+ console.warn("[reroute/mdx] @mdx-js/mdx not installed. Install with: bun add @mdx-js/mdx");
221
+ return null;
222
+ }
223
+ try {
224
+ let compile;
225
+ const pathsToTry = [
226
+ join2(cwd, "node_modules/@mdx-js/mdx"),
227
+ join2(cwd, "../node_modules/@mdx-js/mdx"),
228
+ join2(cwd, "../../node_modules/@mdx-js/mdx"),
229
+ join2(cwd, "../../../node_modules/@mdx-js/mdx")
230
+ ];
231
+ for (const mdxPath of pathsToTry) {
232
+ try {
233
+ const mdxModule = __require(mdxPath);
234
+ compile = mdxModule.compile;
235
+ break;
236
+ } catch {}
237
+ }
238
+ if (!compile) {
239
+ console.error("[reroute/mdx] Could not load @mdx-js/mdx from any path");
240
+ return null;
241
+ }
242
+ const { enableGfm = true, theme = "github-dark" } = options;
243
+ const remarkPlugins = [];
244
+ const rehypePlugins = [];
245
+ if (enableGfm) {
246
+ const gfmPathsToTry = [
247
+ join2(cwd, "node_modules/remark-gfm"),
248
+ join2(cwd, "../node_modules/remark-gfm"),
249
+ join2(cwd, "../../node_modules/remark-gfm"),
250
+ join2(cwd, "../../../node_modules/remark-gfm")
251
+ ];
252
+ let remarkGfm = null;
253
+ for (const gfmPath of gfmPathsToTry) {
254
+ try {
255
+ remarkGfm = __require(gfmPath);
256
+ break;
257
+ } catch {}
258
+ }
259
+ if (remarkGfm) {
260
+ remarkPlugins.push(remarkGfm.default || remarkGfm);
261
+ } else {
262
+ console.warn("[reroute/mdx] remark-gfm not available, skipping");
263
+ }
264
+ }
265
+ const shikiPathsToTry = [
266
+ join2(cwd, "node_modules/@shikijs/rehype"),
267
+ join2(cwd, "../node_modules/@shikijs/rehype"),
268
+ join2(cwd, "../../node_modules/@shikijs/rehype"),
269
+ join2(cwd, "../../../node_modules/@shikijs/rehype")
270
+ ];
271
+ let rehypeShiki = null;
272
+ for (const shikiPath of shikiPathsToTry) {
273
+ try {
274
+ rehypeShiki = __require(shikiPath);
275
+ break;
276
+ } catch {}
277
+ }
278
+ if (rehypeShiki) {
279
+ rehypePlugins.push([rehypeShiki.default || rehypeShiki, { theme }]);
280
+ } else {
281
+ console.warn("[reroute/mdx] @shikijs/rehype not available, skipping highlighting");
282
+ }
283
+ const result = await compile(content, {
284
+ remarkPlugins,
285
+ rehypePlugins,
286
+ outputFormat: "program",
287
+ development: false
288
+ });
289
+ return String(result);
290
+ } catch (error) {
291
+ console.error("[reroute/mdx] Compilation failed:", error);
292
+ return null;
293
+ }
294
+ }
295
+ async function generateMarkdownModule(processed, cwd, options = {}) {
296
+ const { frontmatter, content } = processed;
297
+ const { enableGfm = true, theme = "github-dark", isMdx = false } = options;
298
+ const compiledMdx = await compileMdx(content, cwd, { enableGfm, theme });
299
+ if (compiledMdx) {
300
+ return `// Generated from ${isMdx ? "MDX" : "Markdown"} file with @mdx-js/mdx
301
+
302
+ // MDX compiled content with imports and component
303
+ ${compiledMdx}
304
+
305
+ // Add metadata exports
306
+ export const meta = ${JSON.stringify(frontmatter, null, 2)};
307
+
308
+ export const ssr = {
309
+ head: [
310
+ '<meta property="og:type" content="article" />',
311
+ '<meta name="twitter:card" content="summary_large_image" />',
312
+ ],
313
+ };
314
+ `;
315
+ }
316
+ console.warn("[reroute/mdx] MDX compile unavailable; falling back to react-markdown");
317
+ const escapedContent = content.replace(/\\/g, "\\\\").replace(/`/g, "\\`").replace(/\$/g, "\\$");
318
+ const pluginImports = [];
319
+ const remarkPluginsList = [];
320
+ const rehypePluginsList = [];
321
+ if (enableGfm) {
322
+ pluginImports.push("import remarkGfm from 'remark-gfm';");
323
+ remarkPluginsList.push("remarkGfm");
324
+ }
325
+ const pluginImportsStr = pluginImports.length > 0 ? `${pluginImports.join(`
326
+ `)}
327
+ ` : "";
328
+ const remarkPluginsStr = remarkPluginsList.length > 0 ? `remarkPlugins={[${remarkPluginsList.join(", ")}]}` : "";
329
+ const rehypePluginsStr = rehypePluginsList.length > 0 ? `rehypePlugins={[${rehypePluginsList.join(", ")}]}` : "";
330
+ const pluginsPropsStr = [remarkPluginsStr, rehypePluginsStr].filter(Boolean).join(`
331
+ `);
332
+ return `// Generated from markdown file
333
+ import { Markdown } from 'reroute-js/react';
334
+ ${pluginImportsStr}
335
+ const meta = ${JSON.stringify(frontmatter, null, 2)};
336
+
337
+ const ssr = {
338
+ head: [
339
+ '<meta property="og:type" content="article" />',
340
+ '<meta name="twitter:card" content="summary_large_image" />',
341
+ ],
342
+ };
343
+
344
+ const markdownContent = \`${escapedContent}\`;
345
+
346
+ function MarkdownRoute() {
347
+ return (
348
+ <div>
349
+ <Markdown
350
+ ${pluginsPropsStr}
351
+ >
352
+ {markdownContent}
353
+ </Markdown>
354
+ </div>
355
+ );
356
+ }
357
+
358
+ export { meta, ssr };
359
+ export default MarkdownRoute;
360
+ `;
361
+ }
362
+ function generatePlainMarkdownModule(processed) {
363
+ const { frontmatter, content } = processed;
364
+ const escapedContent = content.replace(/\\/g, "\\\\").split(String.fromCharCode(96)).join(`\\${String.fromCharCode(96)}`).split("$").join("\\$");
365
+ return `// Generated from markdown file (plain fallback)
366
+ // No markdown renderer detected; showing raw content.
367
+ export const meta = ${JSON.stringify(frontmatter, null, 2)};
368
+
369
+ export const ssr = {
370
+ head: [
371
+ '<meta property="og:type" content="article" />',
372
+ '<meta name="twitter:card" content="summary_large_image" />',
373
+ ],
374
+ };
375
+
376
+ const markdownContent = \`${escapedContent}\`;
377
+
378
+ export default function MarkdownFallback() {
379
+ return (
380
+ <div style={{ padding: '1rem' }}>
381
+ {meta?.title && (
382
+ <h1 style={{ fontSize: '1.5rem', fontWeight: 600 }}>{meta.title}</h1>
383
+ )}
384
+ <div style={{
385
+ margin: '0.75rem 0',
386
+ padding: '0.75rem',
387
+ borderLeft: '4px solid #eab308',
388
+ background: '#fefce8',
389
+ color: '#713f12'
390
+ }}>
391
+ <strong>Markdown rendering not configured.</strong>
392
+ <div style={{ marginTop: '0.25rem' }}>
393
+ Install: react-markdown (required), remark-gfm (optional), @mdx-js/mdx (optional), @shikijs/rehype (optional).
394
+ </div>
395
+ </div>
396
+ <pre style={{ whiteSpace: 'pre-wrap', lineHeight: 1.6 }}>{markdownContent}</pre>
397
+ </div>
398
+ );
399
+ }
400
+ `;
401
+ }
402
+ function isMarkdownFile(filePath) {
403
+ const ext = extname(filePath).toLowerCase();
404
+ return ext === ".md" || ext === ".mdx";
405
+ }
406
+ function getMarkdownRouteName(filePath) {
407
+ const name = basename(filePath, extname(filePath));
408
+ return name;
409
+ }
11
410
  // packages/cli/src/libs/tailwind.ts
12
411
  import { spawn } from "node:child_process";
13
- import { existsSync, readFileSync } from "node:fs";
14
- import { join } from "node:path";
412
+ import { existsSync as existsSync3, readFileSync as readFileSync3 } from "node:fs";
413
+ import { join as join3 } from "node:path";
15
414
  function isTailwindAvailable(cwd) {
16
- const packageJsonPath = join(cwd, "package.json");
17
- if (!existsSync(packageJsonPath)) {
415
+ const packageJsonPath = join3(cwd, "package.json");
416
+ if (!existsSync3(packageJsonPath)) {
18
417
  return false;
19
418
  }
20
419
  try {
21
- const packageJson = JSON.parse(readFileSync(packageJsonPath, "utf-8"));
420
+ const packageJson = JSON.parse(readFileSync3(packageJsonPath, "utf-8"));
22
421
  const deps = {
23
422
  ...packageJson.dependencies,
24
423
  ...packageJson.devDependencies
@@ -29,17 +428,17 @@ function isTailwindAvailable(cwd) {
29
428
  }
30
429
  }
31
430
  function getTailwindPaths(cwd) {
32
- const input = join(cwd, "src/client/theme.css");
33
- const output = join(cwd, ".reroute/theme.css");
431
+ const input = join3(cwd, "src/client/theme.css");
432
+ const output = join3(cwd, ".reroute/theme.css");
34
433
  return { input, output };
35
434
  }
36
435
  function hasTailwindInput(cwd) {
37
436
  const { input } = getTailwindPaths(cwd);
38
- if (!existsSync(input)) {
437
+ if (!existsSync3(input)) {
39
438
  return false;
40
439
  }
41
440
  try {
42
- const content = readFileSync(input, "utf-8");
441
+ const content = readFileSync3(input, "utf-8");
43
442
  return content.includes("tailwindcss");
44
443
  } catch {
45
444
  return false;
@@ -47,16 +446,16 @@ function hasTailwindInput(cwd) {
47
446
  }
48
447
  function resolveTailwindBin(cwd) {
49
448
  const pathsToTry = [
50
- join(cwd, "../node_modules/.bin/tailwindcss"),
51
- join(cwd, "../../node_modules/.bin/tailwindcss"),
52
- join(cwd, "../../../node_modules/.bin/tailwindcss")
449
+ join3(cwd, "../node_modules/.bin/tailwindcss"),
450
+ join3(cwd, "../../node_modules/.bin/tailwindcss"),
451
+ join3(cwd, "../../../node_modules/.bin/tailwindcss")
53
452
  ];
54
453
  for (const binPath of pathsToTry) {
55
- if (existsSync(binPath)) {
454
+ if (existsSync3(binPath)) {
56
455
  return binPath;
57
456
  }
58
457
  }
59
- return join(cwd, "node_modules/.bin/tailwindcss");
458
+ return join3(cwd, "node_modules/.bin/tailwindcss");
60
459
  }
61
460
  async function buildTailwind(cwd) {
62
461
  const { input, output } = getTailwindPaths(cwd);
@@ -106,7 +505,19 @@ async function buildTailwindIfConfigured(cwd) {
106
505
  }
107
506
  }
108
507
  export {
109
- buildTailwindIfConfigured
508
+ processMarkdownFile,
509
+ logMarkdownStatus,
510
+ isMarkdownFile,
511
+ getMarkdownRouteName,
512
+ getMarkdownConfig,
513
+ generatePlainMarkdownModule,
514
+ generateMarkdownModule,
515
+ createServerOutputHandler,
516
+ colorizeLogPrefixAnsi,
517
+ colorizeLogPrefix,
518
+ checkMarkdownIfConfigured,
519
+ buildTailwindIfConfigured,
520
+ blessedToAnsi
110
521
  };
111
522
 
112
- //# debugId=AE5FEA4DE86FC3C064756E2164756E21
523
+ //# debugId=F20ACEE6ED01450C64756E2164756E21
package/cli/index.js.map CHANGED
@@ -1,10 +1,13 @@
1
1
  {
2
2
  "version": 3,
3
- "sources": ["../../packages/cli/src/libs/tailwind.ts"],
3
+ "sources": ["../../packages/cli/src/libs/log.ts", "../../packages/cli/src/libs/markdown.ts", "../../packages/cli/src/libs/markdown-processor.ts", "../../packages/cli/src/libs/tailwind.ts"],
4
4
  "sourcesContent": [
5
+ "/**\n * Log colorization utilities\n *\n * Provides colorization for log prefixes in both blessed (TUI) and ANSI (console) formats.\n * Consumers can use [api] and [app] prefixes to differentiate their application logs.\n */\n\n/**\n * Map of prefix patterns to blessed color tags\n * Each prefix has a unique color for easy visual distinction\n */\nconst PREFIX_COLORS: Record<string, string> = {\n\t'[reroute]': '{cyan-fg}', // Base reroute logs - cyan\n\t'[reroute/gen]': '{bright-blue-fg}', // Generator - bright blue\n\t'[reroute/content]': '{magenta-fg}', // Content - magenta\n\t'[reroute/markdown]': '{yellow-fg}', // Markdown - yellow\n\t'[reroute/tailwind]': '{red-fg}', // Tailwind - red\n\t'[reroute/mdx]': '{bright-red-fg}', // MDX - bright red\n\t'[reroute/boot]': '{bright-yellow-fg}', // Boot - bright yellow\n\t'[reroute/dev]': '{green-fg}', // Dev command - green\n\t'[reroute/start]': '{bright-cyan-fg}', // Start command - bright cyan\n\t'[reroute/build]': '{bright-magenta-fg}', // Build command - bright magenta\n\t'[api]': '{blue-fg}', // API logs - blue\n\t'[app]': '{bright-green-fg}', // App logs - bright green\n};\n\n/**\n * Colorize log prefixes using blessed color tags (for TUI output)\n *\n * @param text - The log text to colorize\n * @returns Text with colorized prefixes using blessed tags\n */\nexport function colorizeLogPrefix(text: string): string {\n\tlet colored = text;\n\t// Sort prefixes by length (longest first) to match more specific prefixes before general ones\n\t// e.g., [reroute/markdown] should match before [reroute]\n\tconst sortedPrefixes = Object.entries(PREFIX_COLORS).sort(\n\t\t([a], [b]) => b.length - a.length,\n\t);\n\n\tfor (const [prefix, colorTag] of sortedPrefixes) {\n\t\t// Replace prefix with colored version\n\t\tcolored = colored.replace(\n\t\t\tnew RegExp(`(${prefix.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&')})`, 'g'),\n\t\t\t`${colorTag}$1{/}`,\n\t\t);\n\t}\n\n\treturn colored;\n}\n\n/**\n * Convert blessed color tags to ANSI escape codes (for console output)\n *\n * @param text - Text with blessed color tags\n * @returns Text with ANSI escape codes\n */\nexport function blessedToAnsi(text: string): string {\n\treturn (\n\t\ttext\n\t\t\t// Standard colors\n\t\t\t.replace(/\\{cyan-fg\\}/g, '\\x1b[36m')\n\t\t\t.replace(/\\{blue-fg\\}/g, '\\x1b[34m')\n\t\t\t.replace(/\\{magenta-fg\\}/g, '\\x1b[35m')\n\t\t\t.replace(/\\{yellow-fg\\}/g, '\\x1b[33m')\n\t\t\t.replace(/\\{green-fg\\}/g, '\\x1b[32m')\n\t\t\t.replace(/\\{red-fg\\}/g, '\\x1b[31m')\n\t\t\t// Bright colors\n\t\t\t.replace(/\\{bright-cyan-fg\\}/g, '\\x1b[96m')\n\t\t\t.replace(/\\{bright-blue-fg\\}/g, '\\x1b[94m')\n\t\t\t.replace(/\\{bright-magenta-fg\\}/g, '\\x1b[95m')\n\t\t\t.replace(/\\{bright-yellow-fg\\}/g, '\\x1b[93m')\n\t\t\t.replace(/\\{bright-green-fg\\}/g, '\\x1b[92m')\n\t\t\t.replace(/\\{bright-red-fg\\}/g, '\\x1b[91m')\n\t\t\t.replace(/\\{\\/\\}/g, '\\x1b[0m')\n\t);\n}\n\n/**\n * Colorize log prefixes and convert to ANSI codes (for console output)\n *\n * @param text - The log text to colorize\n * @returns Text with colorized prefixes using ANSI escape codes\n */\nexport function colorizeLogPrefixAnsi(text: string): string {\n\treturn blessedToAnsi(colorizeLogPrefix(text));\n}\n\n/**\n * Create a handler function for server output that colorizes logs\n *\n * @param buffer - Reference to a buffer string for incomplete lines\n * @returns Handler function for Buffer data\n */\nexport function createServerOutputHandler(buffer: { current: string }) {\n\treturn (data: Buffer) => {\n\t\tconst text = buffer.current + data.toString();\n\t\tconst lines = text.split('\\n');\n\t\tbuffer.current = lines.pop() || ''; // Keep last incomplete line\n\n\t\tfor (const line of lines) {\n\t\t\t// Skip empty lines\n\t\t\tconst trimmedLine = line.trim();\n\t\t\tif (!trimmedLine) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// Colorize the line (preserves existing prefixes or leaves unprefixed lines as-is)\n\t\t\tconst ansiColored = colorizeLogPrefixAnsi(line);\n\t\t\tconsole.log(ansiColored);\n\t\t}\n\t};\n}\n",
6
+ "/**\n * Markdown integration utility\n *\n * Manages markdown and syntax highlighting packages for the Reroute framework\n */\n\nimport { existsSync, readFileSync } from 'node:fs';\nimport { dirname, join } from 'node:path';\n\n/**\n * Check if a package is available in package.json, searching up to 5 parent directories\n */\nfunction isPackageAvailable(cwd: string, packageName: string): boolean {\n\tlet currentDir = cwd;\n\n\t// Check up to 5 levels up\n\tfor (let i = 0; i < 5; i++) {\n\t\tconst packageJsonPath = join(currentDir, 'package.json');\n\n\t\tif (existsSync(packageJsonPath)) {\n\t\t\ttry {\n\t\t\t\tconst packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8'));\n\t\t\t\tconst deps = {\n\t\t\t\t\t...packageJson.dependencies,\n\t\t\t\t\t...packageJson.devDependencies,\n\t\t\t\t\t...packageJson.peerDependencies,\n\t\t\t\t};\n\n\t\t\t\tif (packageName in deps) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t} catch {\n\t\t\t\t// Continue to parent directory\n\t\t\t}\n\t\t}\n\n\t\tconst parentDir = dirname(currentDir);\n\t\t// Stop if we've reached the root\n\t\tif (parentDir === currentDir) {\n\t\t\tbreak;\n\t\t}\n\t\tcurrentDir = parentDir;\n\t}\n\n\treturn false;\n}\n\n/**\n * Check if react-markdown is available in the project\n */\nfunction isReactMarkdownAvailable(cwd: string): boolean {\n\treturn isPackageAvailable(cwd, 'react-markdown');\n}\n\n/**\n * Check if Shiki is available in the project\n */\nfunction isShikiAvailable(cwd: string): boolean {\n\treturn isPackageAvailable(cwd, '@shikijs/rehype');\n}\n\n/**\n * Check if remark-gfm (GitHub Flavored Markdown) is available\n */\nfunction isRemarkGfmAvailable(cwd: string): boolean {\n\treturn isPackageAvailable(cwd, 'remark-gfm');\n}\n\n/**\n * Check if @mdx-js/mdx is available in the project\n */\nfunction isMdxAvailable(cwd: string): boolean {\n\treturn isPackageAvailable(cwd, '@mdx-js/mdx');\n}\n\n/**\n * Get markdown configuration status for the project\n */\nexport function getMarkdownConfig(cwd: string) {\n\tconst hasReactMarkdown = isReactMarkdownAvailable(cwd);\n\tconst hasShiki = isShikiAvailable(cwd);\n\tconst hasRemarkGfm = isRemarkGfmAvailable(cwd);\n\tconst hasMdx = isMdxAvailable(cwd);\n\n\treturn {\n\t\thasReactMarkdown,\n\t\thasShiki,\n\t\thasRemarkGfm,\n\t\thasMdx,\n\t\tisConfigured: hasReactMarkdown,\n\t};\n}\n\n/**\n * Log markdown configuration status\n */\nexport function logMarkdownStatus(cwd: string): void {\n\tconst config = getMarkdownConfig(cwd);\n\n\tif (config.isConfigured) {\n\t\tconsole.log('[reroute/markdown] Markdown rendering available');\n\t\tif (config.hasShiki) {\n\t\t\tconsole.log('[reroute/markdown] ✓ Shiki syntax highlighting enabled');\n\t\t}\n\t\tif (config.hasRemarkGfm) {\n\t\t\tconsole.log('[reroute/markdown] ✓ GitHub Flavored Markdown enabled');\n\t\t}\n\t\tif (config.hasMdx) {\n\t\t\tconsole.log(\n\t\t\t\t'[reroute/markdown] ✓ MDX compiler enabled (JSX in markdown)',\n\t\t\t);\n\t\t}\n\t}\n}\n\n/**\n * Check if markdown packages are configured and log status\n */\nexport function checkMarkdownIfConfigured(cwd: string): boolean {\n\ttry {\n\t\tconst config = getMarkdownConfig(cwd);\n\t\tif (config.isConfigured) {\n\t\t\tlogMarkdownStatus(cwd);\n\t\t}\n\t\treturn config.isConfigured;\n\t} catch (e) {\n\t\tconsole.warn('[reroute/markdown] check failed:', e);\n\t\treturn false;\n\t}\n}\n",
7
+ "/**\n * Markdown/MDX processor for route files\n *\n * Processes .md and .mdx files with frontmatter support for use in the routing system\n */\n\nimport { existsSync, readFileSync } from 'node:fs';\nimport { basename, dirname, extname, join } from 'node:path';\n\n// biome-ignore lint/suspicious/noExplicitAny: yes\ntype Any = any;\n\nexport interface MarkdownFrontmatter {\n\ttitle?: string;\n\tdescription?: string;\n\tdate?: string;\n\tslug?: string;\n\texcerpt?: string;\n\t[key: string]: Any;\n}\n\nexport interface ProcessedMarkdown {\n\tfrontmatter: MarkdownFrontmatter;\n\tcontent: string;\n\trawContent: string;\n\thasJSX: boolean;\n}\n\n/**\n * Simple frontmatter parser (YAML-like)\n * Extracts metadata from markdown files between --- delimiters\n */\nfunction parseFrontmatter(content: string): {\n\tdata: MarkdownFrontmatter;\n\tcontent: string;\n} {\n\tconst frontmatterRegex = /^---\\s*\\n([\\s\\S]*?)\\n---\\s*\\n([\\s\\S]*)$/;\n\tconst match = content.match(frontmatterRegex);\n\n\tif (!match) {\n\t\treturn { data: {}, content };\n\t}\n\n\tconst [, frontmatterText, markdownContent] = match;\n\tconst data: MarkdownFrontmatter = {};\n\n\t// Parse simple key: value pairs\n\tconst lines = frontmatterText.split('\\n');\n\tfor (const line of lines) {\n\t\tconst colonIndex = line.indexOf(':');\n\t\tif (colonIndex === -1) continue;\n\n\t\tconst key = line.slice(0, colonIndex).trim();\n\t\tlet value = line.slice(colonIndex + 1).trim();\n\n\t\t// Remove quotes if present\n\t\tif (\n\t\t\t(value.startsWith('\"') && value.endsWith('\"')) ||\n\t\t\t(value.startsWith(\"'\") && value.endsWith(\"'\"))\n\t\t) {\n\t\t\tvalue = value.slice(1, -1);\n\t\t}\n\n\t\t// Parse arrays (simple format: [item1, item2])\n\t\tif (value.startsWith('[') && value.endsWith(']')) {\n\t\t\tdata[key] = value\n\t\t\t\t.slice(1, -1)\n\t\t\t\t.split(',')\n\t\t\t\t.map((v) => v.trim().replace(/^[\"']|[\"']$/g, ''));\n\t\t}\n\t\t// Parse booleans\n\t\telse if (value === 'true' || value === 'false') {\n\t\t\tdata[key] = value === 'true';\n\t\t}\n\t\t// Parse numbers\n\t\telse if (!Number.isNaN(Number(value))) {\n\t\t\tdata[key] = Number(value);\n\t\t}\n\t\t// String value\n\t\telse {\n\t\t\tdata[key] = value;\n\t\t}\n\t}\n\n\treturn { data, content: markdownContent };\n}\n\n/**\n * Check if markdown content contains JSX syntax\n */\nfunction hasJSXSyntax(content: string): boolean {\n\t// Simple heuristic: look for JSX-like patterns\n\tconst jsxPatterns = [\n\t\t/<[A-Z][a-zA-Z0-9]*/, // Component tags like <Component>\n\t\t/import\\s+{[^}]+}\\s+from/, // ES6 imports\n\t\t/import\\s+\\w+\\s+from/, // Default imports\n\t\t/export\\s+(default|const|function)/, // Exports\n\t\t/<\\w+\\s+\\w+={/, // Props with curly braces\n\t];\n\n\treturn jsxPatterns.some((pattern) => pattern.test(content));\n}\n\n/**\n * Process a markdown/MDX file and return structured data\n */\nexport function processMarkdownFile(filePath: string): ProcessedMarkdown {\n\tconst rawContent = readFileSync(filePath, 'utf-8');\n\tconst { data, content } = parseFrontmatter(rawContent);\n\tconst hasJSX = hasJSXSyntax(content);\n\n\treturn {\n\t\tfrontmatter: data,\n\t\tcontent,\n\t\trawContent,\n\t\thasJSX,\n\t};\n}\n\n/**\n * Check if @mdx-js/mdx is available in the project\n */\nfunction isMdxAvailable(cwd: string): boolean {\n\tlet currentDir = cwd;\n\n\t// Check up to 5 levels up\n\tfor (let i = 0; i < 5; i++) {\n\t\tconst packageJsonPath = join(currentDir, 'package.json');\n\n\t\tif (existsSync(packageJsonPath)) {\n\t\t\ttry {\n\t\t\t\tconst packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8'));\n\t\t\t\tconst deps = {\n\t\t\t\t\t...packageJson.dependencies,\n\t\t\t\t\t...packageJson.devDependencies,\n\t\t\t\t\t...packageJson.peerDependencies,\n\t\t\t\t};\n\n\t\t\t\tif ('@mdx-js/mdx' in deps) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t} catch {\n\t\t\t\t// Continue to parent directory\n\t\t\t}\n\t\t}\n\n\t\tconst parentDir = dirname(currentDir);\n\t\t// Stop if we've reached the root\n\t\tif (parentDir === currentDir) {\n\t\t\tbreak;\n\t\t}\n\t\tcurrentDir = parentDir;\n\t}\n\n\treturn false;\n}\n\n/**\n * Compile MDX content to JSX using @mdx-js/mdx\n */\nasync function compileMdx(\n\tcontent: string,\n\tcwd: string,\n\toptions: {\n\t\tenableGfm?: boolean;\n\t\ttheme?: string;\n\t} = {},\n): Promise<string | null> {\n\tif (!isMdxAvailable(cwd)) {\n\t\tconsole.warn(\n\t\t\t'[reroute/mdx] @mdx-js/mdx not installed. Install with: bun add @mdx-js/mdx',\n\t\t);\n\t\treturn null;\n\t}\n\n\ttry {\n\t\t// Try to find @mdx-js/mdx in workspace, checking up to 3 levels up\n\t\tlet compile: Any;\n\t\tconst pathsToTry = [\n\t\t\tjoin(cwd, 'node_modules/@mdx-js/mdx'),\n\t\t\tjoin(cwd, '../node_modules/@mdx-js/mdx'),\n\t\t\tjoin(cwd, '../../node_modules/@mdx-js/mdx'),\n\t\t\tjoin(cwd, '../../../node_modules/@mdx-js/mdx'),\n\t\t];\n\n\t\tfor (const mdxPath of pathsToTry) {\n\t\t\ttry {\n\t\t\t\tconst mdxModule = require(mdxPath);\n\t\t\t\tcompile = mdxModule.compile;\n\t\t\t\tbreak;\n\t\t\t} catch {\n\t\t\t\t// Try next path\n\t\t\t}\n\t\t}\n\n\t\tif (!compile) {\n\t\t\tconsole.error('[reroute/mdx] Could not load @mdx-js/mdx from any path');\n\t\t\treturn null;\n\t\t}\n\n\t\tconst { enableGfm = true, theme = 'github-dark' } = options;\n\n\t\tconst remarkPlugins = [] as Any[];\n\t\tconst rehypePlugins = [] as Any[];\n\n\t\t// Add GFM support if requested\n\t\tif (enableGfm) {\n\t\t\tconst gfmPathsToTry = [\n\t\t\t\tjoin(cwd, 'node_modules/remark-gfm'),\n\t\t\t\tjoin(cwd, '../node_modules/remark-gfm'),\n\t\t\t\tjoin(cwd, '../../node_modules/remark-gfm'),\n\t\t\t\tjoin(cwd, '../../../node_modules/remark-gfm'),\n\t\t\t];\n\n\t\t\tlet remarkGfm: Any = null;\n\t\t\tfor (const gfmPath of gfmPathsToTry) {\n\t\t\t\ttry {\n\t\t\t\t\tremarkGfm = require(gfmPath);\n\t\t\t\t\tbreak;\n\t\t\t\t} catch {\n\t\t\t\t\t// Try next path\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (remarkGfm) {\n\t\t\t\tremarkPlugins.push(remarkGfm.default || remarkGfm);\n\t\t\t} else {\n\t\t\t\tconsole.warn('[reroute/mdx] remark-gfm not available, skipping');\n\t\t\t}\n\t\t}\n\n\t\t// Add Shiki highlighting if available (compile-time only, safe for SSR)\n\t\tconst shikiPathsToTry = [\n\t\t\tjoin(cwd, 'node_modules/@shikijs/rehype'),\n\t\t\tjoin(cwd, '../node_modules/@shikijs/rehype'),\n\t\t\tjoin(cwd, '../../node_modules/@shikijs/rehype'),\n\t\t\tjoin(cwd, '../../../node_modules/@shikijs/rehype'),\n\t\t];\n\n\t\tlet rehypeShiki: Any = null;\n\t\tfor (const shikiPath of shikiPathsToTry) {\n\t\t\ttry {\n\t\t\t\trehypeShiki = require(shikiPath);\n\t\t\t\tbreak;\n\t\t\t} catch {\n\t\t\t\t// Try next path\n\t\t\t}\n\t\t}\n\n\t\tif (rehypeShiki) {\n\t\t\trehypePlugins.push([rehypeShiki.default || rehypeShiki, { theme }]);\n\t\t} else {\n\t\t\tconsole.warn(\n\t\t\t\t'[reroute/mdx] @shikijs/rehype not available, skipping highlighting',\n\t\t\t);\n\t\t}\n\n\t\tconst result = await compile(content, {\n\t\t\tremarkPlugins,\n\t\t\trehypePlugins,\n\t\t\toutputFormat: 'program',\n\t\t\tdevelopment: false,\n\t\t});\n\n\t\treturn String(result);\n\t} catch (error) {\n\t\tconsole.error('[reroute/mdx] Compilation failed:', error);\n\t\treturn null;\n\t}\n}\n\n/**\n * Generate a React component wrapper for markdown content\n * This creates a TypeScript module that can be imported as a route\n */\nexport async function generateMarkdownModule(\n\tprocessed: ProcessedMarkdown,\n\tcwd: string,\n\toptions: {\n\t\tenableGfm?: boolean;\n\t\ttheme?: string;\n\t\tisMdx?: boolean;\n\t} = {},\n): Promise<string> {\n\tconst { frontmatter, content } = processed;\n\tconst { enableGfm = true, theme = 'github-dark', isMdx = false } = options;\n\n\t// Prefer MDX engine for both .mdx and .md when available to ensure\n\t// identical SSR/CSR output. MD is a subset of MDX, so compile it too.\n\tconst compiledMdx = await compileMdx(content, cwd, { enableGfm, theme });\n\tif (compiledMdx) {\n\t\t// Already a complete program that exports a default component\n\t\treturn `// Generated from ${isMdx ? 'MDX' : 'Markdown'} file with @mdx-js/mdx\n\n// MDX compiled content with imports and component\n${compiledMdx}\n\n// Add metadata exports\nexport const meta = ${JSON.stringify(frontmatter, null, 2)};\n\nexport const ssr = {\n head: [\n '<meta property=\"og:type\" content=\"article\" />',\n '<meta name=\"twitter:card\" content=\"summary_large_image\" />',\n ],\n};\n`;\n\t}\n\n\tconsole.warn(\n\t\t'[reroute/mdx] MDX compile unavailable; falling back to react-markdown',\n\t);\n\n\t// Regular markdown - wrap with Markdown component\n\tconst escapedContent = content\n\t\t.replace(/\\\\/g, '\\\\\\\\')\n\t\t.replace(/`/g, '\\\\`')\n\t\t.replace(/\\$/g, '\\\\$');\n\n\t// Build plugin imports\n\tconst pluginImports: string[] = [];\n\tconst remarkPluginsList: string[] = [];\n\tconst rehypePluginsList: string[] = [];\n\n\tif (enableGfm) {\n\t\tpluginImports.push(\"import remarkGfm from 'remark-gfm';\");\n\t\tremarkPluginsList.push('remarkGfm');\n\t}\n\n\t// Note: Shiki is async and breaks SSR - skip it\n\n\tconst pluginImportsStr =\n\t\tpluginImports.length > 0 ? `${pluginImports.join('\\n')}\\n` : '';\n\tconst remarkPluginsStr =\n\t\tremarkPluginsList.length > 0\n\t\t\t? `remarkPlugins={[${remarkPluginsList.join(', ')}]}`\n\t\t\t: '';\n\tconst rehypePluginsStr =\n\t\trehypePluginsList.length > 0\n\t\t\t? `rehypePlugins={[${rehypePluginsList.join(', ')}]}`\n\t\t\t: '';\n\tconst pluginsPropsStr = [remarkPluginsStr, rehypePluginsStr]\n\t\t.filter(Boolean)\n\t\t.join('\\n ');\n\n\treturn `// Generated from markdown file\nimport { Markdown } from 'reroute-js/react';\n${pluginImportsStr}\nconst meta = ${JSON.stringify(frontmatter, null, 2)};\n\nconst ssr = {\n head: [\n '<meta property=\"og:type\" content=\"article\" />',\n '<meta name=\"twitter:card\" content=\"summary_large_image\" />',\n ],\n};\n\nconst markdownContent = \\`${escapedContent}\\`;\n\nfunction MarkdownRoute() {\n return (\n <div>\n <Markdown\n ${pluginsPropsStr}\n >\n {markdownContent}\n </Markdown>\n </div>\n );\n}\n\nexport { meta, ssr };\nexport default MarkdownRoute;\n`;\n}\n\n/**\n * Generate a plain fallback TSX module when markdown tooling\n * is not configured. Renders raw markdown content in a <pre> block\n * so routes still work without extra deps.\n */\nexport function generatePlainMarkdownModule(\n\tprocessed: ProcessedMarkdown,\n): string {\n\tconst { frontmatter, content } = processed;\n\tconst escapedContent = content\n\t\t.replace(/\\\\/g, '\\\\\\\\')\n\t\t.split(String.fromCharCode(96))\n\t\t.join(`\\\\${String.fromCharCode(96)}`)\n\t\t.split('$')\n\t\t.join('\\\\$');\n\n\treturn `// Generated from markdown file (plain fallback)\n// No markdown renderer detected; showing raw content.\nexport const meta = ${JSON.stringify(frontmatter, null, 2)};\n\nexport const ssr = {\n head: [\n '<meta property=\"og:type\" content=\"article\" />',\n '<meta name=\"twitter:card\" content=\"summary_large_image\" />',\n ],\n};\n\nconst markdownContent = \\`${escapedContent}\\`;\n\nexport default function MarkdownFallback() {\n return (\n <div style={{ padding: '1rem' }}>\n {meta?.title && (\n <h1 style={{ fontSize: '1.5rem', fontWeight: 600 }}>{meta.title}</h1>\n )}\n <div style={{\n margin: '0.75rem 0',\n padding: '0.75rem',\n borderLeft: '4px solid #eab308',\n background: '#fefce8',\n color: '#713f12'\n }}>\n <strong>Markdown rendering not configured.</strong>\n <div style={{ marginTop: '0.25rem' }}>\n Install: react-markdown (required), remark-gfm (optional), @mdx-js/mdx (optional), @shikijs/rehype (optional).\n </div>\n </div>\n <pre style={{ whiteSpace: 'pre-wrap', lineHeight: 1.6 }}>{markdownContent}</pre>\n </div>\n );\n}\n`;\n}\n\n/**\n * Check if a file is a markdown file\n */\nexport function isMarkdownFile(filePath: string): boolean {\n\tconst ext = extname(filePath).toLowerCase();\n\treturn ext === '.md' || ext === '.mdx';\n}\n\n/**\n * Get the route name from a markdown filename\n * Converts kebab-case to the route path\n */\nexport function getMarkdownRouteName(filePath: string): string {\n\tconst name = basename(filePath, extname(filePath));\n\treturn name;\n}\n",
5
8
  "/**\n * Tailwind CSS v4 integration utility\n *\n * Manages Tailwind CLI integration for the Reroute framework\n */\n\nimport { spawn } from 'node:child_process';\nimport { existsSync, readFileSync } from 'node:fs';\nimport { join } from 'node:path';\n\n/**\n * Check if Tailwind CSS v4 is available in the project\n */\nfunction isTailwindAvailable(cwd: string): boolean {\n\tconst packageJsonPath = join(cwd, 'package.json');\n\tif (!existsSync(packageJsonPath)) {\n\t\treturn false;\n\t}\n\n\ttry {\n\t\tconst packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8'));\n\t\tconst deps = {\n\t\t\t...packageJson.dependencies,\n\t\t\t...packageJson.devDependencies,\n\t\t};\n\n\t\treturn '@tailwindcss/cli' in deps;\n\t} catch {\n\t\treturn false;\n\t}\n}\n\n/**\n * Get the default paths for Tailwind CSS\n */\nfunction getTailwindPaths(cwd: string) {\n\tconst input = join(cwd, 'src/client/theme.css');\n\tconst output = join(cwd, '.reroute/theme.css');\n\n\treturn { input, output };\n}\n\n/**\n * Check if Tailwind input file exists and uses v4 format\n */\nfunction hasTailwindInput(cwd: string): boolean {\n\tconst { input } = getTailwindPaths(cwd);\n\n\tif (!existsSync(input)) {\n\t\treturn false;\n\t}\n\n\ttry {\n\t\tconst content = readFileSync(input, 'utf-8');\n\t\t// v4 uses @import \"tailwindcss\"\n\t\treturn content.includes('tailwindcss');\n\t} catch {\n\t\treturn false;\n\t}\n}\n\nfunction resolveTailwindBin(cwd: string): string {\n\t// Try to find tailwindcss binary, checking up to 3 levels up\n\tconst pathsToTry = [\n\t\tjoin(cwd, '../node_modules/.bin/tailwindcss'),\n\t\tjoin(cwd, '../../node_modules/.bin/tailwindcss'),\n\t\tjoin(cwd, '../../../node_modules/.bin/tailwindcss'),\n\t];\n\n\tfor (const binPath of pathsToTry) {\n\t\tif (existsSync(binPath)) {\n\t\t\treturn binPath;\n\t\t}\n\t}\n\n\t// Fallback to the default location if none found\n\treturn join(cwd, 'node_modules/.bin/tailwindcss');\n}\n\n/**\n * Build Tailwind CSS using the Tailwind CLI\n */\nasync function buildTailwind(cwd: string): Promise<void> {\n\tconst { input, output } = getTailwindPaths(cwd);\n\n\treturn new Promise((resolve, reject) => {\n\t\tconst tailwindBin = resolveTailwindBin(cwd);\n\t\tconst args = ['-i', input, '-o', output];\n\n\t\tconsole.log(`[reroute/tailwind] build start: -i ${input} -o ${output}`);\n\n\t\tconst tailwind = spawn(tailwindBin, args, {\n\t\t\tcwd,\n\t\t\tstdio: 'pipe',\n\t\t});\n\n\t\ttailwind.stdout?.on('data', (data) => {\n\t\t\tconst line = data.toString().trim();\n\t\t\tif (line) {\n\t\t\t\tconsole.log(`[reroute/tailwind] ${line}`);\n\t\t\t}\n\t\t});\n\n\t\ttailwind.stderr?.on('data', (data) => {\n\t\t\tconst line = data.toString().trim();\n\t\t\tif (line) {\n\t\t\t\tconsole.log(`[reroute/tailwind] ${line}`);\n\t\t\t}\n\t\t});\n\n\t\ttailwind.on('close', (code) => {\n\t\t\tif (code === 0) {\n\t\t\t\tconsole.log('[reroute/tailwind] build done');\n\t\t\t\tresolve();\n\t\t\t} else {\n\t\t\t\treject(new Error(`Tailwind CSS build failed with code ${code}`));\n\t\t\t}\n\t\t});\n\n\t\ttailwind.on('error', (error) => {\n\t\t\treject(error);\n\t\t});\n\t});\n}\n\n/**\n * Build Tailwind once if installed and input exists.\n * Returns true if a build was attempted (success or failure), false if skipped.\n */\nexport async function buildTailwindIfConfigured(cwd: string): Promise<boolean> {\n\ttry {\n\t\tif (isTailwindAvailable(cwd) && hasTailwindInput(cwd)) {\n\t\t\tawait buildTailwind(cwd);\n\t\t\treturn true;\n\t\t}\n\t\treturn false;\n\t} catch (e) {\n\t\tconsole.warn('[reroute/tailwind] build failed:', e);\n\t\treturn true;\n\t}\n}\n"
6
9
  ],
7
- "mappings": "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",
8
- "debugId": "AE5FEA4DE86FC3C064756E2164756E21",
10
+ "mappings": "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",
11
+ "debugId": "F20ACEE6ED01450C64756E2164756E21",
9
12
  "names": []
10
13
  }
package/cli/src/cli.d.ts CHANGED
@@ -1,5 +1,5 @@
1
1
  /**
2
- * reroute-js v0.5.0
2
+ * reroute-js v0.7.0
3
3
  *
4
4
  * @license MIT
5
5
  * @copyright 2025 stewones <hi@stewan.io>
@@ -1,5 +1,5 @@
1
1
  /**
2
- * reroute-js v0.5.0
2
+ * reroute-js v0.7.0
3
3
  *
4
4
  * @license MIT
5
5
  * @copyright 2025 stewones <hi@stewan.io>
@@ -0,0 +1,19 @@
1
+ /**
2
+ * reroute-js v0.7.0
3
+ *
4
+ * @license MIT
5
+ * @copyright 2025 stewones <hi@stewan.io>
6
+ * @see https://github.com/stewones/reroute
7
+ *
8
+ * Built with Bun <3
9
+ */
10
+ /**
11
+ * Build command - Build production binary
12
+ *
13
+ * This command runs:
14
+ * 1. `reroute boot` (waits for completion)
15
+ * 2. `reroute gen` (waits for completion, no watch)
16
+ * 3. `bun build src/index.ts --target bun --compile --outfile ./dist/{name}`
17
+ */
18
+ export default function build(args: string[]): Promise<void>;
19
+ //# sourceMappingURL=build.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"build.d.ts","sourceRoot":"","sources":["../../../../../../packages/cli/src/commands/build.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAgEH,wBAA8B,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,iBAiLjD"}
@@ -0,0 +1,18 @@
1
+ /**
2
+ * reroute-js v0.7.0
3
+ *
4
+ * @license MIT
5
+ * @copyright 2025 stewones <hi@stewan.io>
6
+ * @see https://github.com/stewones/reroute
7
+ *
8
+ * Built with Bun <3
9
+ */
10
+ /**
11
+ * Dev command - Start development environment
12
+ *
13
+ * This command runs:
14
+ * 1. `reroute boot` (waits for completion)
15
+ * 2. `reroute gen --watch` and `bun --watch src/index.ts` in parallel with side-by-side logs
16
+ */
17
+ export default function dev(args: string[]): Promise<void>;
18
+ //# sourceMappingURL=dev.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dev.d.ts","sourceRoot":"","sources":["../../../../../../packages/cli/src/commands/dev.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AA0CH,wBAA8B,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,iBAsU/C"}
@@ -1,5 +1,5 @@
1
1
  /**
2
- * reroute-js v0.5.0
2
+ * reroute-js v0.7.0
3
3
  *
4
4
  * @license MIT
5
5
  * @copyright 2025 stewones <hi@stewan.io>
@@ -1 +1 @@
1
- {"version":3,"file":"gen.d.ts","sourceRoot":"","sources":["../../../../../../packages/cli/src/commands/gen.ts"],"names":[],"mappings":";AAmkBA,wBAA8B,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,iBA8F/C"}
1
+ {"version":3,"file":"gen.d.ts","sourceRoot":"","sources":["../../../../../../packages/cli/src/commands/gen.ts"],"names":[],"mappings":";AA0uBA,wBAA8B,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,iBAgG/C"}