@tanstack/cta-engine 0.10.0-alpha.6

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 (272) hide show
  1. package/LICENSE +21 -0
  2. package/dist/add-ons.js +109 -0
  3. package/dist/add.js +127 -0
  4. package/dist/cli.js +112 -0
  5. package/dist/config-file.js +23 -0
  6. package/dist/constants.js +5 -0
  7. package/dist/create-app.js +491 -0
  8. package/dist/custom-add-on.js +254 -0
  9. package/dist/environment.js +119 -0
  10. package/dist/index.js +1 -0
  11. package/dist/mcp.js +211 -0
  12. package/dist/options.js +309 -0
  13. package/dist/package-manager.js +30 -0
  14. package/dist/templates.js +6 -0
  15. package/dist/toolchain.js +6 -0
  16. package/dist/types/add-ons.d.ts +5 -0
  17. package/dist/types/add.d.ts +3 -0
  18. package/dist/types/cli.d.ts +1 -0
  19. package/dist/types/config-file.d.ts +7 -0
  20. package/dist/types/constants.d.ts +6 -0
  21. package/dist/types/create-app.d.ts +6 -0
  22. package/dist/types/custom-add-on.d.ts +3 -0
  23. package/dist/types/environment.d.ts +12 -0
  24. package/dist/types/index.d.ts +1 -0
  25. package/dist/types/mcp.d.ts +1 -0
  26. package/dist/types/options.d.ts +3 -0
  27. package/dist/types/package-manager.d.ts +6 -0
  28. package/dist/types/templates.d.ts +1 -0
  29. package/dist/types/toolchain.d.ts +3 -0
  30. package/dist/types/types.d.ts +107 -0
  31. package/dist/types/utils.d.ts +1 -0
  32. package/dist/types.js +1 -0
  33. package/dist/utils.js +8 -0
  34. package/package.json +49 -0
  35. package/src/add-ons.ts +145 -0
  36. package/src/add.ts +184 -0
  37. package/src/cli.ts +163 -0
  38. package/src/config-file.ts +45 -0
  39. package/src/constants.ts +9 -0
  40. package/src/create-app.ts +791 -0
  41. package/src/custom-add-on.ts +323 -0
  42. package/src/environment.ts +144 -0
  43. package/src/index.ts +1 -0
  44. package/src/mcp.ts +252 -0
  45. package/src/options.ts +359 -0
  46. package/src/package-manager.ts +46 -0
  47. package/src/templates.ts +7 -0
  48. package/src/toolchain.ts +7 -0
  49. package/src/types.ts +119 -0
  50. package/src/utils.ts +10 -0
  51. package/templates/react/add-on/clerk/README.md +3 -0
  52. package/templates/react/add-on/clerk/assets/_dot_env.local.append +2 -0
  53. package/templates/react/add-on/clerk/assets/src/integrations/clerk/header-user.tsx +19 -0
  54. package/templates/react/add-on/clerk/assets/src/integrations/clerk/provider.tsx +18 -0
  55. package/templates/react/add-on/clerk/assets/src/routes/demo.clerk.tsx +20 -0
  56. package/templates/react/add-on/clerk/info.json +13 -0
  57. package/templates/react/add-on/clerk/package.json +5 -0
  58. package/templates/react/add-on/convex/README.md +4 -0
  59. package/templates/react/add-on/convex/assets/_dot_cursorrules.append +93 -0
  60. package/templates/react/add-on/convex/assets/_dot_env.local.append +3 -0
  61. package/templates/react/add-on/convex/assets/convex/products.ts +8 -0
  62. package/templates/react/add-on/convex/assets/convex/schema.ts +10 -0
  63. package/templates/react/add-on/convex/assets/src/integrations/convex/provider.tsx +20 -0
  64. package/templates/react/add-on/convex/assets/src/routes/demo.convex.tsx +33 -0
  65. package/templates/react/add-on/convex/info.json +13 -0
  66. package/templates/react/add-on/convex/package.json +6 -0
  67. package/templates/react/add-on/form/assets/src/components/demo.FormComponents.tsx.ejs +300 -0
  68. package/templates/react/add-on/form/assets/src/hooks/demo.form-context.ts +4 -0
  69. package/templates/react/add-on/form/assets/src/hooks/demo.form.ts +22 -0
  70. package/templates/react/add-on/form/assets/src/routes/demo.form.address.tsx.ejs +213 -0
  71. package/templates/react/add-on/form/assets/src/routes/demo.form.simple.tsx.ejs +77 -0
  72. package/templates/react/add-on/form/info.json +26 -0
  73. package/templates/react/add-on/form/package.json +6 -0
  74. package/templates/react/add-on/module-federation/assets/module-federation.config.js.ejs +31 -0
  75. package/templates/react/add-on/module-federation/assets/src/demo-mf-component.tsx +3 -0
  76. package/templates/react/add-on/module-federation/assets/src/demo-mf-self-contained.tsx +11 -0
  77. package/templates/react/add-on/module-federation/info.json +7 -0
  78. package/templates/react/add-on/module-federation/package.json +5 -0
  79. package/templates/react/add-on/netlify/README.md +11 -0
  80. package/templates/react/add-on/netlify/info.json +7 -0
  81. package/templates/react/add-on/sentry/assets/_dot_cursorrules.append +22 -0
  82. package/templates/react/add-on/sentry/assets/_dot_env.local.append +2 -0
  83. package/templates/react/add-on/sentry/assets/src/app/global-middleware.ts +25 -0
  84. package/templates/react/add-on/sentry/assets/src/routes/demo.sentry.testing.tsx +480 -0
  85. package/templates/react/add-on/sentry/info.json +14 -0
  86. package/templates/react/add-on/sentry/package.json +7 -0
  87. package/templates/react/add-on/shadcn/README.md +7 -0
  88. package/templates/react/add-on/shadcn/assets/_dot_cursorrules.append +7 -0
  89. package/templates/react/add-on/shadcn/assets/components.json +21 -0
  90. package/templates/react/add-on/shadcn/assets/src/lib/utils.ts +6 -0
  91. package/templates/react/add-on/shadcn/assets/src/styles.css +138 -0
  92. package/templates/react/add-on/shadcn/info.json +7 -0
  93. package/templates/react/add-on/shadcn/package.json +9 -0
  94. package/templates/react/add-on/start/assets/_dot_gitignore.append +2 -0
  95. package/templates/react/add-on/start/assets/app.config.ts.ejs +19 -0
  96. package/templates/react/add-on/start/assets/src/api.ts +6 -0
  97. package/templates/react/add-on/start/assets/src/client.tsx +8 -0
  98. package/templates/react/add-on/start/assets/src/router.tsx.ejs +77 -0
  99. package/templates/react/add-on/start/assets/src/routes/api.demo-names.ts +11 -0
  100. package/templates/react/add-on/start/assets/src/routes/demo.start.api-request.tsx.ejs +33 -0
  101. package/templates/react/add-on/start/assets/src/routes/demo.start.server-funcs.tsx +50 -0
  102. package/templates/react/add-on/start/assets/src/ssr.tsx +12 -0
  103. package/templates/react/add-on/start/info.json +18 -0
  104. package/templates/react/add-on/start/package.json +13 -0
  105. package/templates/react/add-on/store/assets/src/lib/demo-store.ts +13 -0
  106. package/templates/react/add-on/store/assets/src/routes/demo.store.tsx.ejs +75 -0
  107. package/templates/react/add-on/store/info.json +13 -0
  108. package/templates/react/add-on/store/package.json +6 -0
  109. package/templates/react/add-on/tRPC/assets/src/integrations/trpc/init.ts +9 -0
  110. package/templates/react/add-on/tRPC/assets/src/integrations/trpc/react.ts +4 -0
  111. package/templates/react/add-on/tRPC/assets/src/integrations/trpc/router.ts +18 -0
  112. package/templates/react/add-on/tRPC/assets/src/routes/api.trpc.$.tsx +16 -0
  113. package/templates/react/add-on/tRPC/info.json +9 -0
  114. package/templates/react/add-on/tRPC/package.json +9 -0
  115. package/templates/react/add-on/table/assets/src/data/demo-table-data.ts +50 -0
  116. package/templates/react/add-on/table/assets/src/routes/demo.table.tsx.ejs +373 -0
  117. package/templates/react/add-on/table/info.json +13 -0
  118. package/templates/react/add-on/table/package.json +7 -0
  119. package/templates/react/add-on/tanstack-query/assets/src/integrations/tanstack-query/layout.tsx +5 -0
  120. package/templates/react/add-on/tanstack-query/assets/src/integrations/tanstack-query/root-provider.tsx.ejs +70 -0
  121. package/templates/react/add-on/tanstack-query/assets/src/routes/demo.tanstack-query.tsx.ejs +53 -0
  122. package/templates/react/add-on/tanstack-query/info.json +13 -0
  123. package/templates/react/add-on/tanstack-query/package.json +6 -0
  124. package/templates/react/base/README.md.ejs +558 -0
  125. package/templates/react/base/_dot_gitignore +5 -0
  126. package/templates/react/base/_dot_vscode/settings.biome.json +38 -0
  127. package/templates/react/base/_dot_vscode/settings.json +11 -0
  128. package/templates/react/base/index.html.ejs +20 -0
  129. package/templates/react/base/package.biome.json +10 -0
  130. package/templates/react/base/package.eslintprettier.json +11 -0
  131. package/templates/react/base/package.json +29 -0
  132. package/templates/react/base/package.ts.json +7 -0
  133. package/templates/react/base/package.tw.json +6 -0
  134. package/templates/react/base/public/favicon.ico +0 -0
  135. package/templates/react/base/public/logo192.png +0 -0
  136. package/templates/react/base/public/logo512.png +0 -0
  137. package/templates/react/base/public/manifest.json +25 -0
  138. package/templates/react/base/public/robots.txt +3 -0
  139. package/templates/react/base/src/App.css +38 -0
  140. package/templates/react/base/src/App.test.tsx.ejs +10 -0
  141. package/templates/react/base/src/App.tsx.ejs +74 -0
  142. package/templates/react/base/src/components/Header.tsx.ejs +27 -0
  143. package/templates/react/base/src/logo.svg +44 -0
  144. package/templates/react/base/src/reportWebVitals.ts.ejs +28 -0
  145. package/templates/react/base/src/styles.css.ejs +15 -0
  146. package/templates/react/base/toolchain/.prettierignore +3 -0
  147. package/templates/react/base/toolchain/biome.json +31 -0
  148. package/templates/react/base/toolchain/eslint.config.js +5 -0
  149. package/templates/react/base/toolchain/prettier.config.js +10 -0
  150. package/templates/react/base/tsconfig.json.ejs +29 -0
  151. package/templates/react/base/vite.config.js.ejs +23 -0
  152. package/templates/react/code-router/src/main.tsx.ejs +92 -0
  153. package/templates/react/example/tanchat/README.md +37 -0
  154. package/templates/react/example/tanchat/assets/_dot_env.local.append +2 -0
  155. package/templates/react/example/tanchat/assets/public/example-guitar-flowers.jpg +0 -0
  156. package/templates/react/example/tanchat/assets/public/example-guitar-motherboard.jpg +0 -0
  157. package/templates/react/example/tanchat/assets/public/example-guitar-racing.jpg +0 -0
  158. package/templates/react/example/tanchat/assets/public/example-guitar-steamer-trunk.jpg +0 -0
  159. package/templates/react/example/tanchat/assets/public/example-guitar-superhero.jpg +0 -0
  160. package/templates/react/example/tanchat/assets/public/example-guitar-traveling.jpg +0 -0
  161. package/templates/react/example/tanchat/assets/public/example-guitar-video-games.jpg +0 -0
  162. package/templates/react/example/tanchat/assets/src/components/example-AIAssistant.tsx +173 -0
  163. package/templates/react/example/tanchat/assets/src/components/example-GuitarRecommendation.tsx +47 -0
  164. package/templates/react/example/tanchat/assets/src/data/example-guitars.ts +83 -0
  165. package/templates/react/example/tanchat/assets/src/demo.index.css +220 -0
  166. package/templates/react/example/tanchat/assets/src/integrations/tanchat/header-user.tsx +5 -0
  167. package/templates/react/example/tanchat/assets/src/routes/example.chat.tsx +159 -0
  168. package/templates/react/example/tanchat/assets/src/routes/example.guitars/$guitarId.tsx +50 -0
  169. package/templates/react/example/tanchat/assets/src/routes/example.guitars/index.tsx +54 -0
  170. package/templates/react/example/tanchat/assets/src/store/example-assistant.ts +3 -0
  171. package/templates/react/example/tanchat/assets/src/utils/demo.ai.ts +62 -0
  172. package/templates/react/example/tanchat/assets/src/utils/demo.tools.ts +47 -0
  173. package/templates/react/example/tanchat/info.json +19 -0
  174. package/templates/react/example/tanchat/package.json +15 -0
  175. package/templates/react/file-router/package.fr.json +5 -0
  176. package/templates/react/file-router/src/main.tsx.ejs +55 -0
  177. package/templates/react/file-router/src/routes/__root.tsx.ejs +82 -0
  178. package/templates/solid/add-on/form/assets/src/routes/demo.form.tsx.ejs +352 -0
  179. package/templates/solid/add-on/form/info.json +13 -0
  180. package/templates/solid/add-on/form/package.json +5 -0
  181. package/templates/solid/add-on/module-federation/assets/module-federation.config.js.ejs +27 -0
  182. package/templates/solid/add-on/module-federation/assets/src/demo-mf-component.tsx +3 -0
  183. package/templates/solid/add-on/module-federation/assets/src/demo-mf-self-contained.tsx +9 -0
  184. package/templates/solid/add-on/module-federation/info.json +7 -0
  185. package/templates/solid/add-on/module-federation/package.json +5 -0
  186. package/templates/solid/add-on/sentry/assets/_dot_cursorrules.append +22 -0
  187. package/templates/solid/add-on/sentry/assets/_dot_env.local.append +2 -0
  188. package/templates/solid/add-on/sentry/assets/src/routes/demo.sentry.bad-event-handler.tsx +20 -0
  189. package/templates/solid/add-on/sentry/info.json +13 -0
  190. package/templates/solid/add-on/sentry/package.json +5 -0
  191. package/templates/solid/add-on/solid-ui/README.md +9 -0
  192. package/templates/solid/add-on/solid-ui/assets/src/lib/utils.ts +6 -0
  193. package/templates/solid/add-on/solid-ui/assets/src/styles.css +138 -0
  194. package/templates/solid/add-on/solid-ui/assets/ui.config.json +13 -0
  195. package/templates/solid/add-on/solid-ui/info.json +11 -0
  196. package/templates/solid/add-on/solid-ui/package.json +9 -0
  197. package/templates/solid/add-on/start/assets/app.config.ts +16 -0
  198. package/templates/solid/add-on/start/assets/src/api.ts +6 -0
  199. package/templates/solid/add-on/start/assets/src/client.tsx +7 -0
  200. package/templates/solid/add-on/start/assets/src/router.tsx.ejs +24 -0
  201. package/templates/solid/add-on/start/assets/src/routes/demo.start.server-funcs.tsx +49 -0
  202. package/templates/solid/add-on/start/assets/src/ssr.tsx +12 -0
  203. package/templates/solid/add-on/start/info.json +14 -0
  204. package/templates/solid/add-on/start/package.json +12 -0
  205. package/templates/solid/add-on/store/assets/src/lib/demo-store.ts +13 -0
  206. package/templates/solid/add-on/store/assets/src/routes/demo.store.tsx.ejs +77 -0
  207. package/templates/solid/add-on/store/info.json +13 -0
  208. package/templates/solid/add-on/store/package.json +6 -0
  209. package/templates/solid/add-on/tanstack-query/assets/src/integrations/tanstack-query/header-user.tsx +5 -0
  210. package/templates/solid/add-on/tanstack-query/assets/src/integrations/tanstack-query/provider.tsx +15 -0
  211. package/templates/solid/add-on/tanstack-query/assets/src/routes/demo.tanstack-query.tsx +30 -0
  212. package/templates/solid/add-on/tanstack-query/info.json +13 -0
  213. package/templates/solid/add-on/tanstack-query/package.json +6 -0
  214. package/templates/solid/base/README.md.ejs +215 -0
  215. package/templates/solid/base/_dot_cursorrules.append +35 -0
  216. package/templates/solid/base/_dot_gitignore +5 -0
  217. package/templates/solid/base/_dot_vscode/settings.biome.json +38 -0
  218. package/templates/solid/base/_dot_vscode/settings.json +11 -0
  219. package/templates/solid/base/index.html.ejs +20 -0
  220. package/templates/solid/base/package.biome.json +10 -0
  221. package/templates/solid/base/package.eslintprettier.json +11 -0
  222. package/templates/solid/base/package.json +22 -0
  223. package/templates/solid/base/package.ts.json +5 -0
  224. package/templates/solid/base/package.tw.json +6 -0
  225. package/templates/solid/base/public/favicon.ico +0 -0
  226. package/templates/solid/base/public/logo192.png +0 -0
  227. package/templates/solid/base/public/logo512.png +0 -0
  228. package/templates/solid/base/public/manifest.json +25 -0
  229. package/templates/solid/base/public/robots.txt +3 -0
  230. package/templates/solid/base/src/App.css +0 -0
  231. package/templates/solid/base/src/App.tsx.ejs +47 -0
  232. package/templates/solid/base/src/components/Header.tsx.ejs +26 -0
  233. package/templates/solid/base/src/logo.svg +120 -0
  234. package/templates/solid/base/src/styles.css.ejs +15 -0
  235. package/templates/solid/base/toolchain/.prettierignore +3 -0
  236. package/templates/solid/base/toolchain/biome.json +31 -0
  237. package/templates/solid/base/toolchain/eslint.config.js +5 -0
  238. package/templates/solid/base/toolchain/prettier.config.js +10 -0
  239. package/templates/solid/base/tsconfig.json.ejs +31 -0
  240. package/templates/solid/base/vite.config.js.ejs +22 -0
  241. package/templates/solid/code-router/src/main.tsx.ejs +71 -0
  242. package/templates/solid/example/tanchat/README.md +52 -0
  243. package/templates/solid/example/tanchat/assets/ai-streaming-server/README.md +110 -0
  244. package/templates/solid/example/tanchat/assets/ai-streaming-server/_dot_env.example +1 -0
  245. package/templates/solid/example/tanchat/assets/ai-streaming-server/package.json +26 -0
  246. package/templates/solid/example/tanchat/assets/ai-streaming-server/src/index.ts +102 -0
  247. package/templates/solid/example/tanchat/assets/ai-streaming-server/tsconfig.json +15 -0
  248. package/templates/solid/example/tanchat/assets/src/components/demo.SettingsDialog.tsx +149 -0
  249. package/templates/solid/example/tanchat/assets/src/demo.index.css +227 -0
  250. package/templates/solid/example/tanchat/assets/src/lib/demo-store.ts +13 -0
  251. package/templates/solid/example/tanchat/assets/src/routes/example.chat.tsx +435 -0
  252. package/templates/solid/example/tanchat/assets/src/store/demo.hooks.ts +17 -0
  253. package/templates/solid/example/tanchat/assets/src/store/demo.store.ts +133 -0
  254. package/templates/solid/example/tanchat/info.json +14 -0
  255. package/templates/solid/example/tanchat/package.json +7 -0
  256. package/templates/solid/file-router/package.fr.json +5 -0
  257. package/templates/solid/file-router/src/main.tsx.ejs +47 -0
  258. package/templates/solid/file-router/src/routes/__root.tsx.ejs +41 -0
  259. package/templates/solid/file-router/src/routes/index.tsx +43 -0
  260. package/tests/cra.test.ts +293 -0
  261. package/tests/snapshots/cra/cr-js-npm.json +33 -0
  262. package/tests/snapshots/cra/cr-ts-npm.json +34 -0
  263. package/tests/snapshots/cra/cr-ts-start-npm.json +38 -0
  264. package/tests/snapshots/cra/fr-ts-npm.json +34 -0
  265. package/tests/snapshots/cra/fr-ts-tw-npm.json +33 -0
  266. package/tests/snapshots/cra/solid-cr-js-npm.json +31 -0
  267. package/tests/snapshots/cra/solid-cr-ts-npm.json +32 -0
  268. package/tests/snapshots/cra/solid-cr-ts-start-npm.json +36 -0
  269. package/tests/snapshots/cra/solid-fr-ts-npm.json +33 -0
  270. package/tests/snapshots/cra/solid-fr-ts-tw-npm.json +32 -0
  271. package/tests/test-utilities.ts +87 -0
  272. package/tsconfig.json +11 -0
@@ -0,0 +1,309 @@
1
+ import { cancel, confirm, isCancel, multiselect, select, text, } from '@clack/prompts';
2
+ import { DEFAULT_PACKAGE_MANAGER, SUPPORTED_PACKAGE_MANAGERS, getPackageManager, } from './package-manager.js';
3
+ import { DEFAULT_TOOLCHAIN, SUPPORTED_TOOLCHAINS } from './toolchain.js';
4
+ import { CODE_ROUTER, DEFAULT_FRAMEWORK, FILE_ROUTER } from './constants.js';
5
+ import { finalizeAddOns, getAllAddOns, loadRemoteAddOn } from './add-ons.js';
6
+ // If all CLI options are provided, use them directly
7
+ export async function normalizeOptions(cliOptions) {
8
+ // in some cases, if you use windows/powershell, the argument for addons
9
+ // if sepparated by comma is not really passed as an array, but as a string
10
+ // with spaces, We need to normalize this edge case.
11
+ if (Array.isArray(cliOptions.addOns) && cliOptions.addOns.length === 1) {
12
+ const parseSeparatedArgs = cliOptions.addOns[0].split(' ');
13
+ if (parseSeparatedArgs.length > 1) {
14
+ cliOptions.addOns = parseSeparatedArgs;
15
+ }
16
+ }
17
+ if (cliOptions.projectName) {
18
+ let typescript = cliOptions.template === 'typescript' ||
19
+ cliOptions.template === 'file-router' ||
20
+ cliOptions.framework === 'solid';
21
+ let tailwind = !!cliOptions.tailwind;
22
+ if (cliOptions.framework === 'solid') {
23
+ tailwind = true;
24
+ }
25
+ let mode = cliOptions.template === 'file-router' ? FILE_ROUTER : CODE_ROUTER;
26
+ const overlay = cliOptions.overlay
27
+ ? (await loadRemoteAddOn(cliOptions.overlay))
28
+ : undefined;
29
+ if (overlay) {
30
+ tailwind = overlay.tailwind;
31
+ typescript = overlay.typescript;
32
+ cliOptions.framework = overlay.framework;
33
+ mode = overlay.mode;
34
+ }
35
+ let addOns = false;
36
+ let chosenAddOns = [];
37
+ if (Array.isArray(cliOptions.addOns) || overlay?.dependsOn) {
38
+ addOns = true;
39
+ let finalAddOns = [...(overlay?.dependsOn || [])];
40
+ if (cliOptions.addOns && Array.isArray(cliOptions.addOns)) {
41
+ finalAddOns = [...finalAddOns, ...cliOptions.addOns];
42
+ }
43
+ chosenAddOns = await finalizeAddOns(cliOptions.framework || DEFAULT_FRAMEWORK, cliOptions.template === 'file-router' ? FILE_ROUTER : CODE_ROUTER, finalAddOns);
44
+ tailwind = true;
45
+ typescript = true;
46
+ }
47
+ return {
48
+ framework: cliOptions.framework || 'react',
49
+ projectName: cliOptions.projectName,
50
+ typescript,
51
+ tailwind,
52
+ packageManager: cliOptions.packageManager ||
53
+ getPackageManager() ||
54
+ DEFAULT_PACKAGE_MANAGER,
55
+ toolchain: cliOptions.toolchain || DEFAULT_TOOLCHAIN,
56
+ mode,
57
+ git: !!cliOptions.git,
58
+ addOns,
59
+ chosenAddOns,
60
+ variableValues: {},
61
+ overlay,
62
+ };
63
+ }
64
+ }
65
+ async function collectVariables(variables) {
66
+ const responses = {};
67
+ for (const variable of variables) {
68
+ if (variable.type === 'string') {
69
+ const response = await text({
70
+ message: variable.description,
71
+ initialValue: variable.default,
72
+ });
73
+ if (isCancel(response)) {
74
+ cancel('Operation cancelled.');
75
+ process.exit(0);
76
+ }
77
+ responses[variable.name] = response;
78
+ }
79
+ else if (variable.type === 'number') {
80
+ const response = await text({
81
+ message: variable.description,
82
+ initialValue: variable.default.toString(),
83
+ });
84
+ if (isCancel(response)) {
85
+ cancel('Operation cancelled.');
86
+ process.exit(0);
87
+ }
88
+ responses[variable.name] = Number(response);
89
+ }
90
+ else {
91
+ const response = await confirm({
92
+ message: variable.description,
93
+ initialValue: variable.default === true,
94
+ });
95
+ if (isCancel(response)) {
96
+ cancel('Operation cancelled.');
97
+ process.exit(0);
98
+ }
99
+ responses[variable.name] = response;
100
+ }
101
+ }
102
+ return responses;
103
+ }
104
+ export async function promptForOptions(cliOptions) {
105
+ const options = {};
106
+ options.framework = cliOptions.framework || DEFAULT_FRAMEWORK;
107
+ if (options.framework === 'solid') {
108
+ options.typescript = true;
109
+ options.tailwind = true;
110
+ }
111
+ if (cliOptions.addOns) {
112
+ options.typescript = true;
113
+ }
114
+ if (!cliOptions.projectName) {
115
+ const value = await text({
116
+ message: 'What would you like to name your project?',
117
+ defaultValue: 'my-app',
118
+ validate(value) {
119
+ if (!value) {
120
+ return 'Please enter a name';
121
+ }
122
+ },
123
+ });
124
+ if (isCancel(value)) {
125
+ cancel('Operation cancelled.');
126
+ process.exit(0);
127
+ }
128
+ options.projectName = value;
129
+ }
130
+ // Router type selection
131
+ if (!cliOptions.template) {
132
+ const routerType = await select({
133
+ message: 'Select the router type:',
134
+ options: [
135
+ {
136
+ value: FILE_ROUTER,
137
+ label: 'File Router - File-based routing structure',
138
+ },
139
+ {
140
+ value: CODE_ROUTER,
141
+ label: 'Code Router - Traditional code-based routing',
142
+ },
143
+ ],
144
+ initialValue: FILE_ROUTER,
145
+ });
146
+ if (isCancel(routerType)) {
147
+ cancel('Operation cancelled.');
148
+ process.exit(0);
149
+ }
150
+ options.mode = routerType;
151
+ }
152
+ else {
153
+ options.mode =
154
+ cliOptions.template === 'file-router' ? FILE_ROUTER : CODE_ROUTER;
155
+ if (options.mode === FILE_ROUTER) {
156
+ options.typescript = true;
157
+ }
158
+ }
159
+ // TypeScript selection (if using Code Router)
160
+ if (!options.typescript) {
161
+ if (options.mode === CODE_ROUTER) {
162
+ const typescriptEnable = await confirm({
163
+ message: 'Would you like to use TypeScript?',
164
+ initialValue: true,
165
+ });
166
+ if (isCancel(typescriptEnable)) {
167
+ cancel('Operation cancelled.');
168
+ process.exit(0);
169
+ }
170
+ options.typescript = typescriptEnable;
171
+ }
172
+ else {
173
+ options.typescript = true;
174
+ }
175
+ }
176
+ // Tailwind selection
177
+ if (!cliOptions.tailwind && options.framework === 'react') {
178
+ const tailwind = await confirm({
179
+ message: 'Would you like to use Tailwind CSS?',
180
+ initialValue: true,
181
+ });
182
+ if (isCancel(tailwind)) {
183
+ cancel('Operation cancelled.');
184
+ process.exit(0);
185
+ }
186
+ options.tailwind = tailwind;
187
+ }
188
+ else {
189
+ options.tailwind = options.framework === 'solid' || !!cliOptions.tailwind;
190
+ }
191
+ // Package manager selection
192
+ if (cliOptions.packageManager === undefined) {
193
+ const detectedPackageManager = getPackageManager();
194
+ if (!detectedPackageManager) {
195
+ const pm = await select({
196
+ message: 'Select package manager:',
197
+ options: SUPPORTED_PACKAGE_MANAGERS.map((pm) => ({
198
+ value: pm,
199
+ label: pm,
200
+ })),
201
+ initialValue: DEFAULT_PACKAGE_MANAGER,
202
+ });
203
+ if (isCancel(pm)) {
204
+ cancel('Operation cancelled.');
205
+ process.exit(0);
206
+ }
207
+ options.packageManager = pm;
208
+ }
209
+ else {
210
+ options.packageManager = detectedPackageManager;
211
+ }
212
+ }
213
+ else {
214
+ options.packageManager = cliOptions.packageManager;
215
+ }
216
+ // Toolchain selection
217
+ if (cliOptions.toolchain === undefined) {
218
+ const tc = await select({
219
+ message: 'Select toolchain',
220
+ options: SUPPORTED_TOOLCHAINS.map((tc) => ({
221
+ value: tc,
222
+ label: tc,
223
+ })),
224
+ initialValue: DEFAULT_TOOLCHAIN,
225
+ });
226
+ if (isCancel(tc)) {
227
+ cancel('Operation cancelled.');
228
+ process.exit(0);
229
+ }
230
+ options.toolchain = tc;
231
+ }
232
+ else {
233
+ options.toolchain = cliOptions.toolchain;
234
+ }
235
+ options.chosenAddOns = [];
236
+ if (Array.isArray(cliOptions.addOns)) {
237
+ options.chosenAddOns = await finalizeAddOns(options.framework, options.mode, cliOptions.addOns);
238
+ options.tailwind = true;
239
+ }
240
+ else if (cliOptions.addOns) {
241
+ // Select any add-ons
242
+ const allAddOns = await getAllAddOns(options.framework, options.mode);
243
+ const addOns = allAddOns.filter((addOn) => addOn.type === 'add-on');
244
+ let selectedAddOns = [];
245
+ if (options.typescript && addOns.length > 0) {
246
+ const value = await multiselect({
247
+ message: 'What add-ons would you like for your project:',
248
+ options: addOns.map((addOn) => ({
249
+ value: addOn.id,
250
+ label: addOn.name,
251
+ hint: addOn.description,
252
+ })),
253
+ required: false,
254
+ });
255
+ if (isCancel(value)) {
256
+ cancel('Operation cancelled.');
257
+ process.exit(0);
258
+ }
259
+ selectedAddOns = value;
260
+ }
261
+ // Select any examples
262
+ let selectedExamples = [];
263
+ const examples = allAddOns.filter((addOn) => addOn.type === 'example');
264
+ if (options.typescript && examples.length > 0) {
265
+ const value = await multiselect({
266
+ message: 'Would you like any examples?',
267
+ options: examples.map((addOn) => ({
268
+ value: addOn.id,
269
+ label: addOn.name,
270
+ hint: addOn.description,
271
+ })),
272
+ required: false,
273
+ });
274
+ if (isCancel(value)) {
275
+ cancel('Operation cancelled.');
276
+ process.exit(0);
277
+ }
278
+ selectedExamples = value;
279
+ }
280
+ if (selectedAddOns.length > 0 || selectedExamples.length > 0) {
281
+ options.chosenAddOns = await finalizeAddOns(options.framework, options.mode, [...selectedAddOns, ...selectedExamples]);
282
+ options.tailwind = true;
283
+ }
284
+ }
285
+ // Collect variables
286
+ const variables = [];
287
+ for (const addOn of options.chosenAddOns) {
288
+ for (const variable of addOn.variables ?? []) {
289
+ variables.push(variable);
290
+ }
291
+ }
292
+ options.variableValues = await collectVariables(variables);
293
+ // Git selection
294
+ if (cliOptions.git === undefined) {
295
+ const git = await confirm({
296
+ message: 'Would you like to initialize a new git repository?',
297
+ initialValue: true,
298
+ });
299
+ if (isCancel(git)) {
300
+ cancel('Operation cancelled.');
301
+ process.exit(0);
302
+ }
303
+ options.git = git;
304
+ }
305
+ else {
306
+ options.git = !!cliOptions.git;
307
+ }
308
+ return options;
309
+ }
@@ -0,0 +1,30 @@
1
+ export const SUPPORTED_PACKAGE_MANAGERS = [
2
+ 'npm',
3
+ 'yarn',
4
+ 'pnpm',
5
+ 'bun',
6
+ 'deno',
7
+ ];
8
+ export const DEFAULT_PACKAGE_MANAGER = 'npm';
9
+ export function getPackageManager() {
10
+ const userAgent = process.env.npm_config_user_agent;
11
+ if (userAgent === undefined) {
12
+ return DEFAULT_PACKAGE_MANAGER;
13
+ }
14
+ const packageManager = SUPPORTED_PACKAGE_MANAGERS.find((manager) => userAgent.startsWith(manager));
15
+ return packageManager;
16
+ }
17
+ export function packageManagerExecute(environment, packagerManager, pkg, args, cwd) {
18
+ switch (packagerManager) {
19
+ case 'yarn':
20
+ return environment.execute('yarn', ['dlx', pkg, ...args], cwd);
21
+ case 'pnpm':
22
+ return environment.execute('pnpx', [pkg, ...args], cwd);
23
+ case 'bun':
24
+ return environment.execute('bunx', ['--bun', pkg, ...args], cwd);
25
+ case 'deno':
26
+ return environment.execute('deno', ['run', `npm:${pkg}`, ...args], cwd);
27
+ default:
28
+ return environment.execute('npx', [pkg, ...args], cwd);
29
+ }
30
+ }
@@ -0,0 +1,6 @@
1
+ import { dirname, join } from 'node:path';
2
+ import { fileURLToPath } from 'node:url';
3
+ export function getTemplatesRoot() {
4
+ // Get the parent directory because the compiled code is in the dist folder
5
+ return join(dirname(dirname(fileURLToPath(import.meta.url))), 'templates');
6
+ }
@@ -0,0 +1,6 @@
1
+ export const SUPPORTED_TOOLCHAINS = [
2
+ 'none',
3
+ 'biome',
4
+ 'eslint+prettier',
5
+ ];
6
+ export const DEFAULT_TOOLCHAIN = 'none';
@@ -0,0 +1,5 @@
1
+ import type { AddOn, CliOptions, Framework } from './types.js';
2
+ export declare function getAllAddOns(framework: Framework, template: string): Promise<Array<AddOn>>;
3
+ export declare function finalizeAddOns(framework: Framework, template: string, chosenAddOnIDs: Array<string>): Promise<Array<AddOn>>;
4
+ export declare function listAddOns(options: CliOptions): Promise<void>;
5
+ export declare function loadRemoteAddOn(url: string): Promise<AddOn>;
@@ -0,0 +1,3 @@
1
+ export declare function add(addOns: Array<string>, { silent, }?: {
2
+ silent?: boolean;
3
+ }): Promise<void>;
@@ -0,0 +1 @@
1
+ export declare function cli(): void;
@@ -0,0 +1,7 @@
1
+ import type { Environment, Options } from './types';
2
+ export type PersistedOptions = Exclude<Partial<Options>, 'addOns' | 'chosenAddOns'> & {
3
+ version: number;
4
+ existingAddOns: Array<string>;
5
+ };
6
+ export declare function writeConfigFile(environment: Environment, targetDir: string, options: Options): Promise<void>;
7
+ export declare function readConfigFile(targetDir: string): Promise<PersistedOptions | null>;
@@ -0,0 +1,6 @@
1
+ import type { Framework } from './types.js';
2
+ export declare const CODE_ROUTER = "code-router";
3
+ export declare const FILE_ROUTER = "file-router";
4
+ export declare const SUPPORTED_FRAMEWORKS: Array<Framework>;
5
+ export declare const DEFAULT_FRAMEWORK: Framework;
6
+ export declare const CONFIG_FILE = ".cta.json";
@@ -0,0 +1,6 @@
1
+ import type { Environment, Options } from './types.js';
2
+ export declare function createApp(options: Options, { silent, environment, cwd, }: {
3
+ silent?: boolean;
4
+ environment: Environment;
5
+ cwd?: string;
6
+ }): Promise<void>;
@@ -0,0 +1,3 @@
1
+ type AddOnMode = 'add-on' | 'overlay';
2
+ export declare function initAddOn(mode: AddOnMode): Promise<void>;
3
+ export {};
@@ -0,0 +1,12 @@
1
+ import type { Environment } from './types.js';
2
+ export declare function createDefaultEnvironment(): Environment;
3
+ export declare function createMemoryEnvironment(): {
4
+ environment: Environment;
5
+ output: {
6
+ files: Record<string, string>;
7
+ commands: Array<{
8
+ command: string;
9
+ args: Array<string>;
10
+ }>;
11
+ };
12
+ };
@@ -0,0 +1 @@
1
+ export { cli } from './cli.js';
@@ -0,0 +1 @@
1
+ export default function runServer(sse: boolean): Promise<void>;
@@ -0,0 +1,3 @@
1
+ import type { CliOptions, Options } from './types.js';
2
+ export declare function normalizeOptions(cliOptions: CliOptions): Promise<Options | undefined>;
3
+ export declare function promptForOptions(cliOptions: CliOptions): Promise<Required<Options>>;
@@ -0,0 +1,6 @@
1
+ import type { Environment } from './types.js';
2
+ export declare const SUPPORTED_PACKAGE_MANAGERS: readonly ["npm", "yarn", "pnpm", "bun", "deno"];
3
+ export type PackageManager = (typeof SUPPORTED_PACKAGE_MANAGERS)[number];
4
+ export declare const DEFAULT_PACKAGE_MANAGER: PackageManager;
5
+ export declare function getPackageManager(): PackageManager | undefined;
6
+ export declare function packageManagerExecute(environment: Environment, packagerManager: PackageManager, pkg: string, args: Array<string>, cwd: string): Promise<void>;
@@ -0,0 +1 @@
1
+ export declare function getTemplatesRoot(): string;
@@ -0,0 +1,3 @@
1
+ export declare const SUPPORTED_TOOLCHAINS: readonly ["none", "biome", "eslint+prettier"];
2
+ export type ToolChain = (typeof SUPPORTED_TOOLCHAINS)[number];
3
+ export declare const DEFAULT_TOOLCHAIN: ToolChain;
@@ -0,0 +1,107 @@
1
+ import type { CODE_ROUTER, FILE_ROUTER } from './constants.js';
2
+ import type { PackageManager } from './package-manager.js';
3
+ import type { ToolChain } from './toolchain.js';
4
+ export type Framework = 'solid' | 'react';
5
+ export interface Options {
6
+ framework: Framework;
7
+ projectName: string;
8
+ typescript: boolean;
9
+ tailwind: boolean;
10
+ packageManager: PackageManager;
11
+ toolchain: ToolChain;
12
+ mode: typeof CODE_ROUTER | typeof FILE_ROUTER;
13
+ addOns: boolean;
14
+ chosenAddOns: Array<AddOn>;
15
+ git: boolean;
16
+ variableValues: Record<string, string | number | boolean>;
17
+ overlay?: AddOn | undefined;
18
+ }
19
+ export interface CliOptions {
20
+ template?: 'typescript' | 'javascript' | 'file-router';
21
+ framework?: Framework;
22
+ tailwind?: boolean;
23
+ packageManager?: PackageManager;
24
+ toolchain?: ToolChain;
25
+ projectName?: string;
26
+ git?: boolean;
27
+ addOns?: Array<string> | boolean;
28
+ listAddOns?: boolean;
29
+ mcp?: boolean;
30
+ mcpSse?: boolean;
31
+ overlay?: string;
32
+ targetDir?: string;
33
+ }
34
+ export type Environment = {
35
+ startRun: () => void;
36
+ finishRun: () => void;
37
+ getErrors: () => Array<string>;
38
+ appendFile: (path: string, contents: string) => Promise<void>;
39
+ copyFile: (from: string, to: string) => Promise<void>;
40
+ writeFile: (path: string, contents: string) => Promise<void>;
41
+ execute: (command: string, args: Array<string>, cwd: string) => Promise<void>;
42
+ deleteFile: (path: string) => Promise<void>;
43
+ readFile: (path: string, encoding?: BufferEncoding) => Promise<string>;
44
+ exists: (path: string) => boolean;
45
+ readdir: (path: string) => Array<string>;
46
+ isDirectory: (path: string) => boolean;
47
+ };
48
+ type BooleanVariable = {
49
+ name: string;
50
+ default: boolean;
51
+ description: string;
52
+ type: 'boolean';
53
+ };
54
+ type NumberVariable = {
55
+ name: string;
56
+ default: number;
57
+ description: string;
58
+ type: 'number';
59
+ };
60
+ type StringVariable = {
61
+ name: string;
62
+ default: string;
63
+ description: string;
64
+ type: 'string';
65
+ };
66
+ export type Variable = BooleanVariable | NumberVariable | StringVariable;
67
+ export type AddOn = {
68
+ id: string;
69
+ name: string;
70
+ description: string;
71
+ type: 'add-on' | 'example' | 'overlay';
72
+ link: string;
73
+ templates: Array<string>;
74
+ routes: Array<{
75
+ url: string;
76
+ name: string;
77
+ }>;
78
+ packageAdditions: {
79
+ dependencies?: Record<string, string>;
80
+ devDependencies?: Record<string, string>;
81
+ scripts?: Record<string, string>;
82
+ };
83
+ command?: {
84
+ command: string;
85
+ args?: Array<string>;
86
+ };
87
+ readme?: string;
88
+ phase: 'setup' | 'add-on';
89
+ shadcnComponents?: Array<string>;
90
+ warning?: string;
91
+ dependsOn?: Array<string>;
92
+ variables?: Array<Variable>;
93
+ files?: Record<string, string>;
94
+ deletedFiles?: Array<string>;
95
+ };
96
+ export type Overlay = AddOn & {
97
+ type: 'overlay';
98
+ version: string;
99
+ author: string;
100
+ link: string;
101
+ license: string;
102
+ mode: typeof CODE_ROUTER | typeof FILE_ROUTER;
103
+ framework: Framework;
104
+ typescript: boolean;
105
+ tailwind: boolean;
106
+ };
107
+ export {};
@@ -0,0 +1 @@
1
+ export declare function sortObject(obj: Record<string, string>): Record<string, string>;
package/dist/types.js ADDED
@@ -0,0 +1 @@
1
+ export {};
package/dist/utils.js ADDED
@@ -0,0 +1,8 @@
1
+ export function sortObject(obj) {
2
+ return Object.keys(obj)
3
+ .sort()
4
+ .reduce((acc, key) => {
5
+ acc[key] = obj[key];
6
+ return acc;
7
+ }, {});
8
+ }
package/package.json ADDED
@@ -0,0 +1,49 @@
1
+ {
2
+ "name": "@tanstack/cta-engine",
3
+ "version": "0.10.0-alpha.6",
4
+ "description": "Tanstack Application Builder Engine",
5
+ "type": "module",
6
+ "main": "./dist/index.js",
7
+ "types": "./dist/types/index.d.ts",
8
+ "repository": {
9
+ "type": "git",
10
+ "url": "https://github.com/TanStack/create-tsrouter-app.git"
11
+ },
12
+ "homepage": "https://tanstack.com/router",
13
+ "funding": {
14
+ "type": "github",
15
+ "url": "https://github.com/sponsors/tannerlinsley"
16
+ },
17
+ "keywords": [
18
+ "react",
19
+ "tanstack",
20
+ "router",
21
+ "create-react-app"
22
+ ],
23
+ "author": "Jack Herrington <jherr@pobox.com>",
24
+ "license": "MIT",
25
+ "dependencies": {
26
+ "@clack/prompts": "^0.10.0",
27
+ "@modelcontextprotocol/sdk": "^1.6.0",
28
+ "chalk": "^5.4.1",
29
+ "commander": "^13.1.0",
30
+ "ejs": "^3.1.10",
31
+ "execa": "^9.5.2",
32
+ "express": "^4.21.2",
33
+ "memfs": "^4.17.0",
34
+ "prettier": "^3.5.0",
35
+ "zod": "^3.24.2"
36
+ },
37
+ "devDependencies": {
38
+ "@tanstack/config": "^0.16.2",
39
+ "@types/ejs": "^3.1.5",
40
+ "@types/express": "^5.0.0",
41
+ "@types/node": "^22.13.4",
42
+ "eslint": "^9.20.0",
43
+ "eslint-plugin-react-hooks": "^5.1.0",
44
+ "eslint-plugin-unused-imports": "^4.1.4",
45
+ "typescript": "^5.6.3",
46
+ "vitest": "^3.0.8"
47
+ },
48
+ "scripts": {}
49
+ }