@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,254 @@
1
+ import { readFile, readdir } from 'node:fs/promises';
2
+ import { existsSync, mkdirSync, writeFileSync } from 'node:fs';
3
+ import { basename, dirname, resolve } from 'node:path';
4
+ import chalk from 'chalk';
5
+ import { createMemoryEnvironment } from './environment.js';
6
+ import { createApp } from './create-app.js';
7
+ import { readConfigFile } from './config-file.js';
8
+ import { finalizeAddOns } from './add-ons.js';
9
+ const INFO_FILE = {
10
+ 'add-on': '.add-on/info.json',
11
+ overlay: 'overlay-info.json',
12
+ };
13
+ const COMPILED_FILE = {
14
+ 'add-on': 'add-on.json',
15
+ overlay: 'overlay.json',
16
+ };
17
+ const ADD_ON_DIR = '.add-on';
18
+ const ASSETS_DIR = 'assets';
19
+ const IGNORE_FILES = [
20
+ ADD_ON_DIR,
21
+ 'node_modules',
22
+ 'dist',
23
+ 'build',
24
+ '.git',
25
+ 'pnpm-lock.yaml',
26
+ 'package-lock.json',
27
+ 'yarn.lock',
28
+ 'bun.lockb',
29
+ 'bun.lock',
30
+ 'deno.lock',
31
+ 'add-on.json',
32
+ 'add-on-info.json',
33
+ 'package.json',
34
+ ];
35
+ const ADD_ON_IGNORE_FILES = [
36
+ 'main.jsx',
37
+ 'App.jsx',
38
+ 'main.tsx',
39
+ 'App.tsx',
40
+ 'routeTree.gen.ts',
41
+ ];
42
+ function templatize(routeCode, routeFile) {
43
+ let code = routeCode;
44
+ // Replace the import
45
+ code = code.replace(/import { createFileRoute } from '@tanstack\/react-router'/g, `import { <% if (fileRouter) { %>createFileRoute<% } else { %>createRoute<% } %> } from '@tanstack/react-router'`);
46
+ // Extract route path and definition, then transform the route declaration
47
+ const routeMatch = code.match(/export\s+const\s+Route\s*=\s*createFileRoute\(['"]([^'"]+)['"]\)\s*\(\{([^}]+)\}\)/);
48
+ let path = '';
49
+ if (routeMatch) {
50
+ const fullMatch = routeMatch[0];
51
+ path = routeMatch[1];
52
+ const routeDefinition = routeMatch[2];
53
+ code = code.replace(fullMatch, `<% if (codeRouter) { %>
54
+ import type { RootRoute } from '@tanstack/react-router'
55
+ <% } else { %>
56
+ export const Route = createFileRoute('${path}')({${routeDefinition}})
57
+ <% } %>`);
58
+ code += `
59
+ <% if (codeRouter) { %>
60
+ export default (parentRoute: RootRoute) => createRoute({
61
+ path: '${path}',
62
+ ${routeDefinition}
63
+ getParentRoute: () => parentRoute,
64
+ })
65
+ <% } %>
66
+ `;
67
+ }
68
+ else {
69
+ console.error(`No route found in the file: ${routeFile}`);
70
+ }
71
+ const name = basename(path)
72
+ .replace('.tsx', '')
73
+ .replace(/^demo/, '')
74
+ .replace('.', ' ')
75
+ .trim();
76
+ return { url: path, code, name };
77
+ }
78
+ async function createOptions(json) {
79
+ return {
80
+ ...json,
81
+ chosenAddOns: await finalizeAddOns(json.framework, json.mode, [
82
+ ...json.existingAddOns,
83
+ ]),
84
+ };
85
+ }
86
+ async function runCreateApp(options) {
87
+ const { environment, output } = createMemoryEnvironment();
88
+ await createApp(options, {
89
+ silent: true,
90
+ environment,
91
+ cwd: process.cwd(),
92
+ });
93
+ return output;
94
+ }
95
+ async function recursivelyGatherFiles(path, files) {
96
+ const dirFiles = await readdir(path, { withFileTypes: true });
97
+ for (const file of dirFiles) {
98
+ if (file.isDirectory()) {
99
+ await recursivelyGatherFiles(resolve(path, file.name), files);
100
+ }
101
+ else {
102
+ files[resolve(path, file.name)] = (await readFile(resolve(path, file.name))).toString();
103
+ }
104
+ }
105
+ }
106
+ async function compareFiles(path, ignore, original, changedFiles) {
107
+ const files = await readdir(path, { withFileTypes: true });
108
+ for (const file of files) {
109
+ const filePath = `${path}/${file.name}`;
110
+ if (!ignore.includes(file.name)) {
111
+ if (file.isDirectory()) {
112
+ await compareFiles(filePath, ignore, original, changedFiles);
113
+ }
114
+ else {
115
+ const contents = (await readFile(filePath)).toString();
116
+ const absolutePath = resolve(process.cwd(), filePath);
117
+ if (!original[absolutePath] || original[absolutePath] !== contents) {
118
+ changedFiles[filePath] = contents;
119
+ }
120
+ }
121
+ }
122
+ }
123
+ }
124
+ export async function initAddOn(mode) {
125
+ const persistedOptions = await readConfigFile(process.cwd());
126
+ if (!persistedOptions) {
127
+ console.error(`${chalk.red('There is no .cta.json file in your project.')}
128
+
129
+ This is probably because this was created with an older version of create-tsrouter-app.`);
130
+ return;
131
+ }
132
+ if (mode === 'add-on') {
133
+ if (persistedOptions.mode !== 'file-router') {
134
+ console.error(`${chalk.red('This project is not using file-router mode.')}
135
+
136
+ To create an add-on, the project must be created with the file-router mode.`);
137
+ return;
138
+ }
139
+ if (!persistedOptions.tailwind) {
140
+ console.error(`${chalk.red('This project is not using Tailwind CSS.')}
141
+
142
+ To create an add-on, the project must be created with Tailwind CSS.`);
143
+ return;
144
+ }
145
+ if (!persistedOptions.typescript) {
146
+ console.error(`${chalk.red('This project is not using TypeScript.')}
147
+
148
+ To create an add-on, the project must be created with TypeScript.`);
149
+ return;
150
+ }
151
+ }
152
+ const info = existsSync(INFO_FILE[mode])
153
+ ? JSON.parse((await readFile(INFO_FILE[mode])).toString())
154
+ : {
155
+ name: `${persistedOptions.projectName}-${mode}`,
156
+ version: '0.0.1',
157
+ description: mode === 'add-on' ? 'Add-on' : 'Project overlay',
158
+ author: 'Jane Smith <jane.smith@example.com>',
159
+ license: 'MIT',
160
+ link: `https://github.com/jane-smith/${persistedOptions.projectName}-${mode}`,
161
+ command: {},
162
+ shadcnComponents: [],
163
+ templates: [persistedOptions.mode],
164
+ routes: [],
165
+ warning: '',
166
+ variables: {},
167
+ phase: 'add-on',
168
+ type: mode,
169
+ packageAdditions: {
170
+ scripts: {},
171
+ dependencies: {},
172
+ devDependencies: {},
173
+ },
174
+ };
175
+ const compiledInfo = JSON.parse(JSON.stringify(info));
176
+ const originalOutput = await runCreateApp(await createOptions(persistedOptions));
177
+ const originalPackageJson = JSON.parse(originalOutput.files[resolve(process.cwd(), 'package.json')]);
178
+ const currentPackageJson = JSON.parse((await readFile('package.json')).toString());
179
+ for (const script of Object.keys(currentPackageJson.scripts)) {
180
+ if (originalPackageJson.scripts[script] !== currentPackageJson.scripts[script]) {
181
+ info.packageAdditions.scripts[script] = currentPackageJson.scripts[script];
182
+ }
183
+ }
184
+ const dependencies = {};
185
+ for (const dependency of Object.keys(currentPackageJson.dependencies)) {
186
+ if (originalPackageJson.dependencies[dependency] !==
187
+ currentPackageJson.dependencies[dependency]) {
188
+ dependencies[dependency] = currentPackageJson.dependencies[dependency];
189
+ }
190
+ }
191
+ info.packageAdditions.dependencies = dependencies;
192
+ const devDependencies = {};
193
+ for (const dependency of Object.keys(currentPackageJson.devDependencies)) {
194
+ if (originalPackageJson.devDependencies[dependency] !==
195
+ currentPackageJson.devDependencies[dependency]) {
196
+ devDependencies[dependency] =
197
+ currentPackageJson.devDependencies[dependency];
198
+ }
199
+ }
200
+ info.packageAdditions.devDependencies = devDependencies;
201
+ // Find altered files
202
+ const changedFiles = {};
203
+ await compareFiles('.', IGNORE_FILES, originalOutput.files, changedFiles);
204
+ if (mode === 'overlay') {
205
+ compiledInfo.files = changedFiles;
206
+ }
207
+ else {
208
+ const assetsDir = resolve(ADD_ON_DIR, ASSETS_DIR);
209
+ if (!existsSync(assetsDir)) {
210
+ await compareFiles('.', IGNORE_FILES, originalOutput.files, changedFiles);
211
+ for (const file of Object.keys(changedFiles).filter((file) => !ADD_ON_IGNORE_FILES.includes(basename(file)))) {
212
+ mkdirSync(dirname(resolve(assetsDir, file)), {
213
+ recursive: true,
214
+ });
215
+ if (file.includes('/routes/')) {
216
+ const { url, code, name } = templatize(changedFiles[file], file);
217
+ info.routes.push({
218
+ url,
219
+ name,
220
+ });
221
+ writeFileSync(resolve(assetsDir, `${file}.ejs`), code);
222
+ }
223
+ else {
224
+ writeFileSync(resolve(assetsDir, file), changedFiles[file]);
225
+ }
226
+ }
227
+ }
228
+ const addOnFiles = {};
229
+ await recursivelyGatherFiles(assetsDir, addOnFiles);
230
+ compiledInfo.files = Object.keys(addOnFiles).reduce((acc, file) => {
231
+ acc[file.replace(assetsDir, '.')] = addOnFiles[file];
232
+ return acc;
233
+ }, {});
234
+ }
235
+ compiledInfo.routes = info.routes;
236
+ compiledInfo.framework = persistedOptions.framework;
237
+ compiledInfo.addDependencies = persistedOptions.existingAddOns;
238
+ if (mode === 'overlay') {
239
+ compiledInfo.mode = persistedOptions.mode;
240
+ compiledInfo.typescript = persistedOptions.typescript;
241
+ compiledInfo.tailwind = persistedOptions.tailwind;
242
+ compiledInfo.deletedFiles = [];
243
+ for (const file of Object.keys(originalOutput.files)) {
244
+ if (!existsSync(file)) {
245
+ compiledInfo.deletedFiles.push(file.replace(process.cwd(), '.'));
246
+ }
247
+ }
248
+ }
249
+ if (!existsSync(resolve(INFO_FILE[mode]))) {
250
+ mkdirSync(resolve(dirname(INFO_FILE[mode])), { recursive: true });
251
+ writeFileSync(INFO_FILE[mode], JSON.stringify(info, null, 2));
252
+ }
253
+ writeFileSync(COMPILED_FILE[mode], JSON.stringify(compiledInfo, null, 2));
254
+ }
@@ -0,0 +1,119 @@
1
+ import { appendFile, copyFile, mkdir, readFile, unlink, writeFile, } from 'node:fs/promises';
2
+ import { existsSync, readdirSync, statSync } from 'node:fs';
3
+ import { dirname } from 'node:path';
4
+ import { execa } from 'execa';
5
+ import { memfs } from 'memfs';
6
+ export function createDefaultEnvironment() {
7
+ let errors = [];
8
+ return {
9
+ startRun: () => {
10
+ errors = [];
11
+ },
12
+ finishRun: () => { },
13
+ getErrors: () => errors,
14
+ appendFile: async (path, contents) => {
15
+ await mkdir(dirname(path), { recursive: true });
16
+ return appendFile(path, contents);
17
+ },
18
+ copyFile: async (from, to) => {
19
+ await mkdir(dirname(to), { recursive: true });
20
+ return copyFile(from, to);
21
+ },
22
+ writeFile: async (path, contents) => {
23
+ await mkdir(dirname(path), { recursive: true });
24
+ return writeFile(path, contents);
25
+ },
26
+ execute: async (command, args, cwd) => {
27
+ try {
28
+ await execa(command, args, {
29
+ cwd,
30
+ });
31
+ }
32
+ catch {
33
+ errors.push(`Command "${command} ${args.join(' ')}" did not run successfully. Please run this manually in your project.`);
34
+ }
35
+ },
36
+ deleteFile: async (path) => {
37
+ await unlink(path);
38
+ },
39
+ readFile: (path, encoding) => readFile(path, { encoding: encoding || 'utf8' }),
40
+ exists: (path) => existsSync(path),
41
+ readdir: (path) => readdirSync(path),
42
+ isDirectory: (path) => {
43
+ const stat = statSync(path);
44
+ return stat.isDirectory();
45
+ },
46
+ };
47
+ }
48
+ export function createMemoryEnvironment() {
49
+ const environment = createDefaultEnvironment();
50
+ const output = {
51
+ files: {},
52
+ commands: [],
53
+ };
54
+ const { fs, vol } = memfs({});
55
+ const cwd = process.cwd();
56
+ const isInCwd = (path) => path.startsWith(cwd);
57
+ const isTemplatePath = (path) => !isInCwd(path);
58
+ environment.appendFile = async (path, contents) => {
59
+ fs.mkdirSync(dirname(path), { recursive: true });
60
+ await fs.appendFileSync(path, contents);
61
+ };
62
+ environment.copyFile = async (from, to) => {
63
+ if (isTemplatePath(from)) {
64
+ const contents = (await readFile(from)).toString();
65
+ fs.mkdirSync(dirname(to), { recursive: true });
66
+ fs.writeFileSync(to, contents);
67
+ }
68
+ else {
69
+ fs.mkdirSync(dirname(to), { recursive: true });
70
+ fs.copyFileSync(from, to);
71
+ }
72
+ };
73
+ environment.execute = async (command, args) => {
74
+ output.commands.push({
75
+ command,
76
+ args,
77
+ });
78
+ return Promise.resolve();
79
+ };
80
+ environment.readFile = async (path, encoding) => {
81
+ if (isTemplatePath(path)) {
82
+ return (await readFile(path, encoding)).toString();
83
+ }
84
+ return fs.readFileSync(path, 'utf8').toString();
85
+ };
86
+ environment.writeFile = async (path, contents) => {
87
+ fs.mkdirSync(dirname(path), { recursive: true });
88
+ await fs.writeFileSync(path, contents);
89
+ };
90
+ environment.deleteFile = async (path) => {
91
+ await fs.unlinkSync(path);
92
+ };
93
+ environment.exists = (path) => {
94
+ if (isTemplatePath(path)) {
95
+ return existsSync(path);
96
+ }
97
+ return fs.existsSync(path);
98
+ };
99
+ environment.readdir = (path) => {
100
+ if (isTemplatePath(path)) {
101
+ return readdirSync(path);
102
+ }
103
+ return fs.readdirSync(path).map((file) => file.toString());
104
+ };
105
+ environment.isDirectory = (path) => {
106
+ if (isTemplatePath(path)) {
107
+ const stat = statSync(path);
108
+ return stat.isDirectory();
109
+ }
110
+ return fs.statSync(path).isDirectory();
111
+ };
112
+ environment.finishRun = () => {
113
+ output.files = vol.toJSON();
114
+ };
115
+ return {
116
+ environment,
117
+ output,
118
+ };
119
+ }
package/dist/index.js ADDED
@@ -0,0 +1 @@
1
+ export { cli } from './cli.js';
package/dist/mcp.js ADDED
@@ -0,0 +1,211 @@
1
+ import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ import { SSEServerTransport } from '@modelcontextprotocol/sdk/server/sse.js';
3
+ import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
4
+ import express from 'express';
5
+ import { z } from 'zod';
6
+ import { createApp } from './create-app.js';
7
+ import { finalizeAddOns } from './add-ons.js';
8
+ import { createDefaultEnvironment } from './environment.js';
9
+ const server = new McpServer({
10
+ name: 'Demo',
11
+ version: '1.0.0',
12
+ });
13
+ const tanStackReactAddOns = [
14
+ {
15
+ id: 'clerk',
16
+ description: 'Enable authentication with Clerk',
17
+ },
18
+ {
19
+ id: 'convex',
20
+ description: 'Enable a database using Convex',
21
+ },
22
+ {
23
+ id: 'form',
24
+ description: 'Form handling library',
25
+ },
26
+ {
27
+ id: 'netlify',
28
+ description: 'Enable deployments to Netlify',
29
+ },
30
+ {
31
+ id: 'sentry',
32
+ description: 'Enable Sentry error tracking',
33
+ },
34
+ {
35
+ id: 'shadcn',
36
+ description: 'Enable integration of the Shadcn UI component library',
37
+ },
38
+ {
39
+ id: 'start',
40
+ description: 'Set this if you want a TanStack Start application that supports server functions or APIs',
41
+ },
42
+ {
43
+ id: 'tanstack-query',
44
+ description: 'Enable TanStack Query for data fetching',
45
+ },
46
+ {
47
+ id: 'store',
48
+ description: 'Enable the TanStack Store state management library',
49
+ },
50
+ {
51
+ id: 'tanchat',
52
+ description: 'Add an AI chatbot example to the application',
53
+ },
54
+ ];
55
+ server.tool('listTanStackReactAddOns', {}, () => {
56
+ return {
57
+ content: [{ type: 'text', text: JSON.stringify(tanStackReactAddOns) }],
58
+ };
59
+ });
60
+ server.tool('createTanStackReactApplication', {
61
+ projectName: z
62
+ .string()
63
+ .describe('The package.json module name of the application (will also be the directory name)'),
64
+ cwd: z.string().describe('The directory to create the application in'),
65
+ addOns: z
66
+ .array(z.enum([
67
+ 'clerk',
68
+ 'convex',
69
+ 'form',
70
+ 'netlify',
71
+ 'sentry',
72
+ 'shadcn',
73
+ 'start',
74
+ 'store',
75
+ 'tanstack-query',
76
+ 'tanchat',
77
+ ]))
78
+ .describe('The IDs of the add-ons to install'),
79
+ }, async ({ projectName, addOns, cwd }) => {
80
+ try {
81
+ process.chdir(cwd);
82
+ const chosenAddOns = await finalizeAddOns('react', 'file-router', addOns);
83
+ await createApp({
84
+ projectName: projectName.replace(/^\//, './'),
85
+ framework: 'react',
86
+ typescript: true,
87
+ tailwind: true,
88
+ packageManager: 'pnpm',
89
+ toolchain: 'none',
90
+ mode: 'file-router',
91
+ addOns: true,
92
+ chosenAddOns,
93
+ git: true,
94
+ variableValues: {},
95
+ }, {
96
+ silent: true,
97
+ environment: createDefaultEnvironment(),
98
+ });
99
+ return {
100
+ content: [{ type: 'text', text: 'Application created successfully' }],
101
+ };
102
+ }
103
+ catch (error) {
104
+ return {
105
+ content: [
106
+ { type: 'text', text: `Error creating application: ${error}` },
107
+ ],
108
+ };
109
+ }
110
+ });
111
+ const tanStackSolidAddOns = [
112
+ {
113
+ id: 'solid-ui',
114
+ description: 'Enable integration of the Solid UI component library',
115
+ },
116
+ {
117
+ id: 'form',
118
+ description: 'Form handling library',
119
+ },
120
+ {
121
+ id: 'sentry',
122
+ description: 'Enable Sentry error tracking',
123
+ },
124
+ {
125
+ id: 'store',
126
+ description: 'Enable the TanStack Store state management library',
127
+ },
128
+ {
129
+ id: 'tanstack-query',
130
+ description: 'Enable TanStack Query for data fetching',
131
+ },
132
+ {
133
+ id: 'tanchat',
134
+ description: 'Add an AI chatbot example to the application',
135
+ },
136
+ ];
137
+ server.tool('listTanStackSolidAddOns', {}, () => {
138
+ return {
139
+ content: [{ type: 'text', text: JSON.stringify(tanStackSolidAddOns) }],
140
+ };
141
+ });
142
+ server.tool('createTanStackSolidApplication', {
143
+ projectName: z
144
+ .string()
145
+ .describe('The package.json module name of the application (will also be the directory name)'),
146
+ cwd: z.string().describe('The directory to create the application in'),
147
+ addOns: z
148
+ .array(z.enum([
149
+ 'solid-ui',
150
+ 'form',
151
+ 'sentry',
152
+ 'store',
153
+ 'tanstack-query',
154
+ 'tanchat',
155
+ ]))
156
+ .describe('The IDs of the add-ons to install'),
157
+ }, async ({ projectName, addOns, cwd }) => {
158
+ try {
159
+ process.chdir(cwd);
160
+ const chosenAddOns = await finalizeAddOns('solid', 'file-router', addOns);
161
+ await createApp({
162
+ projectName: projectName.replace(/^\//, './'),
163
+ framework: 'solid',
164
+ typescript: true,
165
+ tailwind: true,
166
+ packageManager: 'pnpm',
167
+ toolchain: 'none',
168
+ mode: 'file-router',
169
+ addOns: true,
170
+ chosenAddOns,
171
+ git: true,
172
+ variableValues: {},
173
+ }, {
174
+ silent: true,
175
+ environment: createDefaultEnvironment(),
176
+ });
177
+ return {
178
+ content: [{ type: 'text', text: 'Application created successfully' }],
179
+ };
180
+ }
181
+ catch (error) {
182
+ return {
183
+ content: [
184
+ { type: 'text', text: `Error creating application: ${error}` },
185
+ ],
186
+ };
187
+ }
188
+ });
189
+ export default async function runServer(sse) {
190
+ if (sse) {
191
+ const app = express();
192
+ let transport = null;
193
+ app.get('/sse', (req, res) => {
194
+ transport = new SSEServerTransport('/messages', res);
195
+ server.connect(transport);
196
+ });
197
+ app.post('/messages', (req, res) => {
198
+ if (transport) {
199
+ transport.handlePostMessage(req, res);
200
+ }
201
+ });
202
+ const port = process.env.PORT || 8080;
203
+ app.listen(port, () => {
204
+ console.log(`Server is running on port http://localhost:${port}/sse`);
205
+ });
206
+ }
207
+ else {
208
+ const transport = new StdioServerTransport();
209
+ await server.connect(transport);
210
+ }
211
+ }