toiljs 0.0.14 → 0.0.16

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 (225) hide show
  1. package/.babelrc +13 -13
  2. package/.gitattributes +2 -2
  3. package/.github/ISSUE_TEMPLATE/bug_report.md +38 -38
  4. package/.github/ISSUE_TEMPLATE/bug_report.yml +90 -90
  5. package/.github/ISSUE_TEMPLATE/config.yml +8 -8
  6. package/.github/ISSUE_TEMPLATE/feature_request.md +20 -20
  7. package/.github/PULL_REQUEST_TEMPLATE.md +43 -43
  8. package/.github/changelog-config.json +45 -45
  9. package/.github/dependabot.yml +27 -27
  10. package/.github/workflows/ci.yml +191 -191
  11. package/.prettierrc.json +11 -11
  12. package/.vscode/settings.json +9 -9
  13. package/CHANGELOG.md +5 -5
  14. package/LICENSE +187 -187
  15. package/README.md +339 -315
  16. package/as-pect.asconfig.json +34 -34
  17. package/as-pect.config.js +65 -65
  18. package/assets/logo.svg +36 -36
  19. package/build/backend/.tsbuildinfo +1 -1
  20. package/build/cli/.tsbuildinfo +1 -1
  21. package/build/cli/index.js +2926 -191
  22. package/build/client/.tsbuildinfo +1 -1
  23. package/build/client/dev/devtools.d.ts +6 -0
  24. package/build/client/dev/devtools.js +442 -0
  25. package/build/client/dev/error-overlay.d.ts +9 -0
  26. package/build/client/dev/error-overlay.js +19 -4
  27. package/build/client/head/metadata.d.ts +3 -1
  28. package/build/client/head/metadata.js +8 -0
  29. package/build/client/index.d.ts +4 -4
  30. package/build/client/index.js +2 -2
  31. package/build/client/navigation/navigation.d.ts +2 -0
  32. package/build/client/navigation/navigation.js +9 -1
  33. package/build/client/navigation/prefetch.d.ts +1 -0
  34. package/build/client/navigation/prefetch.js +35 -0
  35. package/build/client/routing/Router.js +1 -1
  36. package/build/client/routing/hooks.js +6 -2
  37. package/build/client/routing/loader.d.ts +25 -0
  38. package/build/client/routing/loader.js +53 -7
  39. package/build/client/routing/mount.js +4 -3
  40. package/build/compiler/.tsbuildinfo +1 -1
  41. package/build/compiler/config.d.ts +18 -0
  42. package/build/compiler/config.js +8 -0
  43. package/build/compiler/docs.js +16 -16
  44. package/build/compiler/generate.js +3 -0
  45. package/build/compiler/index.d.ts +2 -2
  46. package/build/compiler/index.js +3 -1
  47. package/build/compiler/plugin.js +156 -0
  48. package/build/compiler/prerender.d.ts +1 -0
  49. package/build/compiler/prerender.js +2 -1
  50. package/build/compiler/seo.d.ts +2 -2
  51. package/build/compiler/seo.js +8 -6
  52. package/build/compiler/ssg.d.ts +5 -0
  53. package/build/compiler/ssg.js +121 -0
  54. package/build/io/.tsbuildinfo +1 -1
  55. package/build/logger/.tsbuildinfo +1 -1
  56. package/build/shared/.tsbuildinfo +1 -1
  57. package/eslint.config.js +48 -48
  58. package/examples/basic/client/404.tsx +11 -11
  59. package/examples/basic/client/components/.gitkeep +1 -1
  60. package/examples/basic/client/global-error.tsx +13 -13
  61. package/examples/basic/client/layout.tsx +25 -25
  62. package/examples/basic/client/public/images/.gitkeep +1 -1
  63. package/examples/basic/client/public/images/logo.svg +36 -36
  64. package/examples/basic/client/public/robots.txt +2 -2
  65. package/examples/basic/client/routes/docs/[...slug].tsx +12 -12
  66. package/examples/basic/client/routes/features/error/error.tsx +16 -16
  67. package/examples/basic/client/routes/features/template/b.tsx +14 -14
  68. package/examples/basic/client/routes/files/[[...slug]].tsx +21 -21
  69. package/examples/basic/client/routes/gallery/layout.tsx +13 -13
  70. package/examples/basic/client/routes/io.tsx +24 -24
  71. package/examples/basic/client/routes/loader-demo/loading.tsx +13 -13
  72. package/examples/basic/client/routes/search.tsx +61 -61
  73. package/examples/basic/client/toil.tsx +5 -5
  74. package/package.json +155 -147
  75. package/presets/eslint.js +88 -88
  76. package/presets/no-uint8array-tostring.js +200 -200
  77. package/presets/prettier.json +18 -18
  78. package/presets/tsconfig.json +37 -37
  79. package/src/backend/index.ts +160 -160
  80. package/src/cli/proc.ts +50 -50
  81. package/src/cli/updates.ts +69 -69
  82. package/src/cli/validate.ts +31 -31
  83. package/src/client/channel/channel.ts +146 -146
  84. package/src/client/components/Form.tsx +65 -65
  85. package/src/client/components/Script.tsx +113 -113
  86. package/src/client/components/Slot.tsx +21 -21
  87. package/src/client/dev/devtools.tsx +973 -0
  88. package/src/client/dev/error-overlay.tsx +30 -4
  89. package/src/client/head/head.ts +167 -167
  90. package/src/client/head/metadata.ts +19 -1
  91. package/src/client/index.ts +19 -9
  92. package/src/client/navigation/NavLink.tsx +86 -86
  93. package/src/client/navigation/navigation.ts +25 -5
  94. package/src/client/navigation/prefetch.ts +169 -130
  95. package/src/client/navigation/scroll.ts +53 -53
  96. package/src/client/routing/Router.tsx +8 -2
  97. package/src/client/routing/action.ts +122 -122
  98. package/src/client/routing/error-boundary.tsx +43 -43
  99. package/src/client/routing/hooks.ts +21 -6
  100. package/src/client/routing/loader.ts +325 -225
  101. package/src/client/routing/match.ts +47 -47
  102. package/src/client/routing/mount.tsx +54 -52
  103. package/src/client/routing/params-context.ts +10 -10
  104. package/src/client/routing/slot-context.ts +7 -7
  105. package/src/client/search/search.ts +189 -189
  106. package/src/client/search/use-page-search.ts +73 -73
  107. package/src/client/types.ts +73 -73
  108. package/src/compiler/config.ts +47 -1
  109. package/src/compiler/docs.ts +228 -228
  110. package/src/compiler/generate.ts +394 -391
  111. package/src/compiler/index.ts +64 -54
  112. package/src/compiler/pages.ts +70 -70
  113. package/src/compiler/plugin.ts +170 -2
  114. package/src/compiler/prerender.ts +5 -1
  115. package/src/compiler/seo.ts +23 -7
  116. package/src/compiler/ssg.ts +162 -0
  117. package/src/io/BinaryReader.ts +340 -340
  118. package/src/io/BinaryWriter.ts +385 -385
  119. package/src/io/FastMap.ts +127 -127
  120. package/src/io/index.ts +11 -11
  121. package/src/io/lengths.ts +14 -14
  122. package/src/io/types.ts +18 -18
  123. package/src/logger/index.ts +22 -22
  124. package/src/server/index.ts +10 -10
  125. package/src/server/main.ts +13 -13
  126. package/src/server/tsconfig.json +4 -4
  127. package/src/shared/index.ts +10 -10
  128. package/std/client/index.d.ts +15 -15
  129. package/std/client/package.json +3 -3
  130. package/test/assembly/example.spec.ts +7 -7
  131. package/test/channel.test.ts +21 -21
  132. package/test/dom/Link.test.tsx +47 -47
  133. package/test/dom/NavLink.test.tsx +37 -37
  134. package/test/dom/error-overlay.test.tsx +44 -44
  135. package/test/dom/loader.test.tsx +121 -121
  136. package/test/dom/navigation.test.ts +59 -59
  137. package/test/dom/revalidate.test.tsx +38 -38
  138. package/test/dom/route-head.test.tsx +78 -78
  139. package/test/dom/router-loading.test.tsx +44 -44
  140. package/test/dom/scroll.test.ts +56 -56
  141. package/test/dom/use-metadata.test.tsx +58 -0
  142. package/test/io.test.ts +93 -93
  143. package/test/navlink.test.ts +28 -28
  144. package/test/placeholder.test.ts +9 -9
  145. package/test/routes.test.ts +76 -76
  146. package/test/seo.test.ts +175 -164
  147. package/test/slot-layouts.test.ts +69 -69
  148. package/test/ssg.test.ts +36 -0
  149. package/test/update.test.ts +44 -44
  150. package/test/validate.test.ts +42 -42
  151. package/toil-routes.d.ts +7 -0
  152. package/toilconfig.json +30 -30
  153. package/tsconfig.backend.json +13 -13
  154. package/tsconfig.base.json +35 -35
  155. package/tsconfig.cli.json +13 -13
  156. package/tsconfig.client.json +14 -14
  157. package/tsconfig.compiler.json +13 -13
  158. package/tsconfig.io.json +12 -12
  159. package/tsconfig.json +22 -22
  160. package/tsconfig.logger.json +12 -12
  161. package/tsconfig.server.json +10 -10
  162. package/tsconfig.shared.json +12 -12
  163. package/vitest.config.ts +26 -26
  164. package/.idea/codeStyles/Project.xml +0 -54
  165. package/.idea/codeStyles/codeStyleConfig.xml +0 -5
  166. package/.idea/inspectionProfiles/Project_Default.xml +0 -6
  167. package/.idea/modules.xml +0 -8
  168. package/.idea/prettier.xml +0 -7
  169. package/.idea/toiljs.iml +0 -8
  170. package/.idea/vcs.xml +0 -6
  171. package/.toil/entry.tsx +0 -9
  172. package/.toil/index.html +0 -12
  173. package/.toil/routes.ts +0 -9
  174. package/build/cli/configure.d.ts +0 -16
  175. package/build/cli/configure.js +0 -272
  176. package/build/cli/create.d.ts +0 -16
  177. package/build/cli/create.js +0 -420
  178. package/build/cli/diagnostics.d.ts +0 -55
  179. package/build/cli/diagnostics.js +0 -333
  180. package/build/cli/doctor.d.ts +0 -6
  181. package/build/cli/doctor.js +0 -249
  182. package/build/cli/features.d.ts +0 -25
  183. package/build/cli/features.js +0 -107
  184. package/build/cli/index.d.ts +0 -2
  185. package/build/cli/proc.d.ts +0 -6
  186. package/build/cli/proc.js +0 -31
  187. package/build/cli/ui.d.ts +0 -9
  188. package/build/cli/ui.js +0 -75
  189. package/build/cli/update.d.ts +0 -7
  190. package/build/cli/update.js +0 -117
  191. package/build/cli/updates.d.ts +0 -10
  192. package/build/cli/updates.js +0 -45
  193. package/build/cli/validate.d.ts +0 -4
  194. package/build/cli/validate.js +0 -19
  195. package/build/client/Link.d.ts +0 -8
  196. package/build/client/Link.js +0 -44
  197. package/build/client/NavLink.d.ts +0 -14
  198. package/build/client/NavLink.js +0 -37
  199. package/build/client/Router.d.ts +0 -7
  200. package/build/client/Router.js +0 -55
  201. package/build/client/channel.d.ts +0 -23
  202. package/build/client/channel.js +0 -94
  203. package/build/client/error-boundary.d.ts +0 -16
  204. package/build/client/error-boundary.js +0 -19
  205. package/build/client/head.d.ts +0 -26
  206. package/build/client/head.js +0 -87
  207. package/build/client/hooks.d.ts +0 -17
  208. package/build/client/hooks.js +0 -48
  209. package/build/client/lazy.d.ts +0 -16
  210. package/build/client/lazy.js +0 -53
  211. package/build/client/match.d.ts +0 -2
  212. package/build/client/match.js +0 -32
  213. package/build/client/mount.d.ts +0 -2
  214. package/build/client/mount.js +0 -13
  215. package/build/client/navigation.d.ts +0 -13
  216. package/build/client/navigation.js +0 -97
  217. package/build/client/params-context.d.ts +0 -2
  218. package/build/client/params-context.js +0 -2
  219. package/build/client/prefetch.d.ts +0 -11
  220. package/build/client/prefetch.js +0 -100
  221. package/build/client/runtime.d.ts +0 -31
  222. package/build/client/runtime.js +0 -112
  223. package/build/client/scroll.d.ts +0 -8
  224. package/build/client/scroll.js +0 -36
  225. package/toil-env.d.ts +0 -16
@@ -1,69 +1,69 @@
1
- import fs from 'node:fs';
2
- import os from 'node:os';
3
- import path from 'node:path';
4
-
5
- import { afterEach, describe, expect, it } from 'vitest';
6
-
7
- import { loadConfig } from '../src/compiler/config';
8
- import { generate } from '../src/compiler/generate';
9
-
10
- const roots: string[] = [];
11
- function project(files: Record<string, string>): string {
12
- const root = fs.mkdtempSync(path.join(os.tmpdir(), 'toil-gen-'));
13
- roots.push(root);
14
- for (const [rel, content] of Object.entries(files)) {
15
- const abs = path.join(root, rel);
16
- fs.mkdirSync(path.dirname(abs), { recursive: true });
17
- fs.writeFileSync(abs, content);
18
- }
19
- return root;
20
- }
21
- afterEach(() => {
22
- for (const r of roots.splice(0)) fs.rmSync(r, { recursive: true, force: true });
23
- });
24
-
25
- const COMP = `export default function C() { return null; }\n`;
26
- const LAYOUT = `export default function L({ children }) { return children; }\n`;
27
- const HTML = `<!doctype html><html><head></head><body><div id="root"></div></body></html>\n`;
28
-
29
- describe('generate: parallel-slot layout chains', () => {
30
- it('keeps the parent layout on the full-page route but drops it from the @slot route', async () => {
31
- const root = project({
32
- 'client/public/index.html': HTML,
33
- 'client/routes/gallery/layout.tsx': LAYOUT,
34
- 'client/routes/gallery/index.tsx': COMP,
35
- 'client/routes/gallery/photo/[id].tsx': COMP,
36
- 'client/routes/gallery/@modal/(.)photo/[id].tsx': COMP,
37
- });
38
- const cfg = await loadConfig({ root });
39
- generate(cfg);
40
- const lines = fs.readFileSync(path.join(cfg.toilDir, 'routes.ts'), 'utf8').split('\n');
41
-
42
- // The normal full-page route is wrapped by gallery/layout.
43
- const mainLine = lines.find((l) => l.includes('photo/[id]') && !l.includes('@modal'));
44
- expect(mainLine).toMatch(/gallery\/layout/);
45
-
46
- // The intercepting @modal slot route is rendered INTO gallery/layout's <Slot>, so it must not
47
- // re-include that layout (doing so recurses, the slot rendering itself forever).
48
- const slotLine = lines.find((l) => l.includes('@modal'));
49
- expect(slotLine).toContain('intercept: true');
50
- expect(slotLine).toMatch(/layouts: \[\]/);
51
- expect(slotLine).not.toMatch(/gallery\/layout/);
52
- });
53
-
54
- it('still applies a layout placed inside the @slot subtree', async () => {
55
- const root = project({
56
- 'client/public/index.html': HTML,
57
- 'client/routes/gallery/layout.tsx': LAYOUT,
58
- 'client/routes/gallery/@modal/layout.tsx': LAYOUT,
59
- 'client/routes/gallery/@modal/(.)photo/[id].tsx': COMP,
60
- });
61
- const cfg = await loadConfig({ root });
62
- generate(cfg);
63
- const lines = fs.readFileSync(path.join(cfg.toilDir, 'routes.ts'), 'utf8').split('\n');
64
- const slotLine = lines.find((l) => l.includes('@modal/(.)photo'));
65
- // The slot's own layout (inside @modal) applies; the parent gallery layout does not.
66
- expect(slotLine).toMatch(/@modal\/layout/);
67
- expect(slotLine).not.toMatch(/gallery\/layout/);
68
- });
69
- });
1
+ import fs from 'node:fs';
2
+ import os from 'node:os';
3
+ import path from 'node:path';
4
+
5
+ import { afterEach, describe, expect, it } from 'vitest';
6
+
7
+ import { loadConfig } from '../src/compiler/config';
8
+ import { generate } from '../src/compiler/generate';
9
+
10
+ const roots: string[] = [];
11
+ function project(files: Record<string, string>): string {
12
+ const root = fs.mkdtempSync(path.join(os.tmpdir(), 'toil-gen-'));
13
+ roots.push(root);
14
+ for (const [rel, content] of Object.entries(files)) {
15
+ const abs = path.join(root, rel);
16
+ fs.mkdirSync(path.dirname(abs), { recursive: true });
17
+ fs.writeFileSync(abs, content);
18
+ }
19
+ return root;
20
+ }
21
+ afterEach(() => {
22
+ for (const r of roots.splice(0)) fs.rmSync(r, { recursive: true, force: true });
23
+ });
24
+
25
+ const COMP = `export default function C() { return null; }\n`;
26
+ const LAYOUT = `export default function L({ children }) { return children; }\n`;
27
+ const HTML = `<!doctype html><html><head></head><body><div id="root"></div></body></html>\n`;
28
+
29
+ describe('generate: parallel-slot layout chains', () => {
30
+ it('keeps the parent layout on the full-page route but drops it from the @slot route', async () => {
31
+ const root = project({
32
+ 'client/public/index.html': HTML,
33
+ 'client/routes/gallery/layout.tsx': LAYOUT,
34
+ 'client/routes/gallery/index.tsx': COMP,
35
+ 'client/routes/gallery/photo/[id].tsx': COMP,
36
+ 'client/routes/gallery/@modal/(.)photo/[id].tsx': COMP,
37
+ });
38
+ const cfg = await loadConfig({ root });
39
+ generate(cfg);
40
+ const lines = fs.readFileSync(path.join(cfg.toilDir, 'routes.ts'), 'utf8').split('\n');
41
+
42
+ // The normal full-page route is wrapped by gallery/layout.
43
+ const mainLine = lines.find((l) => l.includes('photo/[id]') && !l.includes('@modal'));
44
+ expect(mainLine).toMatch(/gallery\/layout/);
45
+
46
+ // The intercepting @modal slot route is rendered INTO gallery/layout's <Slot>, so it must not
47
+ // re-include that layout (doing so recurses, the slot rendering itself forever).
48
+ const slotLine = lines.find((l) => l.includes('@modal'));
49
+ expect(slotLine).toContain('intercept: true');
50
+ expect(slotLine).toMatch(/layouts: \[\]/);
51
+ expect(slotLine).not.toMatch(/gallery\/layout/);
52
+ });
53
+
54
+ it('still applies a layout placed inside the @slot subtree', async () => {
55
+ const root = project({
56
+ 'client/public/index.html': HTML,
57
+ 'client/routes/gallery/layout.tsx': LAYOUT,
58
+ 'client/routes/gallery/@modal/layout.tsx': LAYOUT,
59
+ 'client/routes/gallery/@modal/(.)photo/[id].tsx': COMP,
60
+ });
61
+ const cfg = await loadConfig({ root });
62
+ generate(cfg);
63
+ const lines = fs.readFileSync(path.join(cfg.toilDir, 'routes.ts'), 'utf8').split('\n');
64
+ const slotLine = lines.find((l) => l.includes('@modal/(.)photo'));
65
+ // The slot's own layout (inside @modal) applies; the parent gallery layout does not.
66
+ expect(slotLine).toMatch(/@modal\/layout/);
67
+ expect(slotLine).not.toMatch(/gallery\/layout/);
68
+ });
69
+ });
@@ -0,0 +1,36 @@
1
+ import { describe, expect, it } from 'vitest';
2
+
3
+ import { sitemapXml, type SeoConfig } from '../src/compiler/seo';
4
+ import { fillPattern } from '../src/compiler/ssg';
5
+ import { type ScannedRoute } from '../src/compiler/routes';
6
+
7
+ describe('fillPattern', () => {
8
+ it('substitutes :param and *catch-all segments', () => {
9
+ expect(fillPattern('/:a/:b/:c', { a: 'x', b: 'y', c: 'z' })).toBe('/x/y/z');
10
+ expect(fillPattern('/blog/:id', { id: '42' })).toBe('/blog/42');
11
+ expect(fillPattern('/docs/*slug', { slug: ['a', 'b'] })).toBe('/docs/a/b');
12
+ expect(fillPattern('/static', {})).toBe('/static');
13
+ });
14
+ });
15
+
16
+ describe('sitemapXml with SSG paths', () => {
17
+ const seo: SeoConfig = { url: 'https://x.dev' };
18
+ const routes: ScannedRoute[] = [
19
+ { file: 'a', pattern: '/' },
20
+ { file: 'b', pattern: '/about' },
21
+ { file: 'c', pattern: '/blog/:id' },
22
+ ];
23
+
24
+ it('lists static routes plus enumerated SSG URLs, deduped, never the bare pattern', () => {
25
+ const xml = sitemapXml(seo, routes, ['/blog/1', '/blog/2', '/about']);
26
+ expect(xml).toContain('https://x.dev/blog/1');
27
+ expect(xml).toContain('https://x.dev/blog/2');
28
+ expect(xml).toContain('https://x.dev/about');
29
+ expect(xml).not.toContain('/blog/:id'); // dynamic pattern is never listed literally
30
+ expect((xml.match(/<loc>[^<]*\/about<\/loc>/g) ?? []).length).toBe(1); // deduped
31
+ });
32
+
33
+ it('is empty without a base url', () => {
34
+ expect(sitemapXml({}, routes, ['/blog/1'])).toBe('');
35
+ });
36
+ });
@@ -1,44 +1,44 @@
1
- import { describe, expect, it } from 'vitest';
2
-
3
- import { buildRows, classifyBump, parseNcuJson } from '../src/cli/updates';
4
-
5
- describe('classifyBump', () => {
6
- it('classifies major / minor / patch from ranges', () => {
7
- expect(classifyBump('^19.2.6', '^20.0.0')).toBe('major');
8
- expect(classifyBump('^19.2.6', '^19.3.0')).toBe('minor');
9
- expect(classifyBump('^19.2.6', '^19.2.7')).toBe('patch');
10
- expect(classifyBump('1.2.3', '1.2.3')).toBe('other');
11
- });
12
- });
13
-
14
- describe('parseNcuJson', () => {
15
- it('parses the jsonUpgraded object, tolerating surrounding noise', () => {
16
- expect(parseNcuJson('{"react":"^19.3.0","eslint":"^10.4.1"}')).toEqual({
17
- react: '^19.3.0',
18
- eslint: '^10.4.1',
19
- });
20
- expect(parseNcuJson('npx noise\n{"a":"1.0.0"}\nbye')).toEqual({ a: '1.0.0' });
21
- expect(parseNcuJson('{}')).toEqual({});
22
- expect(parseNcuJson('not json')).toEqual({});
23
- });
24
- });
25
-
26
- describe('buildRows', () => {
27
- it('joins ncu output with current ranges and sorts major-first then by name', () => {
28
- const rows = buildRows(
29
- { react: '^20.0.0', eslint: '^10.4.1', vite: '^8.0.15' },
30
- { react: '^19.2.6', eslint: '^10.2.0', vite: '^8.0.14' },
31
- );
32
- expect(rows.map((r) => `${r.name}:${r.bump}`)).toEqual([
33
- 'react:major',
34
- 'eslint:minor',
35
- 'vite:patch',
36
- ]);
37
- expect(rows[0]).toMatchObject({ name: 'react', from: '^19.2.6', to: '^20.0.0' });
38
- });
39
-
40
- it('marks a package missing from current deps with a "?" source', () => {
41
- const rows = buildRows({ foo: '^2.0.0' }, {});
42
- expect(rows[0]).toMatchObject({ name: 'foo', from: '?', to: '^2.0.0', bump: 'major' });
43
- });
44
- });
1
+ import { describe, expect, it } from 'vitest';
2
+
3
+ import { buildRows, classifyBump, parseNcuJson } from '../src/cli/updates';
4
+
5
+ describe('classifyBump', () => {
6
+ it('classifies major / minor / patch from ranges', () => {
7
+ expect(classifyBump('^19.2.6', '^20.0.0')).toBe('major');
8
+ expect(classifyBump('^19.2.6', '^19.3.0')).toBe('minor');
9
+ expect(classifyBump('^19.2.6', '^19.2.7')).toBe('patch');
10
+ expect(classifyBump('1.2.3', '1.2.3')).toBe('other');
11
+ });
12
+ });
13
+
14
+ describe('parseNcuJson', () => {
15
+ it('parses the jsonUpgraded object, tolerating surrounding noise', () => {
16
+ expect(parseNcuJson('{"react":"^19.3.0","eslint":"^10.4.1"}')).toEqual({
17
+ react: '^19.3.0',
18
+ eslint: '^10.4.1',
19
+ });
20
+ expect(parseNcuJson('npx noise\n{"a":"1.0.0"}\nbye')).toEqual({ a: '1.0.0' });
21
+ expect(parseNcuJson('{}')).toEqual({});
22
+ expect(parseNcuJson('not json')).toEqual({});
23
+ });
24
+ });
25
+
26
+ describe('buildRows', () => {
27
+ it('joins ncu output with current ranges and sorts major-first then by name', () => {
28
+ const rows = buildRows(
29
+ { react: '^20.0.0', eslint: '^10.4.1', vite: '^8.0.15' },
30
+ { react: '^19.2.6', eslint: '^10.2.0', vite: '^8.0.14' },
31
+ );
32
+ expect(rows.map((r) => `${r.name}:${r.bump}`)).toEqual([
33
+ 'react:major',
34
+ 'eslint:minor',
35
+ 'vite:patch',
36
+ ]);
37
+ expect(rows[0]).toMatchObject({ name: 'react', from: '^19.2.6', to: '^20.0.0' });
38
+ });
39
+
40
+ it('marks a package missing from current deps with a "?" source', () => {
41
+ const rows = buildRows({ foo: '^2.0.0' }, {});
42
+ expect(rows[0]).toMatchObject({ name: 'foo', from: '?', to: '^2.0.0', bump: 'major' });
43
+ });
44
+ });
@@ -1,42 +1,42 @@
1
- import { describe, expect, it } from 'vitest';
2
-
3
- import { isPackageManager, isValidName, resolveProjectDir } from '../src/cli/validate';
4
-
5
- describe('isValidName', () => {
6
- it('accepts simple and nested names', () => {
7
- expect(isValidName('my-app')).toBe(true);
8
- expect(isValidName('apps/web')).toBe(true);
9
- expect(isValidName('@scope/app')).toBe(true);
10
- });
11
-
12
- it('rejects empty and illegal characters', () => {
13
- expect(typeof isValidName('')).toBe('string');
14
- expect(typeof isValidName(' ')).toBe('string');
15
- expect(typeof isValidName('bad name!')).toBe('string');
16
- });
17
- });
18
-
19
- describe('resolveProjectDir', () => {
20
- it('resolves names inside cwd', () => {
21
- expect(resolveProjectDir('/work', 'app')).toBe('/work/app');
22
- expect(resolveProjectDir('/work', 'a/b')).toBe('/work/a/b');
23
- expect(resolveProjectDir('/work', '.')).toBe('/work');
24
- });
25
-
26
- it('refuses to escape cwd (traversal / absolute)', () => {
27
- expect(resolveProjectDir('/work', '../evil')).toBeNull();
28
- expect(resolveProjectDir('/work', '../../etc')).toBeNull();
29
- expect(resolveProjectDir('/work', '/etc/passwd')).toBeNull();
30
- });
31
- });
32
-
33
- describe('isPackageManager', () => {
34
- it('allowlists known package managers only', () => {
35
- expect(isPackageManager('npm')).toBe(true);
36
- expect(isPackageManager('pnpm')).toBe(true);
37
- expect(isPackageManager('yarn')).toBe(true);
38
- expect(isPackageManager('bun')).toBe(true);
39
- expect(isPackageManager('npm && calc')).toBe(false);
40
- expect(isPackageManager('rm -rf /')).toBe(false);
41
- });
42
- });
1
+ import { describe, expect, it } from 'vitest';
2
+
3
+ import { isPackageManager, isValidName, resolveProjectDir } from '../src/cli/validate';
4
+
5
+ describe('isValidName', () => {
6
+ it('accepts simple and nested names', () => {
7
+ expect(isValidName('my-app')).toBe(true);
8
+ expect(isValidName('apps/web')).toBe(true);
9
+ expect(isValidName('@scope/app')).toBe(true);
10
+ });
11
+
12
+ it('rejects empty and illegal characters', () => {
13
+ expect(typeof isValidName('')).toBe('string');
14
+ expect(typeof isValidName(' ')).toBe('string');
15
+ expect(typeof isValidName('bad name!')).toBe('string');
16
+ });
17
+ });
18
+
19
+ describe('resolveProjectDir', () => {
20
+ it('resolves names inside cwd', () => {
21
+ expect(resolveProjectDir('/work', 'app')).toBe('/work/app');
22
+ expect(resolveProjectDir('/work', 'a/b')).toBe('/work/a/b');
23
+ expect(resolveProjectDir('/work', '.')).toBe('/work');
24
+ });
25
+
26
+ it('refuses to escape cwd (traversal / absolute)', () => {
27
+ expect(resolveProjectDir('/work', '../evil')).toBeNull();
28
+ expect(resolveProjectDir('/work', '../../etc')).toBeNull();
29
+ expect(resolveProjectDir('/work', '/etc/passwd')).toBeNull();
30
+ });
31
+ });
32
+
33
+ describe('isPackageManager', () => {
34
+ it('allowlists known package managers only', () => {
35
+ expect(isPackageManager('npm')).toBe(true);
36
+ expect(isPackageManager('pnpm')).toBe(true);
37
+ expect(isPackageManager('yarn')).toBe(true);
38
+ expect(isPackageManager('bun')).toBe(true);
39
+ expect(isPackageManager('npm && calc')).toBe(false);
40
+ expect(isPackageManager('rm -rf /')).toBe(false);
41
+ });
42
+ });
@@ -0,0 +1,7 @@
1
+ // AUTO-GENERATED by toil, do not edit.
2
+ export {};
3
+ declare module 'toiljs/client' {
4
+ interface Register {
5
+ routePath: string;
6
+ }
7
+ }
package/toilconfig.json CHANGED
@@ -1,30 +1,30 @@
1
- {
2
- "entries": ["src/server/main.ts"],
3
- "targets": {
4
- "release": {
5
- "outFile": "build/server/release.wasm",
6
- "textFile": "build/server/release.wat"
7
- }
8
- },
9
- "options": {
10
- "sourceMap": false,
11
- "optimizeLevel": 3,
12
- "shrinkLevel": 1,
13
- "converge": true,
14
- "noAssert": false,
15
- "enable": [
16
- "sign-extension",
17
- "mutable-globals",
18
- "nontrapping-f2i",
19
- "bulk-memory",
20
- "simd",
21
- "reference-types",
22
- "multi-value"
23
- ],
24
- "runtime": "stub",
25
- "memoryBase": 0,
26
- "initialMemory": 1,
27
- "debug": false,
28
- "trapMode": "allow"
29
- }
30
- }
1
+ {
2
+ "entries": ["src/server/main.ts"],
3
+ "targets": {
4
+ "release": {
5
+ "outFile": "build/server/release.wasm",
6
+ "textFile": "build/server/release.wat"
7
+ }
8
+ },
9
+ "options": {
10
+ "sourceMap": false,
11
+ "optimizeLevel": 3,
12
+ "shrinkLevel": 1,
13
+ "converge": true,
14
+ "noAssert": false,
15
+ "enable": [
16
+ "sign-extension",
17
+ "mutable-globals",
18
+ "nontrapping-f2i",
19
+ "bulk-memory",
20
+ "simd",
21
+ "reference-types",
22
+ "multi-value"
23
+ ],
24
+ "runtime": "stub",
25
+ "memoryBase": 0,
26
+ "initialMemory": 1,
27
+ "debug": false,
28
+ "trapMode": "allow"
29
+ }
30
+ }
@@ -1,13 +1,13 @@
1
- {
2
- "extends": "./tsconfig.base.json",
3
- "compilerOptions": {
4
- "module": "ESNext",
5
- "target": "ESNext",
6
- "moduleResolution": "bundler",
7
- "rootDir": "src/backend",
8
- "outDir": "build/backend",
9
- "types": ["node"],
10
- "tsBuildInfoFile": "build/backend/.tsbuildinfo"
11
- },
12
- "include": ["src/backend/**/*"]
13
- }
1
+ {
2
+ "extends": "./tsconfig.base.json",
3
+ "compilerOptions": {
4
+ "module": "ESNext",
5
+ "target": "ESNext",
6
+ "moduleResolution": "bundler",
7
+ "rootDir": "src/backend",
8
+ "outDir": "build/backend",
9
+ "types": ["node"],
10
+ "tsBuildInfoFile": "build/backend/.tsbuildinfo"
11
+ },
12
+ "include": ["src/backend/**/*"]
13
+ }
@@ -1,35 +1,35 @@
1
- {
2
- "compilerOptions": {
3
- "declaration": true,
4
- "noImplicitAny": true,
5
- "removeComments": true,
6
- "suppressImplicitAnyIndexErrors": false,
7
- "preserveConstEnums": true,
8
- "resolveJsonModule": true,
9
- "skipLibCheck": true,
10
- "sourceMap": false,
11
- "moduleDetection": "force",
12
- "experimentalDecorators": true,
13
- "lib": [
14
- "ESNext",
15
- "DOM",
16
- "DOM.Iterable",
17
- "DOM.AsyncIterable",
18
- "WebWorker",
19
- "WebWorker.AsyncIterable",
20
- "WebWorker.ImportScripts"
21
- ],
22
- "strict": true,
23
- "strictNullChecks": true,
24
- "strictFunctionTypes": true,
25
- "strictBindCallApply": true,
26
- "strictPropertyInitialization": true,
27
- "alwaysStrict": true,
28
- "moduleResolution": "bundler",
29
- "allowJs": true,
30
- "incremental": true,
31
- "allowSyntheticDefaultImports": true,
32
- "esModuleInterop": true
33
- },
34
- "include": ["src/**/*.ts", "src/*", "src/**/*.js", "src/*.ts", "src/*.js", "src/*.cjs"]
35
- }
1
+ {
2
+ "compilerOptions": {
3
+ "declaration": true,
4
+ "noImplicitAny": true,
5
+ "removeComments": true,
6
+ "suppressImplicitAnyIndexErrors": false,
7
+ "preserveConstEnums": true,
8
+ "resolveJsonModule": true,
9
+ "skipLibCheck": true,
10
+ "sourceMap": false,
11
+ "moduleDetection": "force",
12
+ "experimentalDecorators": true,
13
+ "lib": [
14
+ "ESNext",
15
+ "DOM",
16
+ "DOM.Iterable",
17
+ "DOM.AsyncIterable",
18
+ "WebWorker",
19
+ "WebWorker.AsyncIterable",
20
+ "WebWorker.ImportScripts"
21
+ ],
22
+ "strict": true,
23
+ "strictNullChecks": true,
24
+ "strictFunctionTypes": true,
25
+ "strictBindCallApply": true,
26
+ "strictPropertyInitialization": true,
27
+ "alwaysStrict": true,
28
+ "moduleResolution": "bundler",
29
+ "allowJs": true,
30
+ "incremental": true,
31
+ "allowSyntheticDefaultImports": true,
32
+ "esModuleInterop": true
33
+ },
34
+ "include": ["src/**/*.ts", "src/*", "src/**/*.js", "src/*.ts", "src/*.js", "src/*.cjs"]
35
+ }
package/tsconfig.cli.json CHANGED
@@ -1,13 +1,13 @@
1
- {
2
- "extends": "./tsconfig.base.json",
3
- "compilerOptions": {
4
- "module": "ESNext",
5
- "target": "ESNext",
6
- "moduleResolution": "bundler",
7
- "rootDir": "src/cli",
8
- "outDir": "build/cli",
9
- "types": ["node"],
10
- "tsBuildInfoFile": "build/cli/.tsbuildinfo"
11
- },
12
- "include": ["src/cli/**/*"]
13
- }
1
+ {
2
+ "extends": "./tsconfig.base.json",
3
+ "compilerOptions": {
4
+ "module": "ESNext",
5
+ "target": "ESNext",
6
+ "moduleResolution": "bundler",
7
+ "rootDir": "src/cli",
8
+ "outDir": "build/cli",
9
+ "types": ["node"],
10
+ "tsBuildInfoFile": "build/cli/.tsbuildinfo"
11
+ },
12
+ "include": ["src/cli/**/*"]
13
+ }
@@ -1,14 +1,14 @@
1
- {
2
- "extends": "./tsconfig.base.json",
3
- "compilerOptions": {
4
- "module": "ESNext",
5
- "target": "ESNext",
6
- "moduleResolution": "bundler",
7
- "jsx": "react-jsx",
8
- "lib": ["ESNext", "DOM", "DOM.Iterable"],
9
- "rootDir": "src/client",
10
- "outDir": "build/client",
11
- "tsBuildInfoFile": "build/client/.tsbuildinfo"
12
- },
13
- "include": ["src/client/**/*"]
14
- }
1
+ {
2
+ "extends": "./tsconfig.base.json",
3
+ "compilerOptions": {
4
+ "module": "ESNext",
5
+ "target": "ESNext",
6
+ "moduleResolution": "bundler",
7
+ "jsx": "react-jsx",
8
+ "lib": ["ESNext", "DOM", "DOM.Iterable"],
9
+ "rootDir": "src/client",
10
+ "outDir": "build/client",
11
+ "tsBuildInfoFile": "build/client/.tsbuildinfo"
12
+ },
13
+ "include": ["src/client/**/*"]
14
+ }
@@ -1,13 +1,13 @@
1
- {
2
- "extends": "./tsconfig.base.json",
3
- "compilerOptions": {
4
- "module": "ESNext",
5
- "target": "ESNext",
6
- "moduleResolution": "bundler",
7
- "rootDir": "src/compiler",
8
- "outDir": "build/compiler",
9
- "types": ["node"],
10
- "tsBuildInfoFile": "build/compiler/.tsbuildinfo"
11
- },
12
- "include": ["src/compiler/**/*"]
13
- }
1
+ {
2
+ "extends": "./tsconfig.base.json",
3
+ "compilerOptions": {
4
+ "module": "ESNext",
5
+ "target": "ESNext",
6
+ "moduleResolution": "bundler",
7
+ "rootDir": "src/compiler",
8
+ "outDir": "build/compiler",
9
+ "types": ["node"],
10
+ "tsBuildInfoFile": "build/compiler/.tsbuildinfo"
11
+ },
12
+ "include": ["src/compiler/**/*"]
13
+ }
package/tsconfig.io.json CHANGED
@@ -1,12 +1,12 @@
1
- {
2
- "extends": "./tsconfig.base.json",
3
- "compilerOptions": {
4
- "module": "ESNext",
5
- "target": "ESNext",
6
- "moduleResolution": "bundler",
7
- "rootDir": "src/io",
8
- "outDir": "build/io",
9
- "tsBuildInfoFile": "build/io/.tsbuildinfo"
10
- },
11
- "include": ["src/io/**/*"]
12
- }
1
+ {
2
+ "extends": "./tsconfig.base.json",
3
+ "compilerOptions": {
4
+ "module": "ESNext",
5
+ "target": "ESNext",
6
+ "moduleResolution": "bundler",
7
+ "rootDir": "src/io",
8
+ "outDir": "build/io",
9
+ "tsBuildInfoFile": "build/io/.tsbuildinfo"
10
+ },
11
+ "include": ["src/io/**/*"]
12
+ }