@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.
- package/LICENSE +21 -0
- package/dist/add-ons.js +109 -0
- package/dist/add.js +127 -0
- package/dist/cli.js +112 -0
- package/dist/config-file.js +23 -0
- package/dist/constants.js +5 -0
- package/dist/create-app.js +491 -0
- package/dist/custom-add-on.js +254 -0
- package/dist/environment.js +119 -0
- package/dist/index.js +1 -0
- package/dist/mcp.js +211 -0
- package/dist/options.js +309 -0
- package/dist/package-manager.js +30 -0
- package/dist/templates.js +6 -0
- package/dist/toolchain.js +6 -0
- package/dist/types/add-ons.d.ts +5 -0
- package/dist/types/add.d.ts +3 -0
- package/dist/types/cli.d.ts +1 -0
- package/dist/types/config-file.d.ts +7 -0
- package/dist/types/constants.d.ts +6 -0
- package/dist/types/create-app.d.ts +6 -0
- package/dist/types/custom-add-on.d.ts +3 -0
- package/dist/types/environment.d.ts +12 -0
- package/dist/types/index.d.ts +1 -0
- package/dist/types/mcp.d.ts +1 -0
- package/dist/types/options.d.ts +3 -0
- package/dist/types/package-manager.d.ts +6 -0
- package/dist/types/templates.d.ts +1 -0
- package/dist/types/toolchain.d.ts +3 -0
- package/dist/types/types.d.ts +107 -0
- package/dist/types/utils.d.ts +1 -0
- package/dist/types.js +1 -0
- package/dist/utils.js +8 -0
- package/package.json +49 -0
- package/src/add-ons.ts +145 -0
- package/src/add.ts +184 -0
- package/src/cli.ts +163 -0
- package/src/config-file.ts +45 -0
- package/src/constants.ts +9 -0
- package/src/create-app.ts +791 -0
- package/src/custom-add-on.ts +323 -0
- package/src/environment.ts +144 -0
- package/src/index.ts +1 -0
- package/src/mcp.ts +252 -0
- package/src/options.ts +359 -0
- package/src/package-manager.ts +46 -0
- package/src/templates.ts +7 -0
- package/src/toolchain.ts +7 -0
- package/src/types.ts +119 -0
- package/src/utils.ts +10 -0
- package/templates/react/add-on/clerk/README.md +3 -0
- package/templates/react/add-on/clerk/assets/_dot_env.local.append +2 -0
- package/templates/react/add-on/clerk/assets/src/integrations/clerk/header-user.tsx +19 -0
- package/templates/react/add-on/clerk/assets/src/integrations/clerk/provider.tsx +18 -0
- package/templates/react/add-on/clerk/assets/src/routes/demo.clerk.tsx +20 -0
- package/templates/react/add-on/clerk/info.json +13 -0
- package/templates/react/add-on/clerk/package.json +5 -0
- package/templates/react/add-on/convex/README.md +4 -0
- package/templates/react/add-on/convex/assets/_dot_cursorrules.append +93 -0
- package/templates/react/add-on/convex/assets/_dot_env.local.append +3 -0
- package/templates/react/add-on/convex/assets/convex/products.ts +8 -0
- package/templates/react/add-on/convex/assets/convex/schema.ts +10 -0
- package/templates/react/add-on/convex/assets/src/integrations/convex/provider.tsx +20 -0
- package/templates/react/add-on/convex/assets/src/routes/demo.convex.tsx +33 -0
- package/templates/react/add-on/convex/info.json +13 -0
- package/templates/react/add-on/convex/package.json +6 -0
- package/templates/react/add-on/form/assets/src/components/demo.FormComponents.tsx.ejs +300 -0
- package/templates/react/add-on/form/assets/src/hooks/demo.form-context.ts +4 -0
- package/templates/react/add-on/form/assets/src/hooks/demo.form.ts +22 -0
- package/templates/react/add-on/form/assets/src/routes/demo.form.address.tsx.ejs +213 -0
- package/templates/react/add-on/form/assets/src/routes/demo.form.simple.tsx.ejs +77 -0
- package/templates/react/add-on/form/info.json +26 -0
- package/templates/react/add-on/form/package.json +6 -0
- package/templates/react/add-on/module-federation/assets/module-federation.config.js.ejs +31 -0
- package/templates/react/add-on/module-federation/assets/src/demo-mf-component.tsx +3 -0
- package/templates/react/add-on/module-federation/assets/src/demo-mf-self-contained.tsx +11 -0
- package/templates/react/add-on/module-federation/info.json +7 -0
- package/templates/react/add-on/module-federation/package.json +5 -0
- package/templates/react/add-on/netlify/README.md +11 -0
- package/templates/react/add-on/netlify/info.json +7 -0
- package/templates/react/add-on/sentry/assets/_dot_cursorrules.append +22 -0
- package/templates/react/add-on/sentry/assets/_dot_env.local.append +2 -0
- package/templates/react/add-on/sentry/assets/src/app/global-middleware.ts +25 -0
- package/templates/react/add-on/sentry/assets/src/routes/demo.sentry.testing.tsx +480 -0
- package/templates/react/add-on/sentry/info.json +14 -0
- package/templates/react/add-on/sentry/package.json +7 -0
- package/templates/react/add-on/shadcn/README.md +7 -0
- package/templates/react/add-on/shadcn/assets/_dot_cursorrules.append +7 -0
- package/templates/react/add-on/shadcn/assets/components.json +21 -0
- package/templates/react/add-on/shadcn/assets/src/lib/utils.ts +6 -0
- package/templates/react/add-on/shadcn/assets/src/styles.css +138 -0
- package/templates/react/add-on/shadcn/info.json +7 -0
- package/templates/react/add-on/shadcn/package.json +9 -0
- package/templates/react/add-on/start/assets/_dot_gitignore.append +2 -0
- package/templates/react/add-on/start/assets/app.config.ts.ejs +19 -0
- package/templates/react/add-on/start/assets/src/api.ts +6 -0
- package/templates/react/add-on/start/assets/src/client.tsx +8 -0
- package/templates/react/add-on/start/assets/src/router.tsx.ejs +77 -0
- package/templates/react/add-on/start/assets/src/routes/api.demo-names.ts +11 -0
- package/templates/react/add-on/start/assets/src/routes/demo.start.api-request.tsx.ejs +33 -0
- package/templates/react/add-on/start/assets/src/routes/demo.start.server-funcs.tsx +50 -0
- package/templates/react/add-on/start/assets/src/ssr.tsx +12 -0
- package/templates/react/add-on/start/info.json +18 -0
- package/templates/react/add-on/start/package.json +13 -0
- package/templates/react/add-on/store/assets/src/lib/demo-store.ts +13 -0
- package/templates/react/add-on/store/assets/src/routes/demo.store.tsx.ejs +75 -0
- package/templates/react/add-on/store/info.json +13 -0
- package/templates/react/add-on/store/package.json +6 -0
- package/templates/react/add-on/tRPC/assets/src/integrations/trpc/init.ts +9 -0
- package/templates/react/add-on/tRPC/assets/src/integrations/trpc/react.ts +4 -0
- package/templates/react/add-on/tRPC/assets/src/integrations/trpc/router.ts +18 -0
- package/templates/react/add-on/tRPC/assets/src/routes/api.trpc.$.tsx +16 -0
- package/templates/react/add-on/tRPC/info.json +9 -0
- package/templates/react/add-on/tRPC/package.json +9 -0
- package/templates/react/add-on/table/assets/src/data/demo-table-data.ts +50 -0
- package/templates/react/add-on/table/assets/src/routes/demo.table.tsx.ejs +373 -0
- package/templates/react/add-on/table/info.json +13 -0
- package/templates/react/add-on/table/package.json +7 -0
- package/templates/react/add-on/tanstack-query/assets/src/integrations/tanstack-query/layout.tsx +5 -0
- package/templates/react/add-on/tanstack-query/assets/src/integrations/tanstack-query/root-provider.tsx.ejs +70 -0
- package/templates/react/add-on/tanstack-query/assets/src/routes/demo.tanstack-query.tsx.ejs +53 -0
- package/templates/react/add-on/tanstack-query/info.json +13 -0
- package/templates/react/add-on/tanstack-query/package.json +6 -0
- package/templates/react/base/README.md.ejs +558 -0
- package/templates/react/base/_dot_gitignore +5 -0
- package/templates/react/base/_dot_vscode/settings.biome.json +38 -0
- package/templates/react/base/_dot_vscode/settings.json +11 -0
- package/templates/react/base/index.html.ejs +20 -0
- package/templates/react/base/package.biome.json +10 -0
- package/templates/react/base/package.eslintprettier.json +11 -0
- package/templates/react/base/package.json +29 -0
- package/templates/react/base/package.ts.json +7 -0
- package/templates/react/base/package.tw.json +6 -0
- package/templates/react/base/public/favicon.ico +0 -0
- package/templates/react/base/public/logo192.png +0 -0
- package/templates/react/base/public/logo512.png +0 -0
- package/templates/react/base/public/manifest.json +25 -0
- package/templates/react/base/public/robots.txt +3 -0
- package/templates/react/base/src/App.css +38 -0
- package/templates/react/base/src/App.test.tsx.ejs +10 -0
- package/templates/react/base/src/App.tsx.ejs +74 -0
- package/templates/react/base/src/components/Header.tsx.ejs +27 -0
- package/templates/react/base/src/logo.svg +44 -0
- package/templates/react/base/src/reportWebVitals.ts.ejs +28 -0
- package/templates/react/base/src/styles.css.ejs +15 -0
- package/templates/react/base/toolchain/.prettierignore +3 -0
- package/templates/react/base/toolchain/biome.json +31 -0
- package/templates/react/base/toolchain/eslint.config.js +5 -0
- package/templates/react/base/toolchain/prettier.config.js +10 -0
- package/templates/react/base/tsconfig.json.ejs +29 -0
- package/templates/react/base/vite.config.js.ejs +23 -0
- package/templates/react/code-router/src/main.tsx.ejs +92 -0
- package/templates/react/example/tanchat/README.md +37 -0
- package/templates/react/example/tanchat/assets/_dot_env.local.append +2 -0
- package/templates/react/example/tanchat/assets/public/example-guitar-flowers.jpg +0 -0
- package/templates/react/example/tanchat/assets/public/example-guitar-motherboard.jpg +0 -0
- package/templates/react/example/tanchat/assets/public/example-guitar-racing.jpg +0 -0
- package/templates/react/example/tanchat/assets/public/example-guitar-steamer-trunk.jpg +0 -0
- package/templates/react/example/tanchat/assets/public/example-guitar-superhero.jpg +0 -0
- package/templates/react/example/tanchat/assets/public/example-guitar-traveling.jpg +0 -0
- package/templates/react/example/tanchat/assets/public/example-guitar-video-games.jpg +0 -0
- package/templates/react/example/tanchat/assets/src/components/example-AIAssistant.tsx +173 -0
- package/templates/react/example/tanchat/assets/src/components/example-GuitarRecommendation.tsx +47 -0
- package/templates/react/example/tanchat/assets/src/data/example-guitars.ts +83 -0
- package/templates/react/example/tanchat/assets/src/demo.index.css +220 -0
- package/templates/react/example/tanchat/assets/src/integrations/tanchat/header-user.tsx +5 -0
- package/templates/react/example/tanchat/assets/src/routes/example.chat.tsx +159 -0
- package/templates/react/example/tanchat/assets/src/routes/example.guitars/$guitarId.tsx +50 -0
- package/templates/react/example/tanchat/assets/src/routes/example.guitars/index.tsx +54 -0
- package/templates/react/example/tanchat/assets/src/store/example-assistant.ts +3 -0
- package/templates/react/example/tanchat/assets/src/utils/demo.ai.ts +62 -0
- package/templates/react/example/tanchat/assets/src/utils/demo.tools.ts +47 -0
- package/templates/react/example/tanchat/info.json +19 -0
- package/templates/react/example/tanchat/package.json +15 -0
- package/templates/react/file-router/package.fr.json +5 -0
- package/templates/react/file-router/src/main.tsx.ejs +55 -0
- package/templates/react/file-router/src/routes/__root.tsx.ejs +82 -0
- package/templates/solid/add-on/form/assets/src/routes/demo.form.tsx.ejs +352 -0
- package/templates/solid/add-on/form/info.json +13 -0
- package/templates/solid/add-on/form/package.json +5 -0
- package/templates/solid/add-on/module-federation/assets/module-federation.config.js.ejs +27 -0
- package/templates/solid/add-on/module-federation/assets/src/demo-mf-component.tsx +3 -0
- package/templates/solid/add-on/module-federation/assets/src/demo-mf-self-contained.tsx +9 -0
- package/templates/solid/add-on/module-federation/info.json +7 -0
- package/templates/solid/add-on/module-federation/package.json +5 -0
- package/templates/solid/add-on/sentry/assets/_dot_cursorrules.append +22 -0
- package/templates/solid/add-on/sentry/assets/_dot_env.local.append +2 -0
- package/templates/solid/add-on/sentry/assets/src/routes/demo.sentry.bad-event-handler.tsx +20 -0
- package/templates/solid/add-on/sentry/info.json +13 -0
- package/templates/solid/add-on/sentry/package.json +5 -0
- package/templates/solid/add-on/solid-ui/README.md +9 -0
- package/templates/solid/add-on/solid-ui/assets/src/lib/utils.ts +6 -0
- package/templates/solid/add-on/solid-ui/assets/src/styles.css +138 -0
- package/templates/solid/add-on/solid-ui/assets/ui.config.json +13 -0
- package/templates/solid/add-on/solid-ui/info.json +11 -0
- package/templates/solid/add-on/solid-ui/package.json +9 -0
- package/templates/solid/add-on/start/assets/app.config.ts +16 -0
- package/templates/solid/add-on/start/assets/src/api.ts +6 -0
- package/templates/solid/add-on/start/assets/src/client.tsx +7 -0
- package/templates/solid/add-on/start/assets/src/router.tsx.ejs +24 -0
- package/templates/solid/add-on/start/assets/src/routes/demo.start.server-funcs.tsx +49 -0
- package/templates/solid/add-on/start/assets/src/ssr.tsx +12 -0
- package/templates/solid/add-on/start/info.json +14 -0
- package/templates/solid/add-on/start/package.json +12 -0
- package/templates/solid/add-on/store/assets/src/lib/demo-store.ts +13 -0
- package/templates/solid/add-on/store/assets/src/routes/demo.store.tsx.ejs +77 -0
- package/templates/solid/add-on/store/info.json +13 -0
- package/templates/solid/add-on/store/package.json +6 -0
- package/templates/solid/add-on/tanstack-query/assets/src/integrations/tanstack-query/header-user.tsx +5 -0
- package/templates/solid/add-on/tanstack-query/assets/src/integrations/tanstack-query/provider.tsx +15 -0
- package/templates/solid/add-on/tanstack-query/assets/src/routes/demo.tanstack-query.tsx +30 -0
- package/templates/solid/add-on/tanstack-query/info.json +13 -0
- package/templates/solid/add-on/tanstack-query/package.json +6 -0
- package/templates/solid/base/README.md.ejs +215 -0
- package/templates/solid/base/_dot_cursorrules.append +35 -0
- package/templates/solid/base/_dot_gitignore +5 -0
- package/templates/solid/base/_dot_vscode/settings.biome.json +38 -0
- package/templates/solid/base/_dot_vscode/settings.json +11 -0
- package/templates/solid/base/index.html.ejs +20 -0
- package/templates/solid/base/package.biome.json +10 -0
- package/templates/solid/base/package.eslintprettier.json +11 -0
- package/templates/solid/base/package.json +22 -0
- package/templates/solid/base/package.ts.json +5 -0
- package/templates/solid/base/package.tw.json +6 -0
- package/templates/solid/base/public/favicon.ico +0 -0
- package/templates/solid/base/public/logo192.png +0 -0
- package/templates/solid/base/public/logo512.png +0 -0
- package/templates/solid/base/public/manifest.json +25 -0
- package/templates/solid/base/public/robots.txt +3 -0
- package/templates/solid/base/src/App.css +0 -0
- package/templates/solid/base/src/App.tsx.ejs +47 -0
- package/templates/solid/base/src/components/Header.tsx.ejs +26 -0
- package/templates/solid/base/src/logo.svg +120 -0
- package/templates/solid/base/src/styles.css.ejs +15 -0
- package/templates/solid/base/toolchain/.prettierignore +3 -0
- package/templates/solid/base/toolchain/biome.json +31 -0
- package/templates/solid/base/toolchain/eslint.config.js +5 -0
- package/templates/solid/base/toolchain/prettier.config.js +10 -0
- package/templates/solid/base/tsconfig.json.ejs +31 -0
- package/templates/solid/base/vite.config.js.ejs +22 -0
- package/templates/solid/code-router/src/main.tsx.ejs +71 -0
- package/templates/solid/example/tanchat/README.md +52 -0
- package/templates/solid/example/tanchat/assets/ai-streaming-server/README.md +110 -0
- package/templates/solid/example/tanchat/assets/ai-streaming-server/_dot_env.example +1 -0
- package/templates/solid/example/tanchat/assets/ai-streaming-server/package.json +26 -0
- package/templates/solid/example/tanchat/assets/ai-streaming-server/src/index.ts +102 -0
- package/templates/solid/example/tanchat/assets/ai-streaming-server/tsconfig.json +15 -0
- package/templates/solid/example/tanchat/assets/src/components/demo.SettingsDialog.tsx +149 -0
- package/templates/solid/example/tanchat/assets/src/demo.index.css +227 -0
- package/templates/solid/example/tanchat/assets/src/lib/demo-store.ts +13 -0
- package/templates/solid/example/tanchat/assets/src/routes/example.chat.tsx +435 -0
- package/templates/solid/example/tanchat/assets/src/store/demo.hooks.ts +17 -0
- package/templates/solid/example/tanchat/assets/src/store/demo.store.ts +133 -0
- package/templates/solid/example/tanchat/info.json +14 -0
- package/templates/solid/example/tanchat/package.json +7 -0
- package/templates/solid/file-router/package.fr.json +5 -0
- package/templates/solid/file-router/src/main.tsx.ejs +47 -0
- package/templates/solid/file-router/src/routes/__root.tsx.ejs +41 -0
- package/templates/solid/file-router/src/routes/index.tsx +43 -0
- package/tests/cra.test.ts +293 -0
- package/tests/snapshots/cra/cr-js-npm.json +33 -0
- package/tests/snapshots/cra/cr-ts-npm.json +34 -0
- package/tests/snapshots/cra/cr-ts-start-npm.json +38 -0
- package/tests/snapshots/cra/fr-ts-npm.json +34 -0
- package/tests/snapshots/cra/fr-ts-tw-npm.json +33 -0
- package/tests/snapshots/cra/solid-cr-js-npm.json +31 -0
- package/tests/snapshots/cra/solid-cr-ts-npm.json +32 -0
- package/tests/snapshots/cra/solid-cr-ts-start-npm.json +36 -0
- package/tests/snapshots/cra/solid-fr-ts-npm.json +33 -0
- package/tests/snapshots/cra/solid-fr-ts-tw-npm.json +32 -0
- package/tests/test-utilities.ts +87 -0
- package/tsconfig.json +11 -0
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2021-present Tanner Linsley
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/dist/add-ons.js
ADDED
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
import { readFile } from 'node:fs/promises';
|
|
2
|
+
import { existsSync, readFileSync, readdirSync, statSync } from 'node:fs';
|
|
3
|
+
import { resolve } from 'node:path';
|
|
4
|
+
import chalk from 'chalk';
|
|
5
|
+
import { getTemplatesRoot } from './templates.js';
|
|
6
|
+
import { DEFAULT_FRAMEWORK } from './constants.js';
|
|
7
|
+
function isDirectory(path) {
|
|
8
|
+
return statSync(path).isDirectory();
|
|
9
|
+
}
|
|
10
|
+
function findFilesRecursively(path, files) {
|
|
11
|
+
const dirFiles = readdirSync(path);
|
|
12
|
+
for (const file of dirFiles) {
|
|
13
|
+
const filePath = resolve(path, file);
|
|
14
|
+
if (isDirectory(filePath)) {
|
|
15
|
+
findFilesRecursively(filePath, files);
|
|
16
|
+
}
|
|
17
|
+
else {
|
|
18
|
+
files[filePath] = readFileSync(filePath, 'utf-8').toString();
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
export async function getAllAddOns(framework, template) {
|
|
23
|
+
const addOns = [];
|
|
24
|
+
for (const type of ['add-on', 'example']) {
|
|
25
|
+
const addOnsBase = resolve(getTemplatesRoot(), framework, type);
|
|
26
|
+
if (!existsSync(addOnsBase)) {
|
|
27
|
+
continue;
|
|
28
|
+
}
|
|
29
|
+
for (const dir of await readdirSync(addOnsBase).filter((file) => isDirectory(resolve(addOnsBase, file)))) {
|
|
30
|
+
const filePath = resolve(addOnsBase, dir, 'info.json');
|
|
31
|
+
const fileContent = await readFile(filePath, 'utf-8');
|
|
32
|
+
const info = JSON.parse(fileContent);
|
|
33
|
+
if (!info.templates.includes(template)) {
|
|
34
|
+
continue;
|
|
35
|
+
}
|
|
36
|
+
let packageAdditions = {};
|
|
37
|
+
if (existsSync(resolve(addOnsBase, dir, 'package.json'))) {
|
|
38
|
+
packageAdditions = JSON.parse(await readFile(resolve(addOnsBase, dir, 'package.json'), 'utf-8'));
|
|
39
|
+
}
|
|
40
|
+
let readme;
|
|
41
|
+
if (existsSync(resolve(addOnsBase, dir, 'README.md'))) {
|
|
42
|
+
readme = await readFile(resolve(addOnsBase, dir, 'README.md'), 'utf-8');
|
|
43
|
+
}
|
|
44
|
+
const absoluteFiles = {};
|
|
45
|
+
const assetsDir = resolve(addOnsBase, dir, 'assets');
|
|
46
|
+
if (existsSync(assetsDir)) {
|
|
47
|
+
await findFilesRecursively(assetsDir, absoluteFiles);
|
|
48
|
+
}
|
|
49
|
+
const files = {};
|
|
50
|
+
for (const file of Object.keys(absoluteFiles)) {
|
|
51
|
+
files[file.replace(assetsDir, '.')] = absoluteFiles[file];
|
|
52
|
+
}
|
|
53
|
+
addOns.push({
|
|
54
|
+
...info,
|
|
55
|
+
id: dir,
|
|
56
|
+
type,
|
|
57
|
+
packageAdditions,
|
|
58
|
+
readme,
|
|
59
|
+
files,
|
|
60
|
+
deletedFiles: [],
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
return addOns;
|
|
65
|
+
}
|
|
66
|
+
// Turn the list of chosen add-on IDs into a final list of add-ons by resolving dependencies
|
|
67
|
+
export async function finalizeAddOns(framework, template, chosenAddOnIDs) {
|
|
68
|
+
const finalAddOnIDs = new Set(chosenAddOnIDs);
|
|
69
|
+
const addOns = await getAllAddOns(framework, template);
|
|
70
|
+
for (const addOnID of finalAddOnIDs) {
|
|
71
|
+
let addOn;
|
|
72
|
+
const localAddOn = addOns.find((a) => a.id === addOnID);
|
|
73
|
+
if (localAddOn) {
|
|
74
|
+
addOn = loadAddOn(localAddOn);
|
|
75
|
+
}
|
|
76
|
+
else if (addOnID.startsWith('http')) {
|
|
77
|
+
addOn = await loadRemoteAddOn(addOnID);
|
|
78
|
+
addOns.push(addOn);
|
|
79
|
+
}
|
|
80
|
+
else {
|
|
81
|
+
throw new Error(`Add-on ${addOnID} not found`);
|
|
82
|
+
}
|
|
83
|
+
for (const dependsOn of addOn.dependsOn || []) {
|
|
84
|
+
const dep = addOns.find((a) => a.id === dependsOn);
|
|
85
|
+
if (!dep) {
|
|
86
|
+
throw new Error(`Dependency ${dependsOn} not found`);
|
|
87
|
+
}
|
|
88
|
+
finalAddOnIDs.add(dep.id);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
const finalAddOns = [...finalAddOnIDs].map((id) => addOns.find((a) => a.id === id));
|
|
92
|
+
return finalAddOns;
|
|
93
|
+
}
|
|
94
|
+
export async function listAddOns(options) {
|
|
95
|
+
const mode = options.template === 'file-router' ? 'file-router' : 'code-router';
|
|
96
|
+
const addOns = await getAllAddOns(options.framework || DEFAULT_FRAMEWORK, mode);
|
|
97
|
+
for (const addOn of addOns) {
|
|
98
|
+
console.log(`${chalk.bold(addOn.id)}: ${addOn.description}`);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
function loadAddOn(addOn) {
|
|
102
|
+
return addOn;
|
|
103
|
+
}
|
|
104
|
+
export async function loadRemoteAddOn(url) {
|
|
105
|
+
const response = await fetch(url);
|
|
106
|
+
const fileContent = await response.json();
|
|
107
|
+
fileContent.id = url;
|
|
108
|
+
return fileContent;
|
|
109
|
+
}
|
package/dist/add.js
ADDED
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import { mkdir, readFile, writeFile } from 'node:fs/promises';
|
|
2
|
+
import { existsSync, statSync } from 'node:fs';
|
|
3
|
+
import { basename, dirname, resolve } from 'node:path';
|
|
4
|
+
import chalk from 'chalk';
|
|
5
|
+
import { execa, execaSync } from 'execa';
|
|
6
|
+
import { cancel, confirm, intro, isCancel, log, outro, spinner, } from '@clack/prompts';
|
|
7
|
+
import { CONFIG_FILE } from './constants.js';
|
|
8
|
+
import { createDefaultEnvironment, createMemoryEnvironment, } from './environment.js';
|
|
9
|
+
import { createApp } from './create-app.js';
|
|
10
|
+
import { finalizeAddOns } from './add-ons.js';
|
|
11
|
+
import { sortObject } from './utils.js';
|
|
12
|
+
import { readConfigFile, writeConfigFile } from './config-file.js';
|
|
13
|
+
function isDirectory(path) {
|
|
14
|
+
return statSync(path).isDirectory();
|
|
15
|
+
}
|
|
16
|
+
async function hasPendingGitChanges() {
|
|
17
|
+
const status = await execaSync('git', ['status', '--porcelain']);
|
|
18
|
+
return status.stdout.length > 0;
|
|
19
|
+
}
|
|
20
|
+
async function createOptions(json, addOns) {
|
|
21
|
+
return {
|
|
22
|
+
...json,
|
|
23
|
+
tailwind: true,
|
|
24
|
+
chosenAddOns: await finalizeAddOns(json.framework, json.mode, [
|
|
25
|
+
...json.existingAddOns,
|
|
26
|
+
...addOns,
|
|
27
|
+
]),
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
async function runCreateApp(options) {
|
|
31
|
+
const { environment, output } = createMemoryEnvironment();
|
|
32
|
+
await createApp(options, {
|
|
33
|
+
silent: true,
|
|
34
|
+
environment,
|
|
35
|
+
cwd: process.cwd(),
|
|
36
|
+
});
|
|
37
|
+
return output;
|
|
38
|
+
}
|
|
39
|
+
export async function add(addOns, { silent = false, } = {}) {
|
|
40
|
+
const persistedOptions = await readConfigFile(process.cwd());
|
|
41
|
+
if (!persistedOptions) {
|
|
42
|
+
console.error(`${chalk.red('There is no .cta.json file in your project.')}
|
|
43
|
+
|
|
44
|
+
This is probably because this was created with an older version of create-tsrouter-app.`);
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
if (!silent) {
|
|
48
|
+
intro(`Adding ${addOns.join(', ')} to the project...`);
|
|
49
|
+
}
|
|
50
|
+
if (await hasPendingGitChanges()) {
|
|
51
|
+
log.error(`${chalk.red('You have pending git changes.')} Please commit or stash them before adding add-ons.`);
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
const newOptions = await createOptions(persistedOptions, addOns);
|
|
55
|
+
const output = await runCreateApp(newOptions);
|
|
56
|
+
const overwrittenFiles = [];
|
|
57
|
+
const changedFiles = [];
|
|
58
|
+
const contentMap = new Map();
|
|
59
|
+
for (const file of Object.keys(output.files)) {
|
|
60
|
+
const relativeFile = file.replace(process.cwd(), '');
|
|
61
|
+
if (existsSync(file)) {
|
|
62
|
+
if (!isDirectory(file)) {
|
|
63
|
+
const contents = (await readFile(file)).toString();
|
|
64
|
+
if (['package.json', CONFIG_FILE].includes(basename(file)) ||
|
|
65
|
+
contents !== output.files[file]) {
|
|
66
|
+
overwrittenFiles.push(relativeFile);
|
|
67
|
+
contentMap.set(relativeFile, output.files[file]);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
else {
|
|
72
|
+
changedFiles.push(relativeFile);
|
|
73
|
+
contentMap.set(relativeFile, output.files[file]);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
if (overwrittenFiles.length > 0 && !silent) {
|
|
77
|
+
log.warn(`${chalk.yellow('The following will be overwritten:')}\n${overwrittenFiles.join('\n')}`);
|
|
78
|
+
const shouldContinue = await confirm({
|
|
79
|
+
message: 'Do you want to continue?',
|
|
80
|
+
});
|
|
81
|
+
if (isCancel(shouldContinue)) {
|
|
82
|
+
cancel('Operation cancelled.');
|
|
83
|
+
process.exit(0);
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
for (const file of [...changedFiles, ...overwrittenFiles]) {
|
|
87
|
+
const targetFile = `.${file}`;
|
|
88
|
+
const fName = basename(file);
|
|
89
|
+
const contents = contentMap.get(file);
|
|
90
|
+
if (fName === 'package.json') {
|
|
91
|
+
const currentJson = JSON.parse((await readFile(resolve(fName), 'utf-8')).toString());
|
|
92
|
+
const newJson = JSON.parse(contents);
|
|
93
|
+
currentJson.scripts = newJson.scripts;
|
|
94
|
+
currentJson.dependencies = sortObject({
|
|
95
|
+
...currentJson.dependencies,
|
|
96
|
+
...newJson.dependencies,
|
|
97
|
+
});
|
|
98
|
+
currentJson.devDependencies = sortObject({
|
|
99
|
+
...currentJson.devDependencies,
|
|
100
|
+
...newJson.devDependencies,
|
|
101
|
+
});
|
|
102
|
+
await writeFile(targetFile, JSON.stringify(currentJson, null, 2));
|
|
103
|
+
}
|
|
104
|
+
else if (fName !== CONFIG_FILE) {
|
|
105
|
+
await mkdir(resolve(dirname(targetFile)), { recursive: true });
|
|
106
|
+
await writeFile(resolve(targetFile), contents);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
// Handle commands
|
|
110
|
+
const originalOutput = await runCreateApp(await createOptions(persistedOptions, []));
|
|
111
|
+
const originalCommands = new Set(originalOutput.commands.map((c) => [c.command, ...c.args].join(' ')));
|
|
112
|
+
for (const command of output.commands) {
|
|
113
|
+
const commandString = [command.command, ...command.args].join(' ');
|
|
114
|
+
if (!originalCommands.has(commandString)) {
|
|
115
|
+
await execa(command.command, command.args);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
const realEnvironment = createDefaultEnvironment();
|
|
119
|
+
writeConfigFile(realEnvironment, process.cwd(), newOptions);
|
|
120
|
+
const s = silent ? null : spinner();
|
|
121
|
+
s?.start(`Installing dependencies via ${newOptions.packageManager}...`);
|
|
122
|
+
await realEnvironment.execute(newOptions.packageManager, ['install'], resolve(process.cwd()));
|
|
123
|
+
s?.stop(`Installed dependencies`);
|
|
124
|
+
if (!silent) {
|
|
125
|
+
outro('Add-ons added successfully!');
|
|
126
|
+
}
|
|
127
|
+
}
|
package/dist/cli.js
ADDED
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import { Command, InvalidArgumentError } from 'commander';
|
|
2
|
+
import { intro, log } from '@clack/prompts';
|
|
3
|
+
import { createApp } from './create-app.js';
|
|
4
|
+
import { normalizeOptions, promptForOptions } from './options.js';
|
|
5
|
+
import { SUPPORTED_PACKAGE_MANAGERS } from './package-manager.js';
|
|
6
|
+
import { SUPPORTED_TOOLCHAINS } from './toolchain.js';
|
|
7
|
+
import runServer from './mcp.js';
|
|
8
|
+
import { listAddOns } from './add-ons.js';
|
|
9
|
+
import { DEFAULT_FRAMEWORK, SUPPORTED_FRAMEWORKS } from './constants.js';
|
|
10
|
+
// import { initAddOn } from './custom-add-on.js'
|
|
11
|
+
import { createDefaultEnvironment } from './environment.js';
|
|
12
|
+
export function cli() {
|
|
13
|
+
const program = new Command();
|
|
14
|
+
program
|
|
15
|
+
.name('create-tsrouter-app')
|
|
16
|
+
.description('CLI to create a new TanStack application');
|
|
17
|
+
// program
|
|
18
|
+
// .command('add')
|
|
19
|
+
// .argument('add-on', 'Name of the add-on (or add-ons separated by commas)')
|
|
20
|
+
// .action(async (addOn: string) => {
|
|
21
|
+
// await add(addOn.split(',').map((addon) => addon.trim()))
|
|
22
|
+
// })
|
|
23
|
+
// program
|
|
24
|
+
// .command('update-add-on')
|
|
25
|
+
// .description('Create or update an add-on from the current project')
|
|
26
|
+
// .action(async () => {
|
|
27
|
+
// await initAddOn('add-on')
|
|
28
|
+
// })
|
|
29
|
+
// program
|
|
30
|
+
// .command('update-overlay')
|
|
31
|
+
// .description('Create or update a project overlay from the current project')
|
|
32
|
+
// .action(async () => {
|
|
33
|
+
// await initAddOn('overlay')
|
|
34
|
+
// })
|
|
35
|
+
program // 104 22
|
|
36
|
+
.argument('[project-name]', 'name of the project')
|
|
37
|
+
.option('--no-git', 'do not create a git repository')
|
|
38
|
+
.option('--target-dir <path>', 'the directory to create the project in')
|
|
39
|
+
.option('--framework <type>', 'project framework (solid, react)', (value) => {
|
|
40
|
+
if (!SUPPORTED_FRAMEWORKS.includes(value)) {
|
|
41
|
+
throw new InvalidArgumentError(`Invalid framework: ${value}. Only the following are allowed: ${SUPPORTED_FRAMEWORKS.join(', ')}`);
|
|
42
|
+
}
|
|
43
|
+
return value;
|
|
44
|
+
}, DEFAULT_FRAMEWORK)
|
|
45
|
+
.option('--template <type>', 'project template (typescript, javascript, file-router)', (value) => {
|
|
46
|
+
if (value !== 'typescript' &&
|
|
47
|
+
value !== 'javascript' &&
|
|
48
|
+
value !== 'file-router') {
|
|
49
|
+
throw new InvalidArgumentError(`Invalid template: ${value}. Only the following are allowed: typescript, javascript, file-router`);
|
|
50
|
+
}
|
|
51
|
+
return value;
|
|
52
|
+
})
|
|
53
|
+
.option(`--package-manager <${SUPPORTED_PACKAGE_MANAGERS.join('|')}>`, `Explicitly tell the CLI to use this package manager`, (value) => {
|
|
54
|
+
if (!SUPPORTED_PACKAGE_MANAGERS.includes(value)) {
|
|
55
|
+
throw new InvalidArgumentError(`Invalid package manager: ${value}. The following are allowed: ${SUPPORTED_PACKAGE_MANAGERS.join(', ')}`);
|
|
56
|
+
}
|
|
57
|
+
return value;
|
|
58
|
+
})
|
|
59
|
+
.option(`--toolchain <${SUPPORTED_TOOLCHAINS.join('|')}>`, `Explicitly tell the CLI to use this toolchain`, (value) => {
|
|
60
|
+
if (!SUPPORTED_TOOLCHAINS.includes(value)) {
|
|
61
|
+
throw new InvalidArgumentError(`Invalid toolchain: ${value}. The following are allowed: ${SUPPORTED_TOOLCHAINS.join(', ')}`);
|
|
62
|
+
}
|
|
63
|
+
return value;
|
|
64
|
+
})
|
|
65
|
+
.option('--tailwind', 'add Tailwind CSS', false)
|
|
66
|
+
.option('--add-ons [...add-ons]', 'pick from a list of available add-ons (comma separated list)', (value) => {
|
|
67
|
+
let addOns = !!value;
|
|
68
|
+
if (typeof value === 'string') {
|
|
69
|
+
addOns = value.split(',').map((addon) => addon.trim());
|
|
70
|
+
}
|
|
71
|
+
return addOns;
|
|
72
|
+
})
|
|
73
|
+
.option('--list-add-ons', 'list all available add-ons', false)
|
|
74
|
+
.option('--overlay [url]', 'add an overlay from a URL', false)
|
|
75
|
+
.option('--mcp', 'run the MCP server', false)
|
|
76
|
+
.option('--mcp-sse', 'run the MCP server in SSE mode', false)
|
|
77
|
+
.action(async (projectName, options) => {
|
|
78
|
+
if (options.listAddOns) {
|
|
79
|
+
await listAddOns(options);
|
|
80
|
+
}
|
|
81
|
+
else if (options.mcp || options.mcpSse) {
|
|
82
|
+
await runServer(!!options.mcpSse);
|
|
83
|
+
}
|
|
84
|
+
else {
|
|
85
|
+
try {
|
|
86
|
+
const cliOptions = {
|
|
87
|
+
projectName,
|
|
88
|
+
...options,
|
|
89
|
+
};
|
|
90
|
+
let finalOptions = await normalizeOptions(cliOptions);
|
|
91
|
+
if (finalOptions) {
|
|
92
|
+
intro(`Creating a new TanStack app in ${projectName}...`);
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
intro("Let's configure your TanStack application");
|
|
96
|
+
finalOptions = await promptForOptions(cliOptions);
|
|
97
|
+
}
|
|
98
|
+
await createApp(finalOptions, {
|
|
99
|
+
environment: createDefaultEnvironment(),
|
|
100
|
+
cwd: options.targetDir || undefined,
|
|
101
|
+
});
|
|
102
|
+
}
|
|
103
|
+
catch (error) {
|
|
104
|
+
log.error(error instanceof Error
|
|
105
|
+
? error.message
|
|
106
|
+
: 'An unknown error occurred');
|
|
107
|
+
process.exit(1);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
});
|
|
111
|
+
program.parse();
|
|
112
|
+
}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { readFile } from 'node:fs/promises';
|
|
2
|
+
import { resolve } from 'node:path';
|
|
3
|
+
import { CONFIG_FILE } from './constants.js';
|
|
4
|
+
export async function writeConfigFile(environment, targetDir, options) {
|
|
5
|
+
const persistedOptions = {
|
|
6
|
+
...options,
|
|
7
|
+
version: 1,
|
|
8
|
+
existingAddOns: options.chosenAddOns.map((addOn) => addOn.id),
|
|
9
|
+
};
|
|
10
|
+
delete persistedOptions.addOns;
|
|
11
|
+
delete persistedOptions.chosenAddOns;
|
|
12
|
+
await environment.writeFile(resolve(targetDir, CONFIG_FILE), JSON.stringify(persistedOptions, null, 2));
|
|
13
|
+
}
|
|
14
|
+
export async function readConfigFile(targetDir) {
|
|
15
|
+
try {
|
|
16
|
+
const configFile = resolve(targetDir, CONFIG_FILE);
|
|
17
|
+
const config = await readFile(configFile, 'utf8');
|
|
18
|
+
return JSON.parse(config);
|
|
19
|
+
}
|
|
20
|
+
catch {
|
|
21
|
+
return null;
|
|
22
|
+
}
|
|
23
|
+
}
|