@coze-arch/cli 0.0.1-alpha.01c0ee
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/README.md +143 -0
- package/bin/main +2 -0
- package/lib/__templates__/expo/.coze +12 -0
- package/lib/__templates__/expo/.cozeproj/scripts/dev_build.sh +46 -0
- package/lib/__templates__/expo/.cozeproj/scripts/dev_run.sh +229 -0
- package/lib/__templates__/expo/.cozeproj/scripts/prod_build.sh +47 -0
- package/lib/__templates__/expo/.cozeproj/scripts/prod_run.sh +34 -0
- package/lib/__templates__/expo/.cozeproj/scripts/server_dev_run.sh +46 -0
- package/lib/__templates__/expo/README.md +74 -0
- package/lib/__templates__/expo/_gitignore +11 -0
- package/lib/__templates__/expo/_npmrc +20 -0
- package/lib/__templates__/expo/client/app/+not-found.tsx +30 -0
- package/lib/__templates__/expo/client/app/_layout.tsx +36 -0
- package/lib/__templates__/expo/client/app/index.tsx +1 -0
- package/lib/__templates__/expo/client/app.config.ts +76 -0
- package/lib/__templates__/expo/client/assets/fonts/SpaceMono-Regular.ttf +0 -0
- package/lib/__templates__/expo/client/assets/images/adaptive-icon.png +0 -0
- package/lib/__templates__/expo/client/assets/images/default-avatar.png +0 -0
- package/lib/__templates__/expo/client/assets/images/favicon.png +0 -0
- package/lib/__templates__/expo/client/assets/images/icon.png +0 -0
- package/lib/__templates__/expo/client/assets/images/partial-react-logo.png +0 -0
- package/lib/__templates__/expo/client/assets/images/react-logo.png +0 -0
- package/lib/__templates__/expo/client/assets/images/react-logo@2x.png +0 -0
- package/lib/__templates__/expo/client/assets/images/react-logo@3x.png +0 -0
- package/lib/__templates__/expo/client/assets/images/splash-icon.png +0 -0
- package/lib/__templates__/expo/client/components/Screen.tsx +314 -0
- package/lib/__templates__/expo/client/components/SmartDateInput.tsx +238 -0
- package/lib/__templates__/expo/client/components/ThemedText.tsx +33 -0
- package/lib/__templates__/expo/client/components/ThemedView.tsx +37 -0
- package/lib/__templates__/expo/client/constants/theme.ts +177 -0
- package/lib/__templates__/expo/client/contexts/AuthContext.tsx +49 -0
- package/lib/__templates__/expo/client/declarations.d.ts +5 -0
- package/lib/__templates__/expo/client/eslint-formatter-simple.mjs +49 -0
- package/lib/__templates__/expo/client/eslint.config.mjs +128 -0
- package/lib/__templates__/expo/client/hooks/useColorScheme.tsx +48 -0
- package/lib/__templates__/expo/client/hooks/useSafeRouter.ts +152 -0
- package/lib/__templates__/expo/client/hooks/useTheme.ts +33 -0
- package/lib/__templates__/expo/client/metro.config.js +124 -0
- package/lib/__templates__/expo/client/package.json +95 -0
- package/lib/__templates__/expo/client/screens/demo/index.tsx +25 -0
- package/lib/__templates__/expo/client/screens/demo/styles.ts +28 -0
- package/lib/__templates__/expo/client/scripts/install-missing-deps.js +105 -0
- package/lib/__templates__/expo/client/tsconfig.json +24 -0
- package/lib/__templates__/expo/client/utils/index.ts +76 -0
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/names.js +1889 -0
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/rule.js +174 -0
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/v5-only-names.js +388 -0
- package/lib/__templates__/expo/eslint-plugins/forbid-emoji/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/forbid-emoji/rule.js +112 -0
- package/lib/__templates__/expo/eslint-plugins/forbid-emoji/tech.md +94 -0
- package/lib/__templates__/expo/eslint-plugins/react-native/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/react-native/rule.js +64 -0
- package/lib/__templates__/expo/eslint-plugins/reanimated/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/reanimated/rule.js +88 -0
- package/lib/__templates__/expo/eslint-plugins/restrict-linear-gradient/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/restrict-linear-gradient/rule.js +120 -0
- package/lib/__templates__/expo/eslint-plugins/restrict-linear-gradient/tech.md +58 -0
- package/lib/__templates__/expo/package.json +25 -0
- package/lib/__templates__/expo/patches/expo@54.0.33.patch +45 -0
- package/lib/__templates__/expo/pnpm-lock.yaml +12657 -0
- package/lib/__templates__/expo/pnpm-workspace.yaml +3 -0
- package/lib/__templates__/expo/server/build.js +21 -0
- package/lib/__templates__/expo/server/package.json +34 -0
- package/lib/__templates__/expo/server/src/index.ts +20 -0
- package/lib/__templates__/expo/server/tsconfig.json +24 -0
- package/lib/__templates__/expo/template.config.js +106 -0
- package/lib/__templates__/expo/tsconfig.json +1 -0
- package/lib/__templates__/native-static/.coze +11 -0
- package/lib/__templates__/native-static/index.html +33 -0
- package/lib/__templates__/native-static/styles/main.css +136 -0
- package/lib/__templates__/native-static/template.config.js +22 -0
- package/lib/__templates__/nextjs/.babelrc +15 -0
- package/lib/__templates__/nextjs/.coze +12 -0
- package/lib/__templates__/nextjs/README.md +363 -0
- package/lib/__templates__/nextjs/_gitignore +99 -0
- package/lib/__templates__/nextjs/_npmrc +23 -0
- package/lib/__templates__/nextjs/components.json +21 -0
- package/lib/__templates__/nextjs/eslint.config.mjs +23 -0
- package/lib/__templates__/nextjs/next-env.d.ts +6 -0
- package/lib/__templates__/nextjs/next.config.ts +18 -0
- package/lib/__templates__/nextjs/package.json +93 -0
- package/lib/__templates__/nextjs/pnpm-lock.yaml +12404 -0
- package/lib/__templates__/nextjs/postcss.config.mjs +7 -0
- package/lib/__templates__/nextjs/public/file.svg +1 -0
- package/lib/__templates__/nextjs/public/globe.svg +1 -0
- package/lib/__templates__/nextjs/public/next.svg +1 -0
- package/lib/__templates__/nextjs/public/vercel.svg +1 -0
- package/lib/__templates__/nextjs/public/window.svg +1 -0
- package/lib/__templates__/nextjs/scripts/build.sh +17 -0
- package/lib/__templates__/nextjs/scripts/dev.sh +39 -0
- package/lib/__templates__/nextjs/scripts/prepare.sh +9 -0
- package/lib/__templates__/nextjs/scripts/start.sh +21 -0
- package/lib/__templates__/nextjs/src/app/favicon.ico +0 -0
- package/lib/__templates__/nextjs/src/app/globals.css +137 -0
- package/lib/__templates__/nextjs/src/app/layout.tsx +74 -0
- package/lib/__templates__/nextjs/src/app/page.tsx +35 -0
- package/lib/__templates__/nextjs/src/app/robots.ts +11 -0
- package/lib/__templates__/nextjs/src/components/ui/accordion.tsx +66 -0
- package/lib/__templates__/nextjs/src/components/ui/alert-dialog.tsx +157 -0
- package/lib/__templates__/nextjs/src/components/ui/alert.tsx +66 -0
- package/lib/__templates__/nextjs/src/components/ui/aspect-ratio.tsx +11 -0
- package/lib/__templates__/nextjs/src/components/ui/avatar.tsx +53 -0
- package/lib/__templates__/nextjs/src/components/ui/badge.tsx +46 -0
- package/lib/__templates__/nextjs/src/components/ui/breadcrumb.tsx +109 -0
- package/lib/__templates__/nextjs/src/components/ui/button-group.tsx +83 -0
- package/lib/__templates__/nextjs/src/components/ui/button.tsx +62 -0
- package/lib/__templates__/nextjs/src/components/ui/calendar.tsx +220 -0
- package/lib/__templates__/nextjs/src/components/ui/card.tsx +92 -0
- package/lib/__templates__/nextjs/src/components/ui/carousel.tsx +241 -0
- package/lib/__templates__/nextjs/src/components/ui/chart.tsx +357 -0
- package/lib/__templates__/nextjs/src/components/ui/checkbox.tsx +32 -0
- package/lib/__templates__/nextjs/src/components/ui/collapsible.tsx +33 -0
- package/lib/__templates__/nextjs/src/components/ui/command.tsx +184 -0
- package/lib/__templates__/nextjs/src/components/ui/context-menu.tsx +252 -0
- package/lib/__templates__/nextjs/src/components/ui/dialog.tsx +143 -0
- package/lib/__templates__/nextjs/src/components/ui/drawer.tsx +135 -0
- package/lib/__templates__/nextjs/src/components/ui/dropdown-menu.tsx +257 -0
- package/lib/__templates__/nextjs/src/components/ui/empty.tsx +104 -0
- package/lib/__templates__/nextjs/src/components/ui/field.tsx +248 -0
- package/lib/__templates__/nextjs/src/components/ui/form.tsx +167 -0
- package/lib/__templates__/nextjs/src/components/ui/hover-card.tsx +44 -0
- package/lib/__templates__/nextjs/src/components/ui/input-group.tsx +170 -0
- package/lib/__templates__/nextjs/src/components/ui/input-otp.tsx +77 -0
- package/lib/__templates__/nextjs/src/components/ui/input.tsx +21 -0
- package/lib/__templates__/nextjs/src/components/ui/item.tsx +193 -0
- package/lib/__templates__/nextjs/src/components/ui/kbd.tsx +28 -0
- package/lib/__templates__/nextjs/src/components/ui/label.tsx +24 -0
- package/lib/__templates__/nextjs/src/components/ui/menubar.tsx +276 -0
- package/lib/__templates__/nextjs/src/components/ui/navigation-menu.tsx +168 -0
- package/lib/__templates__/nextjs/src/components/ui/pagination.tsx +127 -0
- package/lib/__templates__/nextjs/src/components/ui/popover.tsx +48 -0
- package/lib/__templates__/nextjs/src/components/ui/progress.tsx +31 -0
- package/lib/__templates__/nextjs/src/components/ui/radio-group.tsx +45 -0
- package/lib/__templates__/nextjs/src/components/ui/resizable.tsx +63 -0
- package/lib/__templates__/nextjs/src/components/ui/scroll-area.tsx +58 -0
- package/lib/__templates__/nextjs/src/components/ui/select.tsx +190 -0
- package/lib/__templates__/nextjs/src/components/ui/separator.tsx +28 -0
- package/lib/__templates__/nextjs/src/components/ui/sheet.tsx +139 -0
- package/lib/__templates__/nextjs/src/components/ui/sidebar.tsx +724 -0
- package/lib/__templates__/nextjs/src/components/ui/skeleton.tsx +13 -0
- package/lib/__templates__/nextjs/src/components/ui/slider.tsx +63 -0
- package/lib/__templates__/nextjs/src/components/ui/sonner.tsx +40 -0
- package/lib/__templates__/nextjs/src/components/ui/spinner.tsx +16 -0
- package/lib/__templates__/nextjs/src/components/ui/switch.tsx +31 -0
- package/lib/__templates__/nextjs/src/components/ui/table.tsx +116 -0
- package/lib/__templates__/nextjs/src/components/ui/tabs.tsx +66 -0
- package/lib/__templates__/nextjs/src/components/ui/textarea.tsx +18 -0
- package/lib/__templates__/nextjs/src/components/ui/toggle-group.tsx +83 -0
- package/lib/__templates__/nextjs/src/components/ui/toggle.tsx +47 -0
- package/lib/__templates__/nextjs/src/components/ui/tooltip.tsx +61 -0
- package/lib/__templates__/nextjs/src/hooks/use-mobile.ts +19 -0
- package/lib/__templates__/nextjs/src/lib/utils.ts +6 -0
- package/lib/__templates__/nextjs/src/server.ts +35 -0
- package/lib/__templates__/nextjs/template.config.js +120 -0
- package/lib/__templates__/nextjs/tsconfig.json +34 -0
- package/lib/__templates__/nuxt-vue/.coze +12 -0
- package/lib/__templates__/nuxt-vue/README.md +73 -0
- package/lib/__templates__/nuxt-vue/_gitignore +24 -0
- package/lib/__templates__/nuxt-vue/_npmrc +23 -0
- package/lib/__templates__/nuxt-vue/app/app.vue +6 -0
- package/lib/__templates__/nuxt-vue/app/pages/index.vue +23 -0
- package/lib/__templates__/nuxt-vue/assets/css/main.css +24 -0
- package/lib/__templates__/nuxt-vue/nuxt.config.ts +116 -0
- package/lib/__templates__/nuxt-vue/package.json +35 -0
- package/lib/__templates__/nuxt-vue/pnpm-lock.yaml +8759 -0
- package/lib/__templates__/nuxt-vue/postcss.config.mjs +8 -0
- package/lib/__templates__/nuxt-vue/public/favicon.ico +0 -0
- package/lib/__templates__/nuxt-vue/public/robots.txt +2 -0
- package/lib/__templates__/nuxt-vue/scripts/build.sh +14 -0
- package/lib/__templates__/nuxt-vue/scripts/dev.sh +39 -0
- package/lib/__templates__/nuxt-vue/scripts/prepare.sh +14 -0
- package/lib/__templates__/nuxt-vue/scripts/start.sh +21 -0
- package/lib/__templates__/nuxt-vue/server/api/hello.ts +10 -0
- package/lib/__templates__/nuxt-vue/server/middleware/logger.ts +10 -0
- package/lib/__templates__/nuxt-vue/server/routes/health.ts +10 -0
- package/lib/__templates__/nuxt-vue/tailwind.config.js +13 -0
- package/lib/__templates__/nuxt-vue/template.config.js +87 -0
- package/lib/__templates__/nuxt-vue/tsconfig.json +18 -0
- package/lib/__templates__/taro/.coze +14 -0
- package/lib/__templates__/taro/.cozeproj/scripts/deploy_build.sh +19 -0
- package/lib/__templates__/taro/.cozeproj/scripts/deploy_run.sh +14 -0
- package/lib/__templates__/taro/.cozeproj/scripts/dev_build.sh +2 -0
- package/lib/__templates__/taro/.cozeproj/scripts/dev_run.sh +151 -0
- package/lib/__templates__/taro/.cozeproj/scripts/init_env.sh +5 -0
- package/lib/__templates__/taro/.cozeproj/scripts/pack.sh +24 -0
- package/lib/__templates__/taro/README.md +763 -0
- package/lib/__templates__/taro/_gitignore +40 -0
- package/lib/__templates__/taro/_npmrc +18 -0
- package/lib/__templates__/taro/babel.config.js +12 -0
- package/lib/__templates__/taro/config/dev.ts +9 -0
- package/lib/__templates__/taro/config/index.ts +237 -0
- package/lib/__templates__/taro/config/prod.ts +34 -0
- package/lib/__templates__/taro/eslint.config.mjs +135 -0
- package/lib/__templates__/taro/key/private.appid.key +0 -0
- package/lib/__templates__/taro/package.json +112 -0
- package/lib/__templates__/taro/patches/@tarojs__plugin-mini-ci@4.1.9.patch +30 -0
- package/lib/__templates__/taro/pnpm-lock.yaml +23412 -0
- package/lib/__templates__/taro/pnpm-workspace.yaml +2 -0
- package/lib/__templates__/taro/project.config.json +15 -0
- package/lib/__templates__/taro/server/nest-cli.json +10 -0
- package/lib/__templates__/taro/server/package.json +40 -0
- package/lib/__templates__/taro/server/src/app.controller.ts +23 -0
- package/lib/__templates__/taro/server/src/app.module.ts +10 -0
- package/lib/__templates__/taro/server/src/app.service.ts +8 -0
- package/lib/__templates__/taro/server/src/interceptors/http-status.interceptor.ts +23 -0
- package/lib/__templates__/taro/server/src/main.ts +49 -0
- package/lib/__templates__/taro/server/tsconfig.json +24 -0
- package/lib/__templates__/taro/src/app.config.ts +11 -0
- package/lib/__templates__/taro/src/app.css +156 -0
- package/lib/__templates__/taro/src/app.tsx +9 -0
- package/lib/__templates__/taro/src/components/ui/accordion.tsx +159 -0
- package/lib/__templates__/taro/src/components/ui/alert-dialog.tsx +260 -0
- package/lib/__templates__/taro/src/components/ui/alert.tsx +60 -0
- package/lib/__templates__/taro/src/components/ui/aspect-ratio.tsx +36 -0
- package/lib/__templates__/taro/src/components/ui/avatar.tsx +84 -0
- package/lib/__templates__/taro/src/components/ui/badge.tsx +37 -0
- package/lib/__templates__/taro/src/components/ui/breadcrumb.tsx +117 -0
- package/lib/__templates__/taro/src/components/ui/button-group.tsx +83 -0
- package/lib/__templates__/taro/src/components/ui/button.tsx +67 -0
- package/lib/__templates__/taro/src/components/ui/calendar.tsx +394 -0
- package/lib/__templates__/taro/src/components/ui/card.tsx +108 -0
- package/lib/__templates__/taro/src/components/ui/carousel.tsx +228 -0
- package/lib/__templates__/taro/src/components/ui/checkbox.tsx +58 -0
- package/lib/__templates__/taro/src/components/ui/code-block.tsx +169 -0
- package/lib/__templates__/taro/src/components/ui/collapsible.tsx +71 -0
- package/lib/__templates__/taro/src/components/ui/command.tsx +385 -0
- package/lib/__templates__/taro/src/components/ui/context-menu.tsx +614 -0
- package/lib/__templates__/taro/src/components/ui/dialog.tsx +256 -0
- package/lib/__templates__/taro/src/components/ui/drawer.tsx +192 -0
- package/lib/__templates__/taro/src/components/ui/dropdown-menu.tsx +561 -0
- package/lib/__templates__/taro/src/components/ui/field.tsx +228 -0
- package/lib/__templates__/taro/src/components/ui/hover-card.tsx +282 -0
- package/lib/__templates__/taro/src/components/ui/input-group.tsx +197 -0
- package/lib/__templates__/taro/src/components/ui/input-otp.tsx +136 -0
- package/lib/__templates__/taro/src/components/ui/input.tsx +56 -0
- package/lib/__templates__/taro/src/components/ui/label.tsx +24 -0
- package/lib/__templates__/taro/src/components/ui/menubar.tsx +595 -0
- package/lib/__templates__/taro/src/components/ui/navigation-menu.tsx +264 -0
- package/lib/__templates__/taro/src/components/ui/pagination.tsx +118 -0
- package/lib/__templates__/taro/src/components/ui/popover.tsx +291 -0
- package/lib/__templates__/taro/src/components/ui/portal.tsx +19 -0
- package/lib/__templates__/taro/src/components/ui/progress.tsx +28 -0
- package/lib/__templates__/taro/src/components/ui/radio-group.tsx +64 -0
- package/lib/__templates__/taro/src/components/ui/resizable.tsx +346 -0
- package/lib/__templates__/taro/src/components/ui/scroll-area.tsx +34 -0
- package/lib/__templates__/taro/src/components/ui/select.tsx +438 -0
- package/lib/__templates__/taro/src/components/ui/separator.tsx +30 -0
- package/lib/__templates__/taro/src/components/ui/sheet.tsx +262 -0
- package/lib/__templates__/taro/src/components/ui/skeleton.tsx +17 -0
- package/lib/__templates__/taro/src/components/ui/slider.tsx +203 -0
- package/lib/__templates__/taro/src/components/ui/sonner.tsx +1 -0
- package/lib/__templates__/taro/src/components/ui/switch.tsx +55 -0
- package/lib/__templates__/taro/src/components/ui/table.tsx +142 -0
- package/lib/__templates__/taro/src/components/ui/tabs.tsx +114 -0
- package/lib/__templates__/taro/src/components/ui/textarea.tsx +54 -0
- package/lib/__templates__/taro/src/components/ui/toast.tsx +517 -0
- package/lib/__templates__/taro/src/components/ui/toggle-group.tsx +120 -0
- package/lib/__templates__/taro/src/components/ui/toggle.tsx +77 -0
- package/lib/__templates__/taro/src/components/ui/tooltip.tsx +455 -0
- package/lib/__templates__/taro/src/index.html +39 -0
- package/lib/__templates__/taro/src/lib/hooks/use-keyboard-offset.ts +37 -0
- package/lib/__templates__/taro/src/lib/measure.ts +115 -0
- package/lib/__templates__/taro/src/lib/platform.ts +12 -0
- package/lib/__templates__/taro/src/lib/utils.ts +6 -0
- package/lib/__templates__/taro/src/network.ts +39 -0
- package/lib/__templates__/taro/src/pages/index/index.config.ts +3 -0
- package/lib/__templates__/taro/src/pages/index/index.css +1 -0
- package/lib/__templates__/taro/src/pages/index/index.tsx +33 -0
- package/lib/__templates__/taro/src/presets/dev-debug.ts +23 -0
- package/lib/__templates__/taro/src/presets/h5-container.tsx +15 -0
- package/lib/__templates__/taro/src/presets/h5-navbar.tsx +238 -0
- package/lib/__templates__/taro/src/presets/h5-styles.ts +220 -0
- package/lib/__templates__/taro/src/presets/index.tsx +18 -0
- package/lib/__templates__/taro/stylelint.config.mjs +4 -0
- package/lib/__templates__/taro/template.config.js +68 -0
- package/lib/__templates__/taro/tsconfig.json +29 -0
- package/lib/__templates__/taro/types/global.d.ts +32 -0
- package/lib/__templates__/templates.json +162 -0
- package/lib/__templates__/vite/.coze +12 -0
- package/lib/__templates__/vite/README.md +418 -0
- package/lib/__templates__/vite/_gitignore +67 -0
- package/lib/__templates__/vite/_npmrc +23 -0
- package/lib/__templates__/vite/eslint.config.mjs +14 -0
- package/lib/__templates__/vite/index.html +13 -0
- package/lib/__templates__/vite/package.json +44 -0
- package/lib/__templates__/vite/pnpm-lock.yaml +3465 -0
- package/lib/__templates__/vite/postcss.config.js +6 -0
- package/lib/__templates__/vite/scripts/build.sh +17 -0
- package/lib/__templates__/vite/scripts/dev.sh +39 -0
- package/lib/__templates__/vite/scripts/prepare.sh +9 -0
- package/lib/__templates__/vite/scripts/start.sh +21 -0
- package/lib/__templates__/vite/server/routes/index.ts +31 -0
- package/lib/__templates__/vite/server/server.ts +65 -0
- package/lib/__templates__/vite/server/vite.ts +67 -0
- package/lib/__templates__/vite/src/index.css +21 -0
- package/lib/__templates__/vite/src/index.ts +5 -0
- package/lib/__templates__/vite/src/main.ts +34 -0
- package/lib/__templates__/vite/tailwind.config.js +9 -0
- package/lib/__templates__/vite/template.config.js +127 -0
- package/lib/__templates__/vite/tsconfig.json +17 -0
- package/lib/__templates__/vite/vite.config.ts +20 -0
- package/lib/cli.js +2631 -0
- package/package.json +81 -0
package/lib/cli.js
ADDED
|
@@ -0,0 +1,2631 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
var commander = require('commander');
|
|
5
|
+
var path = require('path');
|
|
6
|
+
var fs = require('fs');
|
|
7
|
+
var node_path = require('node:path');
|
|
8
|
+
var node_fs = require('node:fs');
|
|
9
|
+
var shelljs = require('shelljs');
|
|
10
|
+
var perf_hooks = require('perf_hooks');
|
|
11
|
+
var fs$1 = require('fs/promises');
|
|
12
|
+
var os = require('os');
|
|
13
|
+
var jsYaml = require('js-yaml');
|
|
14
|
+
var toml = require('@iarna/toml');
|
|
15
|
+
var child_process = require('child_process');
|
|
16
|
+
var addFormats = require('ajv-formats');
|
|
17
|
+
var Ajv = require('ajv');
|
|
18
|
+
var minimist = require('minimist');
|
|
19
|
+
var changeCase = require('change-case');
|
|
20
|
+
var ejs = require('ejs');
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* 读取 templates.json 配置
|
|
24
|
+
*/
|
|
25
|
+
const loadTemplatesConfig$1 = () => {
|
|
26
|
+
// 尝试多个可能的路径
|
|
27
|
+
const possiblePaths = [
|
|
28
|
+
// 生产模式:lib/cli.js 打包后
|
|
29
|
+
path.join(__dirname, '__templates__/templates.json'),
|
|
30
|
+
// 开发模式:src/utils/template-help.ts
|
|
31
|
+
path.join(__dirname, '../__templates__/templates.json'),
|
|
32
|
+
];
|
|
33
|
+
|
|
34
|
+
for (const templatesPath of possiblePaths) {
|
|
35
|
+
try {
|
|
36
|
+
const content = fs.readFileSync(templatesPath, 'utf-8');
|
|
37
|
+
// eslint-disable-next-line no-restricted-syntax
|
|
38
|
+
return JSON.parse(content) ;
|
|
39
|
+
} catch (error) {
|
|
40
|
+
// 继续尝试下一个路径
|
|
41
|
+
continue;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
// 所有路径都失败,返回 null
|
|
46
|
+
return null;
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
/**
|
|
50
|
+
* 格式化单个参数信息
|
|
51
|
+
*/
|
|
52
|
+
const formatParam = (paramName, param) => {
|
|
53
|
+
const parts = [];
|
|
54
|
+
|
|
55
|
+
// 参数名和类型
|
|
56
|
+
parts.push(` --${paramName} <${param.type}>`);
|
|
57
|
+
|
|
58
|
+
// 描述
|
|
59
|
+
if (param.description) {
|
|
60
|
+
parts.push(` ${param.description}`);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// 默认值
|
|
64
|
+
if (param.default !== undefined) {
|
|
65
|
+
parts.push(` Default: ${param.default}`);
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
// 范围限制
|
|
69
|
+
if (param.minimum !== undefined || param.maximum !== undefined) {
|
|
70
|
+
const range = [];
|
|
71
|
+
if (param.minimum !== undefined) {
|
|
72
|
+
range.push(`min: ${param.minimum}`);
|
|
73
|
+
}
|
|
74
|
+
if (param.maximum !== undefined) {
|
|
75
|
+
range.push(`max: ${param.maximum}`);
|
|
76
|
+
}
|
|
77
|
+
parts.push(` Range: ${range.join(', ')}`);
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// 模式限制
|
|
81
|
+
if (param.pattern) {
|
|
82
|
+
parts.push(` Pattern: ${param.pattern}`);
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
return parts.join('\n');
|
|
86
|
+
};
|
|
87
|
+
|
|
88
|
+
/**
|
|
89
|
+
* 生成模板帮助文本
|
|
90
|
+
*/
|
|
91
|
+
const generateTemplatesHelpText = () => {
|
|
92
|
+
const config = loadTemplatesConfig$1();
|
|
93
|
+
|
|
94
|
+
if (!config || config.templates.length === 0) {
|
|
95
|
+
return '';
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
const lines = [];
|
|
99
|
+
|
|
100
|
+
lines.push('\n📦 Available Templates:\n');
|
|
101
|
+
lines.push(` ${'='.repeat(76)}`);
|
|
102
|
+
|
|
103
|
+
config.templates.forEach((template, index) => {
|
|
104
|
+
if (index > 0) {
|
|
105
|
+
lines.push(` ${'-'.repeat(76)}`);
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
lines.push(`\n Template: ${template.name}`);
|
|
109
|
+
lines.push(` Description: ${template.description}`);
|
|
110
|
+
|
|
111
|
+
const params = Object.entries(template.paramsSchema);
|
|
112
|
+
if (params.length > 0) {
|
|
113
|
+
lines.push(' Parameters:');
|
|
114
|
+
params.forEach(([paramName, param]) => {
|
|
115
|
+
lines.push(formatParam(paramName, param));
|
|
116
|
+
});
|
|
117
|
+
}
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
lines.push(`\n ${'='.repeat(76)}`);
|
|
121
|
+
lines.push('\n Usage:');
|
|
122
|
+
lines.push(
|
|
123
|
+
' coze init <directory> -t <template-name> [--param value ...]',
|
|
124
|
+
);
|
|
125
|
+
lines.push('\n Example:');
|
|
126
|
+
lines.push(' coze init my-app -t nextjs --appName my-app --port 3000');
|
|
127
|
+
lines.push('');
|
|
128
|
+
|
|
129
|
+
return lines.join('\n');
|
|
130
|
+
};
|
|
131
|
+
|
|
132
|
+
function _nullishCoalesce$2(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } function _optionalChain$3(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }var LogLevel; (function (LogLevel) {
|
|
133
|
+
const ERROR = 0; LogLevel[LogLevel["ERROR"] = ERROR] = "ERROR";
|
|
134
|
+
const WARN = 1; LogLevel[LogLevel["WARN"] = WARN] = "WARN";
|
|
135
|
+
const SUCCESS = 2; LogLevel[LogLevel["SUCCESS"] = SUCCESS] = "SUCCESS";
|
|
136
|
+
const INFO = 3; LogLevel[LogLevel["INFO"] = INFO] = "INFO";
|
|
137
|
+
const VERBOSE = 4; LogLevel[LogLevel["VERBOSE"] = VERBOSE] = "VERBOSE";
|
|
138
|
+
})(LogLevel || (LogLevel = {}));
|
|
139
|
+
|
|
140
|
+
|
|
141
|
+
|
|
142
|
+
|
|
143
|
+
|
|
144
|
+
|
|
145
|
+
|
|
146
|
+
const LOG_LEVEL_MAP = {
|
|
147
|
+
error: LogLevel.ERROR,
|
|
148
|
+
warn: LogLevel.WARN,
|
|
149
|
+
success: LogLevel.SUCCESS,
|
|
150
|
+
info: LogLevel.INFO,
|
|
151
|
+
verbose: LogLevel.VERBOSE,
|
|
152
|
+
};
|
|
153
|
+
|
|
154
|
+
const COLOR_CODES = {
|
|
155
|
+
reset: '\x1b[0m',
|
|
156
|
+
red: '\x1b[31m',
|
|
157
|
+
yellow: '\x1b[33m',
|
|
158
|
+
green: '\x1b[32m',
|
|
159
|
+
cyan: '\x1b[36m',
|
|
160
|
+
gray: '\x1b[90m',
|
|
161
|
+
};
|
|
162
|
+
|
|
163
|
+
class Logger {
|
|
164
|
+
|
|
165
|
+
|
|
166
|
+
|
|
167
|
+
|
|
168
|
+
constructor(options = {}) {
|
|
169
|
+
this.level = this.parseLogLevel(options.level);
|
|
170
|
+
this.useColor = _nullishCoalesce$2(options.useColor, () => ( this.isColorSupported()));
|
|
171
|
+
this.prefix = options.prefix;
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
parseLogLevel(level) {
|
|
175
|
+
if (level !== undefined) {
|
|
176
|
+
return level;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
const envLevel = _optionalChain$3([process, 'access', _ => _.env, 'access', _2 => _2.LOG_LEVEL, 'optionalAccess', _3 => _3.toLowerCase, 'call', _4 => _4()]);
|
|
180
|
+
if (envLevel && envLevel in LOG_LEVEL_MAP) {
|
|
181
|
+
return LOG_LEVEL_MAP[envLevel];
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
return LogLevel.INFO;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
isColorSupported() {
|
|
188
|
+
// 简单检测:Node.js 环境且支持 TTY
|
|
189
|
+
return (
|
|
190
|
+
typeof process !== 'undefined' &&
|
|
191
|
+
_optionalChain$3([process, 'access', _5 => _5.stdout, 'optionalAccess', _6 => _6.isTTY]) === true &&
|
|
192
|
+
process.env.NO_COLOR === undefined
|
|
193
|
+
);
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
colorize(text, color) {
|
|
197
|
+
if (!this.useColor) {
|
|
198
|
+
return text;
|
|
199
|
+
}
|
|
200
|
+
return `${COLOR_CODES[color]}${text}${COLOR_CODES.reset}`;
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
log(options
|
|
204
|
+
|
|
205
|
+
|
|
206
|
+
|
|
207
|
+
|
|
208
|
+
|
|
209
|
+
) {
|
|
210
|
+
if (options.level > this.level) {
|
|
211
|
+
return;
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
const icon = this.colorize(options.icon, options.color);
|
|
215
|
+
const prefix = this.prefix ? `${icon} ${this.prefix}` : icon;
|
|
216
|
+
console.log(prefix, options.message, ...(_nullishCoalesce$2(options.args, () => ( []))));
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
error(message, ...args) {
|
|
220
|
+
this.log({
|
|
221
|
+
level: LogLevel.ERROR,
|
|
222
|
+
icon: '✖',
|
|
223
|
+
color: 'red',
|
|
224
|
+
message,
|
|
225
|
+
args,
|
|
226
|
+
});
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
warn(message, ...args) {
|
|
230
|
+
this.log({
|
|
231
|
+
level: LogLevel.WARN,
|
|
232
|
+
icon: '⚠',
|
|
233
|
+
color: 'yellow',
|
|
234
|
+
message,
|
|
235
|
+
args,
|
|
236
|
+
});
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
success(message, ...args) {
|
|
240
|
+
this.log({
|
|
241
|
+
level: LogLevel.SUCCESS,
|
|
242
|
+
icon: '✓',
|
|
243
|
+
color: 'green',
|
|
244
|
+
message,
|
|
245
|
+
args,
|
|
246
|
+
});
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
info(message, ...args) {
|
|
250
|
+
this.log({
|
|
251
|
+
level: LogLevel.INFO,
|
|
252
|
+
icon: 'ℹ',
|
|
253
|
+
color: 'cyan',
|
|
254
|
+
message,
|
|
255
|
+
args,
|
|
256
|
+
});
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
verbose(message, ...args) {
|
|
260
|
+
this.log({
|
|
261
|
+
level: LogLevel.VERBOSE,
|
|
262
|
+
icon: '→',
|
|
263
|
+
color: 'gray',
|
|
264
|
+
message,
|
|
265
|
+
args,
|
|
266
|
+
});
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
// 创建 logger 实例的工厂函数
|
|
271
|
+
const createLogger = (options = {}) =>
|
|
272
|
+
new Logger(options);
|
|
273
|
+
|
|
274
|
+
// 导出默认实例
|
|
275
|
+
const logger = createLogger();
|
|
276
|
+
|
|
277
|
+
/**
|
|
278
|
+
* 时间统计工具
|
|
279
|
+
*/
|
|
280
|
+
class TimeTracker {
|
|
281
|
+
|
|
282
|
+
|
|
283
|
+
|
|
284
|
+
constructor() {
|
|
285
|
+
this.startTime = perf_hooks.performance.now();
|
|
286
|
+
this.lastTime = this.startTime;
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
/**
|
|
290
|
+
* 记录阶段耗时
|
|
291
|
+
* @param phaseName 阶段名称
|
|
292
|
+
*/
|
|
293
|
+
logPhase(phaseName) {
|
|
294
|
+
const now = perf_hooks.performance.now();
|
|
295
|
+
const phaseTime = now - this.lastTime;
|
|
296
|
+
this.lastTime = now;
|
|
297
|
+
|
|
298
|
+
logger.verbose(`⏱ ${phaseName}: ${phaseTime.toFixed(2)}ms`);
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
/**
|
|
302
|
+
* 记录总耗时
|
|
303
|
+
*/
|
|
304
|
+
logTotal() {
|
|
305
|
+
const totalTime = perf_hooks.performance.now() - this.startTime;
|
|
306
|
+
logger.verbose(`⏱ Total time: ${totalTime.toFixed(2)}ms`);
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
/**
|
|
310
|
+
* 获取当前耗时(不输出日志)
|
|
311
|
+
* @returns 从开始到现在的总耗时(毫秒)
|
|
312
|
+
*/
|
|
313
|
+
getElapsedTime() {
|
|
314
|
+
return perf_hooks.performance.now() - this.startTime;
|
|
315
|
+
}
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
/**
|
|
319
|
+
* 获取模板配置文件路径
|
|
320
|
+
* @returns templates.json 的绝对路径
|
|
321
|
+
*/
|
|
322
|
+
const getTemplatesConfigPath = () => {
|
|
323
|
+
const configPath = path.resolve(getTemplatesDir(), 'templates.json');
|
|
324
|
+
logger.verbose(`Templates config path: ${configPath}`);
|
|
325
|
+
logger.verbose(`Config file exists: ${fs.existsSync(configPath)}`);
|
|
326
|
+
return configPath;
|
|
327
|
+
};
|
|
328
|
+
|
|
329
|
+
/**
|
|
330
|
+
* 获取模板目录路径
|
|
331
|
+
* @returns __templates__ 目录的绝对路径
|
|
332
|
+
*/
|
|
333
|
+
const getTemplatesDir = () => {
|
|
334
|
+
const templatesDir = path.resolve(__dirname, './__templates__');
|
|
335
|
+
logger.verbose(`Templates directory: ${templatesDir}`);
|
|
336
|
+
logger.verbose(`Templates directory exists: ${fs.existsSync(templatesDir)}`);
|
|
337
|
+
return templatesDir;
|
|
338
|
+
};
|
|
339
|
+
|
|
340
|
+
/**
|
|
341
|
+
* 加载模板配置文件
|
|
342
|
+
* 支持 .ts 和 .js 文件(通过 sucrase 注册)
|
|
343
|
+
*
|
|
344
|
+
* @param templatePath - 模板目录路径
|
|
345
|
+
* @returns 模板配置对象
|
|
346
|
+
*/
|
|
347
|
+
|
|
348
|
+
const loadTemplateConfig = async (
|
|
349
|
+
templatePath,
|
|
350
|
+
) => {
|
|
351
|
+
logger.verbose(`Loading template config from: ${templatePath}`);
|
|
352
|
+
|
|
353
|
+
const tsConfigPath = path.join(templatePath, 'template.config.ts');
|
|
354
|
+
const jsConfigPath = path.join(templatePath, 'template.config.js');
|
|
355
|
+
|
|
356
|
+
logger.verbose('Checking for config files:');
|
|
357
|
+
logger.verbose(` - TypeScript: ${tsConfigPath}`);
|
|
358
|
+
logger.verbose(` - JavaScript: ${jsConfigPath}`);
|
|
359
|
+
|
|
360
|
+
let configPath;
|
|
361
|
+
|
|
362
|
+
const [tsExists, jsExists] = await Promise.all([
|
|
363
|
+
fs$1.access(tsConfigPath).then(
|
|
364
|
+
() => true,
|
|
365
|
+
() => false,
|
|
366
|
+
),
|
|
367
|
+
fs$1.access(jsConfigPath).then(
|
|
368
|
+
() => true,
|
|
369
|
+
() => false,
|
|
370
|
+
),
|
|
371
|
+
]);
|
|
372
|
+
|
|
373
|
+
logger.verbose('Config file existence check:');
|
|
374
|
+
logger.verbose(` - template.config.ts: ${tsExists}`);
|
|
375
|
+
logger.verbose(` - template.config.js: ${jsExists}`);
|
|
376
|
+
|
|
377
|
+
if (tsExists) {
|
|
378
|
+
configPath = tsConfigPath;
|
|
379
|
+
} else if (jsExists) {
|
|
380
|
+
configPath = jsConfigPath;
|
|
381
|
+
} else {
|
|
382
|
+
throw new Error(
|
|
383
|
+
`Template config not found in ${templatePath}.\n` +
|
|
384
|
+
'Expected: template.config.ts or template.config.js',
|
|
385
|
+
);
|
|
386
|
+
}
|
|
387
|
+
|
|
388
|
+
logger.verbose(`Using config file: ${configPath}`);
|
|
389
|
+
|
|
390
|
+
// eslint-disable-next-line @typescript-eslint/no-require-imports, security/detect-non-literal-require -- Sucrase handles .ts files at runtime, path is validated above
|
|
391
|
+
const config = require(configPath);
|
|
392
|
+
|
|
393
|
+
logger.verbose('Template config loaded successfully');
|
|
394
|
+
|
|
395
|
+
return config.default || config;
|
|
396
|
+
};
|
|
397
|
+
|
|
398
|
+
/**
|
|
399
|
+
* 加载模板列表配置
|
|
400
|
+
*
|
|
401
|
+
* @param configPath - templates.json 配置文件路径
|
|
402
|
+
* @returns 模板列表配置
|
|
403
|
+
*/
|
|
404
|
+
const loadTemplatesConfig = async (
|
|
405
|
+
configPath,
|
|
406
|
+
) => {
|
|
407
|
+
logger.verbose(`Loading templates config from: ${configPath}`);
|
|
408
|
+
|
|
409
|
+
const content = await fs$1.readFile(configPath, 'utf-8');
|
|
410
|
+
// eslint-disable-next-line no-restricted-syntax -- Static config file loaded at build time, safeJsonParse not needed
|
|
411
|
+
const config = JSON.parse(content) ;
|
|
412
|
+
|
|
413
|
+
logger.verbose(
|
|
414
|
+
`Found ${config.templates.length} templates: ${config.templates.map(t => t.name).join(', ')}`,
|
|
415
|
+
);
|
|
416
|
+
|
|
417
|
+
return config;
|
|
418
|
+
};
|
|
419
|
+
|
|
420
|
+
/**
|
|
421
|
+
* 根据模板名称查找模板元信息
|
|
422
|
+
*
|
|
423
|
+
* @param templatesConfig - 模板列表配置
|
|
424
|
+
* @param templateName - 模板名称
|
|
425
|
+
* @returns 模板元信息
|
|
426
|
+
*/
|
|
427
|
+
const findTemplate = (
|
|
428
|
+
templatesConfig,
|
|
429
|
+
templateName,
|
|
430
|
+
) => {
|
|
431
|
+
const template = templatesConfig.templates.find(t => t.name === templateName);
|
|
432
|
+
|
|
433
|
+
if (!template) {
|
|
434
|
+
const availableTemplates = templatesConfig.templates
|
|
435
|
+
.map(t => t.name)
|
|
436
|
+
.join(', ');
|
|
437
|
+
throw new Error(
|
|
438
|
+
`Template "${templateName}" not found.\n` +
|
|
439
|
+
`Available templates: ${availableTemplates}\n` +
|
|
440
|
+
'Use --template <name> to specify a template.',
|
|
441
|
+
);
|
|
442
|
+
}
|
|
443
|
+
|
|
444
|
+
return template;
|
|
445
|
+
};
|
|
446
|
+
|
|
447
|
+
/**
|
|
448
|
+
* 获取模板的完整路径
|
|
449
|
+
*
|
|
450
|
+
* @param basePath - 模板目录(templates.json 所在目录)
|
|
451
|
+
* @param templateMetadata - 模板元信息
|
|
452
|
+
* @returns 模板完整路径
|
|
453
|
+
*/
|
|
454
|
+
const getTemplatePath = async (
|
|
455
|
+
basePath,
|
|
456
|
+
templateMetadata,
|
|
457
|
+
) => {
|
|
458
|
+
logger.verbose('Resolving template path:');
|
|
459
|
+
logger.verbose(` - Base path: ${basePath}`);
|
|
460
|
+
logger.verbose(` - Template location: ${templateMetadata.location}`);
|
|
461
|
+
|
|
462
|
+
// location 是相对于 templates.json 文件的路径
|
|
463
|
+
const templatePath = path.join(basePath, templateMetadata.location);
|
|
464
|
+
|
|
465
|
+
logger.verbose(` - Resolved path: ${templatePath}`);
|
|
466
|
+
|
|
467
|
+
try {
|
|
468
|
+
await fs$1.access(templatePath);
|
|
469
|
+
logger.verbose(' - Template directory exists: ✓');
|
|
470
|
+
// eslint-disable-next-line @coze-arch/use-error-in-catch -- Error handling done in throw statement
|
|
471
|
+
} catch (e) {
|
|
472
|
+
logger.error(' - Template directory does not exist: ✗');
|
|
473
|
+
throw new Error(`Template directory not found: ${templatePath}`);
|
|
474
|
+
}
|
|
475
|
+
|
|
476
|
+
return templatePath;
|
|
477
|
+
};
|
|
478
|
+
|
|
479
|
+
/**
|
|
480
|
+
* 对单个模板执行 pnpm install
|
|
481
|
+
*/
|
|
482
|
+
const warmupTemplate = (templatePath, templateName) => {
|
|
483
|
+
logger.info(`\nWarming up template: ${templateName}`);
|
|
484
|
+
logger.info(` Path: ${templatePath}`);
|
|
485
|
+
|
|
486
|
+
// 检查是否存在 package.json
|
|
487
|
+
const packageJsonPath = node_path.join(templatePath, 'package.json');
|
|
488
|
+
// eslint-disable-next-line security/detect-non-literal-fs-filename
|
|
489
|
+
if (!node_fs.existsSync(packageJsonPath)) {
|
|
490
|
+
logger.info(` ⊘ Skipping ${templateName} (no package.json found)`);
|
|
491
|
+
return;
|
|
492
|
+
}
|
|
493
|
+
|
|
494
|
+
const result = shelljs.exec('pnpm install', {
|
|
495
|
+
cwd: templatePath,
|
|
496
|
+
silent: true,
|
|
497
|
+
});
|
|
498
|
+
|
|
499
|
+
// 输出 stdout
|
|
500
|
+
if (result.stdout) {
|
|
501
|
+
process.stdout.write(result.stdout);
|
|
502
|
+
}
|
|
503
|
+
|
|
504
|
+
// 输出 stderr
|
|
505
|
+
if (result.stderr) {
|
|
506
|
+
process.stderr.write(result.stderr);
|
|
507
|
+
}
|
|
508
|
+
|
|
509
|
+
if (result.code === 0) {
|
|
510
|
+
logger.success(` ✓ ${templateName} warmed up successfully`);
|
|
511
|
+
} else {
|
|
512
|
+
const errorMessage = [
|
|
513
|
+
`pnpm install failed for ${templateName} with exit code ${result.code}`,
|
|
514
|
+
result.stderr ? `\nStderr:\n${result.stderr}` : '',
|
|
515
|
+
result.stdout ? `\nStdout:\n${result.stdout}` : '',
|
|
516
|
+
]
|
|
517
|
+
.filter(Boolean)
|
|
518
|
+
.join('');
|
|
519
|
+
|
|
520
|
+
throw new Error(errorMessage);
|
|
521
|
+
}
|
|
522
|
+
};
|
|
523
|
+
|
|
524
|
+
/**
|
|
525
|
+
* 执行 warmup 命令的内部实现
|
|
526
|
+
*/
|
|
527
|
+
const executeWarmup = async (options) => {
|
|
528
|
+
const timer = new TimeTracker();
|
|
529
|
+
|
|
530
|
+
try {
|
|
531
|
+
const { template: templateFilter } = options;
|
|
532
|
+
|
|
533
|
+
logger.info('Starting template warmup...');
|
|
534
|
+
timer.logPhase('Initialization');
|
|
535
|
+
|
|
536
|
+
// 加载模板配置
|
|
537
|
+
const configPath = getTemplatesConfigPath();
|
|
538
|
+
const templatesConfig = await loadTemplatesConfig(configPath);
|
|
539
|
+
|
|
540
|
+
logger.verbose(
|
|
541
|
+
`Found ${templatesConfig.templates.length} templates in config`,
|
|
542
|
+
);
|
|
543
|
+
|
|
544
|
+
// 过滤模板
|
|
545
|
+
const templatesToWarmup = templateFilter
|
|
546
|
+
? templatesConfig.templates.filter(t => t.name === templateFilter)
|
|
547
|
+
: templatesConfig.templates;
|
|
548
|
+
|
|
549
|
+
if (templatesToWarmup.length === 0) {
|
|
550
|
+
if (templateFilter) {
|
|
551
|
+
logger.warn(`Template "${templateFilter}" not found`);
|
|
552
|
+
logger.info(
|
|
553
|
+
`Available templates: ${templatesConfig.templates.map(t => t.name).join(', ')}`,
|
|
554
|
+
);
|
|
555
|
+
} else {
|
|
556
|
+
logger.warn('No templates found');
|
|
557
|
+
}
|
|
558
|
+
return;
|
|
559
|
+
}
|
|
560
|
+
|
|
561
|
+
logger.info(
|
|
562
|
+
`\nWill warm up ${templatesToWarmup.length} template(s): ${templatesToWarmup.map(t => t.name).join(', ')}`,
|
|
563
|
+
);
|
|
564
|
+
|
|
565
|
+
// 获取模板基础路径
|
|
566
|
+
const basePath = configPath.replace(/\/templates\.json$/, '');
|
|
567
|
+
|
|
568
|
+
// 对每个模板执行 pnpm install
|
|
569
|
+
for (const templateMetadata of templatesToWarmup) {
|
|
570
|
+
const templatePath = await getTemplatePath(basePath, templateMetadata);
|
|
571
|
+
warmupTemplate(templatePath, templateMetadata.name);
|
|
572
|
+
timer.logPhase(`Warmup ${templateMetadata.name}`);
|
|
573
|
+
}
|
|
574
|
+
|
|
575
|
+
logger.success('\n✅ All templates warmed up successfully!');
|
|
576
|
+
logger.info(
|
|
577
|
+
'\nNext time you run `coze init`, it will be much faster as node_modules are pre-installed.',
|
|
578
|
+
);
|
|
579
|
+
|
|
580
|
+
timer.logTotal();
|
|
581
|
+
} catch (error) {
|
|
582
|
+
timer.logTotal();
|
|
583
|
+
logger.error('Failed to warmup templates:');
|
|
584
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
585
|
+
process.exit(1);
|
|
586
|
+
}
|
|
587
|
+
};
|
|
588
|
+
|
|
589
|
+
/**
|
|
590
|
+
* 注册 warmup 命令到 program
|
|
591
|
+
*/
|
|
592
|
+
const registerCommand$4 = program => {
|
|
593
|
+
program
|
|
594
|
+
.command('warmup')
|
|
595
|
+
.description('Pre-install dependencies for templates to speed up init')
|
|
596
|
+
.option('-t, --template <name>', 'Warmup a specific template only')
|
|
597
|
+
.action(async options => {
|
|
598
|
+
await executeWarmup(options);
|
|
599
|
+
});
|
|
600
|
+
};
|
|
601
|
+
|
|
602
|
+
function _optionalChain$2(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
603
|
+
// Safe JSON parsing utilities with type safety and error handling
|
|
604
|
+
// Provides fallback values, validation, and error monitoring capabilities
|
|
605
|
+
|
|
606
|
+
/**
|
|
607
|
+
* Options for safe JSON parsing
|
|
608
|
+
*/
|
|
609
|
+
|
|
610
|
+
|
|
611
|
+
|
|
612
|
+
|
|
613
|
+
|
|
614
|
+
|
|
615
|
+
|
|
616
|
+
|
|
617
|
+
|
|
618
|
+
|
|
619
|
+
|
|
620
|
+
|
|
621
|
+
|
|
622
|
+
|
|
623
|
+
|
|
624
|
+
|
|
625
|
+
|
|
626
|
+
|
|
627
|
+
|
|
628
|
+
|
|
629
|
+
|
|
630
|
+
|
|
631
|
+
|
|
632
|
+
|
|
633
|
+
|
|
634
|
+
|
|
635
|
+
|
|
636
|
+
|
|
637
|
+
|
|
638
|
+
|
|
639
|
+
|
|
640
|
+
|
|
641
|
+
/**
|
|
642
|
+
* Safely parse JSON with error handling and type safety
|
|
643
|
+
*
|
|
644
|
+
* @example
|
|
645
|
+
* ```ts
|
|
646
|
+
* // Basic usage - returns unknown | undefined
|
|
647
|
+
* const data = safeJsonParse('{"a":1}'); // { a: 1 }
|
|
648
|
+
*
|
|
649
|
+
* // With default value - always returns T
|
|
650
|
+
* const config = safeJsonParse(str, {});
|
|
651
|
+
* const user = safeJsonParse(str, null);
|
|
652
|
+
*
|
|
653
|
+
* // With error reporting
|
|
654
|
+
* const data = safeJsonParse(input, null, {
|
|
655
|
+
* onError: (error, input) => logger.error('Parse failed', { error, input })
|
|
656
|
+
* });
|
|
657
|
+
*
|
|
658
|
+
* // With validation
|
|
659
|
+
* const isUser = (data: unknown): data is User => ...;
|
|
660
|
+
* const user = safeJsonParse<User>(input, null, { validate: isUser });
|
|
661
|
+
* ```
|
|
662
|
+
*/
|
|
663
|
+
function safeJsonParse(
|
|
664
|
+
input,
|
|
665
|
+
defaultValueOrOptions,
|
|
666
|
+
optionsArg,
|
|
667
|
+
) {
|
|
668
|
+
// Parse arguments
|
|
669
|
+
let defaultValue;
|
|
670
|
+
let options;
|
|
671
|
+
|
|
672
|
+
if (arguments.length === 2) {
|
|
673
|
+
// safeJsonParse(input, options) or safeJsonParse(input, defaultValue)
|
|
674
|
+
{
|
|
675
|
+
defaultValue = defaultValueOrOptions ;
|
|
676
|
+
options = undefined;
|
|
677
|
+
}
|
|
678
|
+
} else if (arguments.length === 3) {
|
|
679
|
+
// safeJsonParse(input, defaultValue, options)
|
|
680
|
+
defaultValue = defaultValueOrOptions ;
|
|
681
|
+
options = optionsArg;
|
|
682
|
+
}
|
|
683
|
+
|
|
684
|
+
// If input is already an object (and not null), return it directly
|
|
685
|
+
if (typeof input === 'object' && input !== null) {
|
|
686
|
+
return input ;
|
|
687
|
+
}
|
|
688
|
+
|
|
689
|
+
try {
|
|
690
|
+
const parsed = JSON.parse(String(input));
|
|
691
|
+
|
|
692
|
+
// Optional validation
|
|
693
|
+
if (_optionalChain$2([options, 'optionalAccess', _ => _.validate])) {
|
|
694
|
+
if (options.validate(parsed)) {
|
|
695
|
+
return parsed;
|
|
696
|
+
} else {
|
|
697
|
+
const validationError = new Error('JSON validation failed');
|
|
698
|
+
_optionalChain$2([options, 'access', _2 => _2.onError, 'optionalCall', _3 => _3(validationError, input)]);
|
|
699
|
+
|
|
700
|
+
if (options.throwOnValidationError) {
|
|
701
|
+
throw validationError;
|
|
702
|
+
}
|
|
703
|
+
return defaultValue;
|
|
704
|
+
}
|
|
705
|
+
}
|
|
706
|
+
|
|
707
|
+
return parsed;
|
|
708
|
+
} catch (error) {
|
|
709
|
+
// Re-throw validation errors when throwOnValidationError is true
|
|
710
|
+
if (error instanceof Error && error.message === 'JSON validation failed' && _optionalChain$2([options, 'optionalAccess', _4 => _4.throwOnValidationError])) {
|
|
711
|
+
throw error;
|
|
712
|
+
}
|
|
713
|
+
_optionalChain$2([options, 'optionalAccess', _5 => _5.onError, 'optionalCall', _6 => _6(error , input)]);
|
|
714
|
+
return defaultValue;
|
|
715
|
+
}
|
|
716
|
+
}
|
|
717
|
+
|
|
718
|
+
function _optionalChain$1(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }
|
|
719
|
+
|
|
720
|
+
|
|
721
|
+
/**
|
|
722
|
+
* 检测并解析配置文件内容
|
|
723
|
+
*
|
|
724
|
+
* @param content - 文件内容
|
|
725
|
+
* @returns 解析后的配置对象
|
|
726
|
+
*/
|
|
727
|
+
const parseConfigContent = (content) => {
|
|
728
|
+
// 1. 尝试 TOML 格式 (默认)
|
|
729
|
+
try {
|
|
730
|
+
const config = toml.parse(content);
|
|
731
|
+
return config ;
|
|
732
|
+
} catch (error) {
|
|
733
|
+
// TOML 解析失败,继续尝试其他格式
|
|
734
|
+
|
|
735
|
+
console.debug('TOML parse failed:', error);
|
|
736
|
+
}
|
|
737
|
+
|
|
738
|
+
// 2. 尝试 YAML 格式
|
|
739
|
+
try {
|
|
740
|
+
const config = jsYaml.load(content);
|
|
741
|
+
if (config && typeof config === 'object') {
|
|
742
|
+
return config ;
|
|
743
|
+
}
|
|
744
|
+
} catch (error) {
|
|
745
|
+
// YAML 解析失败,继续尝试其他格式
|
|
746
|
+
|
|
747
|
+
console.debug('YAML parse failed:', error);
|
|
748
|
+
}
|
|
749
|
+
|
|
750
|
+
// 3. 尝试 JSON 格式
|
|
751
|
+
const config = safeJsonParse(content);
|
|
752
|
+
if (config) {
|
|
753
|
+
return config;
|
|
754
|
+
}
|
|
755
|
+
|
|
756
|
+
return null;
|
|
757
|
+
};
|
|
758
|
+
|
|
759
|
+
/**
|
|
760
|
+
* 加载 .coze 配置文件
|
|
761
|
+
*
|
|
762
|
+
* @param projectPath - 项目目录路径
|
|
763
|
+
* @returns .coze 配置对象
|
|
764
|
+
*/
|
|
765
|
+
const loadCozeConfig = async (
|
|
766
|
+
projectPath = process.cwd(),
|
|
767
|
+
) => {
|
|
768
|
+
const cozeConfigPath = path.join(projectPath, '.coze');
|
|
769
|
+
|
|
770
|
+
let content;
|
|
771
|
+
try {
|
|
772
|
+
content = await fs$1.readFile(cozeConfigPath, 'utf-8');
|
|
773
|
+
} catch (error) {
|
|
774
|
+
throw new Error(
|
|
775
|
+
`.coze config file not found in ${projectPath}\n` +
|
|
776
|
+
'Please ensure you are in a project directory initialized with coze-coding.\n' +
|
|
777
|
+
`Error: ${error instanceof Error ? error.message : String(error)}`,
|
|
778
|
+
);
|
|
779
|
+
}
|
|
780
|
+
|
|
781
|
+
const config = parseConfigContent(content);
|
|
782
|
+
if (!config) {
|
|
783
|
+
throw new Error(
|
|
784
|
+
'Failed to parse .coze config file.\n' +
|
|
785
|
+
'Please ensure the file is in valid TOML, YAML, or JSON format.',
|
|
786
|
+
);
|
|
787
|
+
}
|
|
788
|
+
|
|
789
|
+
return config;
|
|
790
|
+
};
|
|
791
|
+
|
|
792
|
+
/**
|
|
793
|
+
* 获取指定命令的配置
|
|
794
|
+
*
|
|
795
|
+
* @param config - .coze 配置对象
|
|
796
|
+
* @param commandName - 命令名称 (dev/build/start)
|
|
797
|
+
* @returns 命令配置数组
|
|
798
|
+
*/
|
|
799
|
+
const getCommandConfig = (
|
|
800
|
+
config,
|
|
801
|
+
commandName,
|
|
802
|
+
) => {
|
|
803
|
+
let commandConfig;
|
|
804
|
+
|
|
805
|
+
// 根据命令名称映射到配置路径
|
|
806
|
+
switch (commandName) {
|
|
807
|
+
case 'dev':
|
|
808
|
+
commandConfig = _optionalChain$1([config, 'access', _ => _.dev, 'optionalAccess', _2 => _2.run]);
|
|
809
|
+
break;
|
|
810
|
+
case 'build':
|
|
811
|
+
commandConfig = _optionalChain$1([config, 'access', _3 => _3.deploy, 'optionalAccess', _4 => _4.build]);
|
|
812
|
+
break;
|
|
813
|
+
case 'start':
|
|
814
|
+
commandConfig = _optionalChain$1([config, 'access', _5 => _5.deploy, 'optionalAccess', _6 => _6.run]);
|
|
815
|
+
break;
|
|
816
|
+
default:
|
|
817
|
+
throw new Error(`Unknown command: ${commandName}`);
|
|
818
|
+
}
|
|
819
|
+
|
|
820
|
+
if (!commandConfig || commandConfig.length === 0) {
|
|
821
|
+
throw new Error(
|
|
822
|
+
`Command '${commandName}' is not configured in .coze file.\n` +
|
|
823
|
+
'Please add the corresponding configuration to your .coze file.',
|
|
824
|
+
);
|
|
825
|
+
}
|
|
826
|
+
|
|
827
|
+
return commandConfig;
|
|
828
|
+
};
|
|
829
|
+
|
|
830
|
+
// ABOUTME: Fix rule to comment out problematic outputFileTracingRoot config in Next.js projects
|
|
831
|
+
// ABOUTME: This config can cause issues in monorepo environments and should be removed
|
|
832
|
+
|
|
833
|
+
|
|
834
|
+
|
|
835
|
+
|
|
836
|
+
/**
|
|
837
|
+
* 检查是否为 Next.js 项目
|
|
838
|
+
*/
|
|
839
|
+
const isNextProject = (projectFolder) => {
|
|
840
|
+
const packageJsonPath = path.join(projectFolder, 'package.json');
|
|
841
|
+
|
|
842
|
+
if (!fs.existsSync(packageJsonPath)) {
|
|
843
|
+
return false;
|
|
844
|
+
}
|
|
845
|
+
|
|
846
|
+
try {
|
|
847
|
+
// eslint-disable-next-line no-restricted-syntax
|
|
848
|
+
const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf-8'));
|
|
849
|
+
const deps = {
|
|
850
|
+
...packageJson.dependencies,
|
|
851
|
+
...packageJson.devDependencies,
|
|
852
|
+
};
|
|
853
|
+
|
|
854
|
+
return 'next' in deps;
|
|
855
|
+
} catch (e) {
|
|
856
|
+
return false;
|
|
857
|
+
}
|
|
858
|
+
};
|
|
859
|
+
|
|
860
|
+
/**
|
|
861
|
+
* 查找 Next.js 配置文件
|
|
862
|
+
*/
|
|
863
|
+
const findNextConfigFile = (projectFolder) => {
|
|
864
|
+
const possibleConfigs = [
|
|
865
|
+
'next.config.ts',
|
|
866
|
+
'next.config.js',
|
|
867
|
+
'next.config.mjs',
|
|
868
|
+
];
|
|
869
|
+
|
|
870
|
+
for (const configFile of possibleConfigs) {
|
|
871
|
+
const configPath = path.join(projectFolder, configFile);
|
|
872
|
+
if (fs.existsSync(configPath)) {
|
|
873
|
+
return configPath;
|
|
874
|
+
}
|
|
875
|
+
}
|
|
876
|
+
|
|
877
|
+
return null;
|
|
878
|
+
};
|
|
879
|
+
|
|
880
|
+
/**
|
|
881
|
+
* 注释掉 outputFileTracingRoot 配置
|
|
882
|
+
*/
|
|
883
|
+
const commentOutOutputTracingRoot = (
|
|
884
|
+
configPath,
|
|
885
|
+
) => {
|
|
886
|
+
let content = fs.readFileSync(configPath, 'utf-8');
|
|
887
|
+
let modified = false;
|
|
888
|
+
let originalLine = null;
|
|
889
|
+
|
|
890
|
+
// 匹配包含 outputFileTracingRoot 的行(尚未被注释的)
|
|
891
|
+
// 支持 path.resolve(...) 后面有空格,然后可选逗号
|
|
892
|
+
// 只匹配单行配置,不匹配跨多行的配置
|
|
893
|
+
const pattern =
|
|
894
|
+
/^(\s*)(outputFileTracingRoot:\s*path\.resolve\([^\n\r)]+\)\s*,?)\s*$/gm;
|
|
895
|
+
|
|
896
|
+
const matches = content.match(pattern);
|
|
897
|
+
|
|
898
|
+
if (matches && matches.length > 0) {
|
|
899
|
+
originalLine = matches[0].trim();
|
|
900
|
+
|
|
901
|
+
// 在匹配的行前添加 // 注释
|
|
902
|
+
content = content.replace(pattern, '$1// $2');
|
|
903
|
+
modified = true;
|
|
904
|
+
|
|
905
|
+
fs.writeFileSync(configPath, content, 'utf-8');
|
|
906
|
+
}
|
|
907
|
+
|
|
908
|
+
return { modified, originalLine };
|
|
909
|
+
};
|
|
910
|
+
|
|
911
|
+
// start_aigc
|
|
912
|
+
/**
|
|
913
|
+
* Fix 规则:注释掉 Next.js 项目中的 outputFileTracingRoot 配置
|
|
914
|
+
* 这个配置在 monorepo 环境中可能会导致问题
|
|
915
|
+
*/
|
|
916
|
+
const fixNextOutputTracingRoot = context => {
|
|
917
|
+
const ruleName = 'next-output-tracing-root';
|
|
918
|
+
|
|
919
|
+
// 1. 检查是否为 Next.js 项目
|
|
920
|
+
if (!isNextProject(context.projectFolder)) {
|
|
921
|
+
return {
|
|
922
|
+
ruleName,
|
|
923
|
+
applied: false,
|
|
924
|
+
message: 'Not a Next.js project, skipping',
|
|
925
|
+
};
|
|
926
|
+
}
|
|
927
|
+
|
|
928
|
+
// 2. 查找 Next.js 配置文件
|
|
929
|
+
const configPath = findNextConfigFile(context.projectFolder);
|
|
930
|
+
|
|
931
|
+
if (!configPath) {
|
|
932
|
+
return {
|
|
933
|
+
ruleName,
|
|
934
|
+
applied: false,
|
|
935
|
+
message: 'Next.js config file not found, skipping',
|
|
936
|
+
};
|
|
937
|
+
}
|
|
938
|
+
|
|
939
|
+
// 3. 注释掉 outputFileTracingRoot 配置
|
|
940
|
+
const { modified, originalLine } = commentOutOutputTracingRoot(configPath);
|
|
941
|
+
|
|
942
|
+
if (modified && originalLine) {
|
|
943
|
+
logger.success(
|
|
944
|
+
`Commented out outputFileTracingRoot in ${configPath.split('/').pop()}`,
|
|
945
|
+
);
|
|
946
|
+
logger.info(` Original: ${originalLine}`);
|
|
947
|
+
|
|
948
|
+
return {
|
|
949
|
+
ruleName,
|
|
950
|
+
applied: true,
|
|
951
|
+
message: `Successfully commented out: ${originalLine}`,
|
|
952
|
+
};
|
|
953
|
+
}
|
|
954
|
+
|
|
955
|
+
return {
|
|
956
|
+
ruleName,
|
|
957
|
+
applied: false,
|
|
958
|
+
message: 'No outputFileTracingRoot config found, skipping',
|
|
959
|
+
};
|
|
960
|
+
};
|
|
961
|
+
// end_aigc
|
|
962
|
+
|
|
963
|
+
/**
|
|
964
|
+
* 所有修复规则的数组
|
|
965
|
+
* 按顺序执行,新增规则直接添加到数组中
|
|
966
|
+
*/
|
|
967
|
+
const rules = [
|
|
968
|
+
// Next.js related fixes
|
|
969
|
+
fixNextOutputTracingRoot,
|
|
970
|
+
|
|
971
|
+
// Add more rules here
|
|
972
|
+
] ;
|
|
973
|
+
|
|
974
|
+
// ABOUTME: Fix command for resolving legacy issues from previous project versions
|
|
975
|
+
// ABOUTME: Applies a series of fix rules defined in the fix-rules directory
|
|
976
|
+
|
|
977
|
+
|
|
978
|
+
// start_aigc
|
|
979
|
+
/**
|
|
980
|
+
* 执行 fix 命令的内部实现
|
|
981
|
+
*/
|
|
982
|
+
const executeFix = async (options = {}) => {
|
|
983
|
+
try {
|
|
984
|
+
const cwd = process.cwd();
|
|
985
|
+
const projectFolder = options.directory
|
|
986
|
+
? path.resolve(cwd, options.directory)
|
|
987
|
+
: cwd;
|
|
988
|
+
|
|
989
|
+
logger.info(`Running fix command on: ${projectFolder}`);
|
|
990
|
+
logger.info(`Found ${rules.length} fix rule(s) to apply\n`);
|
|
991
|
+
|
|
992
|
+
const context = {
|
|
993
|
+
cwd,
|
|
994
|
+
projectFolder,
|
|
995
|
+
};
|
|
996
|
+
|
|
997
|
+
let appliedCount = 0;
|
|
998
|
+
let skippedCount = 0;
|
|
999
|
+
|
|
1000
|
+
// 依次执行所有修复规则
|
|
1001
|
+
for (const rule of rules) {
|
|
1002
|
+
try {
|
|
1003
|
+
const result = await Promise.resolve(rule(context));
|
|
1004
|
+
|
|
1005
|
+
if (result.applied) {
|
|
1006
|
+
appliedCount++;
|
|
1007
|
+
logger.success(`✓ ${result.ruleName}: ${result.message}`);
|
|
1008
|
+
} else {
|
|
1009
|
+
skippedCount++;
|
|
1010
|
+
logger.info(`○ ${result.ruleName}: ${result.message}`);
|
|
1011
|
+
}
|
|
1012
|
+
} catch (error) {
|
|
1013
|
+
logger.error(
|
|
1014
|
+
`✗ Rule execution failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
1015
|
+
);
|
|
1016
|
+
}
|
|
1017
|
+
}
|
|
1018
|
+
|
|
1019
|
+
// 输出汇总信息
|
|
1020
|
+
logger.info(
|
|
1021
|
+
`\nSummary: ${appliedCount} fixed, ${skippedCount} skipped, ${rules.length} total`,
|
|
1022
|
+
);
|
|
1023
|
+
|
|
1024
|
+
if (appliedCount > 0) {
|
|
1025
|
+
logger.success('\nFixes applied successfully!');
|
|
1026
|
+
} else {
|
|
1027
|
+
logger.info('\nNo fixes needed');
|
|
1028
|
+
}
|
|
1029
|
+
} catch (error) {
|
|
1030
|
+
logger.error('Failed to run fix command:');
|
|
1031
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
1032
|
+
process.exit(1);
|
|
1033
|
+
}
|
|
1034
|
+
};
|
|
1035
|
+
// end_aigc
|
|
1036
|
+
|
|
1037
|
+
/**
|
|
1038
|
+
* 注册 fix 命令到 program
|
|
1039
|
+
*/
|
|
1040
|
+
const registerCommand$3 = program => {
|
|
1041
|
+
program
|
|
1042
|
+
.command('fix')
|
|
1043
|
+
.description(
|
|
1044
|
+
'Fix legacy issues from previous versions (e.g., problematic configs)',
|
|
1045
|
+
)
|
|
1046
|
+
.argument(
|
|
1047
|
+
'[directory]',
|
|
1048
|
+
'Target directory to fix (defaults to current directory)',
|
|
1049
|
+
)
|
|
1050
|
+
.action(async (directory) => {
|
|
1051
|
+
await executeFix({ directory });
|
|
1052
|
+
});
|
|
1053
|
+
};
|
|
1054
|
+
|
|
1055
|
+
function _nullishCoalesce$1(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }
|
|
1056
|
+
/**
|
|
1057
|
+
* 日志文件名常量
|
|
1058
|
+
*/
|
|
1059
|
+
const LOG_FILE_NAME$1 = 'dev.log';
|
|
1060
|
+
|
|
1061
|
+
/**
|
|
1062
|
+
* 获取日志目录
|
|
1063
|
+
* 优先使用环境变量 COZE_LOG_DIR,否则使用 ~/.coze-logs
|
|
1064
|
+
*/
|
|
1065
|
+
const getLogDir$1 = () =>
|
|
1066
|
+
process.env.COZE_LOG_DIR || path.join(os.homedir(), '.coze-logs');
|
|
1067
|
+
|
|
1068
|
+
/**
|
|
1069
|
+
* 解析日志文件路径
|
|
1070
|
+
* - 如果是绝对路径,直接使用
|
|
1071
|
+
* - 如果是相对路径,基于 getLogDir() + 相对路径
|
|
1072
|
+
* - 如果为空,使用 getLogDir() + LOG_FILE_NAME
|
|
1073
|
+
*/
|
|
1074
|
+
const resolveLogFilePath$1 = (logFile) => {
|
|
1075
|
+
if (!logFile) {
|
|
1076
|
+
return path.join(getLogDir$1(), LOG_FILE_NAME$1);
|
|
1077
|
+
}
|
|
1078
|
+
|
|
1079
|
+
if (path.isAbsolute(logFile)) {
|
|
1080
|
+
return logFile;
|
|
1081
|
+
}
|
|
1082
|
+
|
|
1083
|
+
return path.join(getLogDir$1(), logFile);
|
|
1084
|
+
};
|
|
1085
|
+
|
|
1086
|
+
/**
|
|
1087
|
+
* 创建日志写入流
|
|
1088
|
+
*/
|
|
1089
|
+
const createLogStream$1 = (logFilePath) => {
|
|
1090
|
+
const logDir = path.dirname(logFilePath);
|
|
1091
|
+
|
|
1092
|
+
// 确保日志目录存在
|
|
1093
|
+
if (!fs.existsSync(logDir)) {
|
|
1094
|
+
fs.mkdirSync(logDir, { recursive: true });
|
|
1095
|
+
}
|
|
1096
|
+
|
|
1097
|
+
// 使用 'w' 标志覆盖之前的日志
|
|
1098
|
+
return fs.createWriteStream(logFilePath, { flags: 'w' });
|
|
1099
|
+
};
|
|
1100
|
+
|
|
1101
|
+
/**
|
|
1102
|
+
* 执行命令的内部实现
|
|
1103
|
+
*/
|
|
1104
|
+
const executeRun = async (
|
|
1105
|
+
commandName,
|
|
1106
|
+
options = {},
|
|
1107
|
+
) => {
|
|
1108
|
+
try {
|
|
1109
|
+
logger.info(`Running ${commandName} command...`);
|
|
1110
|
+
|
|
1111
|
+
// 1. 对于 build 命令,先执行 fix 以确保项目配置正确
|
|
1112
|
+
if (['dev', 'build'].includes(commandName)) {
|
|
1113
|
+
logger.info('\n🔧 Running fix command before build...\n');
|
|
1114
|
+
try {
|
|
1115
|
+
await executeFix();
|
|
1116
|
+
// eslint-disable-next-line @coze-arch/no-empty-catch
|
|
1117
|
+
} catch (e) {
|
|
1118
|
+
// just ignore
|
|
1119
|
+
}
|
|
1120
|
+
logger.info('');
|
|
1121
|
+
}
|
|
1122
|
+
|
|
1123
|
+
// 2. 加载 .coze 配置
|
|
1124
|
+
const config = await loadCozeConfig();
|
|
1125
|
+
const commandArgs = getCommandConfig(config, commandName);
|
|
1126
|
+
|
|
1127
|
+
// 3. 准备日志
|
|
1128
|
+
const logFilePath = resolveLogFilePath$1(options.logFile);
|
|
1129
|
+
const logStream = createLogStream$1(logFilePath);
|
|
1130
|
+
|
|
1131
|
+
// 4. 执行命令
|
|
1132
|
+
const commandString = commandArgs.join(' ');
|
|
1133
|
+
|
|
1134
|
+
logger.info(`Executing: ${commandString}`);
|
|
1135
|
+
logger.info(`Working directory: ${process.cwd()}`);
|
|
1136
|
+
logger.info(`Log file: ${logFilePath}`);
|
|
1137
|
+
|
|
1138
|
+
const childProcess = shelljs.exec(commandString, {
|
|
1139
|
+
async: true,
|
|
1140
|
+
silent: true, // 不自动输出,我们手动处理
|
|
1141
|
+
});
|
|
1142
|
+
|
|
1143
|
+
if (!childProcess) {
|
|
1144
|
+
throw new Error('Failed to create child process');
|
|
1145
|
+
}
|
|
1146
|
+
|
|
1147
|
+
// 将输出同时写入控制台和日志文件
|
|
1148
|
+
_optionalChain([childProcess, 'access', _ => _.stdout, 'optionalAccess', _2 => _2.on, 'call', _3 => _3('data', (data) => {
|
|
1149
|
+
process.stdout.write(data);
|
|
1150
|
+
logStream.write(data);
|
|
1151
|
+
})]);
|
|
1152
|
+
|
|
1153
|
+
_optionalChain([childProcess, 'access', _4 => _4.stderr, 'optionalAccess', _5 => _5.on, 'call', _6 => _6('data', (data) => {
|
|
1154
|
+
process.stderr.write(data);
|
|
1155
|
+
logStream.write(data);
|
|
1156
|
+
})]);
|
|
1157
|
+
|
|
1158
|
+
childProcess.on('close', (code, signal) => {
|
|
1159
|
+
logStream.end();
|
|
1160
|
+
|
|
1161
|
+
if (code !== 0) {
|
|
1162
|
+
logger.error(
|
|
1163
|
+
`Command exited with code ${_nullishCoalesce$1(code, () => ( 'unknown'))}${signal ? ` and signal ${signal}` : ''}`,
|
|
1164
|
+
);
|
|
1165
|
+
logger.error(`Check log file for details: ${logFilePath}`);
|
|
1166
|
+
process.exit(code || 1);
|
|
1167
|
+
} else {
|
|
1168
|
+
logger.success('Command completed successfully');
|
|
1169
|
+
logger.info(`Log file: ${logFilePath}`);
|
|
1170
|
+
}
|
|
1171
|
+
});
|
|
1172
|
+
|
|
1173
|
+
childProcess.on('error', (error) => {
|
|
1174
|
+
logger.error('Failed to execute command:');
|
|
1175
|
+
logger.error(`Error: ${error.message}`);
|
|
1176
|
+
if (error.stack) {
|
|
1177
|
+
logger.error(`Stack trace:\n${error.stack}`);
|
|
1178
|
+
}
|
|
1179
|
+
logStream.end();
|
|
1180
|
+
process.exit(1);
|
|
1181
|
+
});
|
|
1182
|
+
} catch (error) {
|
|
1183
|
+
logger.error(`Failed to run ${commandName} command:`);
|
|
1184
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
1185
|
+
process.exit(1);
|
|
1186
|
+
}
|
|
1187
|
+
};
|
|
1188
|
+
|
|
1189
|
+
/**
|
|
1190
|
+
* 注册 dev/build/start 命令到 program
|
|
1191
|
+
*/
|
|
1192
|
+
const registerCommand$2 = program => {
|
|
1193
|
+
// dev 命令
|
|
1194
|
+
program
|
|
1195
|
+
.command('dev')
|
|
1196
|
+
.description('Start development server')
|
|
1197
|
+
.option('--log-file <path>', 'Log file path')
|
|
1198
|
+
.action(async options => {
|
|
1199
|
+
await executeRun('dev', options);
|
|
1200
|
+
});
|
|
1201
|
+
|
|
1202
|
+
// build 命令
|
|
1203
|
+
program
|
|
1204
|
+
.command('build')
|
|
1205
|
+
.description('Build for production')
|
|
1206
|
+
.option('--log-file <path>', 'Log file path')
|
|
1207
|
+
.action(async options => {
|
|
1208
|
+
await executeRun('build', options);
|
|
1209
|
+
});
|
|
1210
|
+
|
|
1211
|
+
// start 命令
|
|
1212
|
+
program
|
|
1213
|
+
.command('start')
|
|
1214
|
+
.description('Start production server')
|
|
1215
|
+
.option('--log-file <path>', 'Log file path')
|
|
1216
|
+
.action(async options => {
|
|
1217
|
+
await executeRun('start', options);
|
|
1218
|
+
});
|
|
1219
|
+
};
|
|
1220
|
+
|
|
1221
|
+
/**
|
|
1222
|
+
* 在后台启动一个独立的子进程
|
|
1223
|
+
* 类似于 `setsid command args >/dev/null 2>&1 &`
|
|
1224
|
+
*
|
|
1225
|
+
* @param command - 要执行的命令 (例如: 'npm', 'node', 'bash')
|
|
1226
|
+
* @param args - 命令参数数组 (例如: ['run', 'dev'])
|
|
1227
|
+
* @param options - 配置选项
|
|
1228
|
+
* @returns 子进程的 PID
|
|
1229
|
+
*/
|
|
1230
|
+
function spawnDetached(
|
|
1231
|
+
command,
|
|
1232
|
+
args,
|
|
1233
|
+
options,
|
|
1234
|
+
) {
|
|
1235
|
+
const { cwd, verbose = true } = options;
|
|
1236
|
+
const isWindows = os.platform() === 'win32';
|
|
1237
|
+
|
|
1238
|
+
if (verbose) {
|
|
1239
|
+
console.log(`Spawning detached process: ${command} ${args.join(' ')}`);
|
|
1240
|
+
console.log(`Working directory: ${cwd}`);
|
|
1241
|
+
}
|
|
1242
|
+
|
|
1243
|
+
// 使用 spawn 创建后台子进程
|
|
1244
|
+
const child = child_process.spawn(command, args, {
|
|
1245
|
+
cwd,
|
|
1246
|
+
detached: !isWindows, // Windows 不完全支持 detached,但仍可以使用
|
|
1247
|
+
stdio: 'ignore', // 忽略所有输入输出,让进程完全独立运行
|
|
1248
|
+
});
|
|
1249
|
+
|
|
1250
|
+
// 分离父子进程引用,允许父进程退出而不等待子进程
|
|
1251
|
+
child.unref();
|
|
1252
|
+
|
|
1253
|
+
if (verbose && child.pid) {
|
|
1254
|
+
console.log(`Process started with PID: ${child.pid}`);
|
|
1255
|
+
}
|
|
1256
|
+
|
|
1257
|
+
return child.pid;
|
|
1258
|
+
}
|
|
1259
|
+
|
|
1260
|
+
/**
|
|
1261
|
+
* 创建 AJV 验证器实例
|
|
1262
|
+
*/
|
|
1263
|
+
const createAjvInstance = () => {
|
|
1264
|
+
const ajv = new Ajv({
|
|
1265
|
+
useDefaults: true, // 自动应用默认值
|
|
1266
|
+
coerceTypes: true, // 类型强制转换
|
|
1267
|
+
removeAdditional: false, // 保留额外属性以便后续报错
|
|
1268
|
+
allErrors: true, // 收集所有错误
|
|
1269
|
+
});
|
|
1270
|
+
|
|
1271
|
+
addFormats(ajv);
|
|
1272
|
+
return ajv;
|
|
1273
|
+
};
|
|
1274
|
+
|
|
1275
|
+
/**
|
|
1276
|
+
* 验证参数
|
|
1277
|
+
*
|
|
1278
|
+
* @param schema - JSON Schema 定义
|
|
1279
|
+
* @param params - 待验证的参数
|
|
1280
|
+
* @returns 验证后的参数(应用了默认值)
|
|
1281
|
+
* @throws 验证失败时抛出错误
|
|
1282
|
+
*/
|
|
1283
|
+
const validateParams = (
|
|
1284
|
+
schema,
|
|
1285
|
+
params,
|
|
1286
|
+
) => {
|
|
1287
|
+
const ajv = createAjvInstance();
|
|
1288
|
+
const validate = ajv.compile(schema);
|
|
1289
|
+
|
|
1290
|
+
const isValid = validate(params);
|
|
1291
|
+
|
|
1292
|
+
if (!isValid) {
|
|
1293
|
+
const errors = validate.errors || [];
|
|
1294
|
+
const errorMessages = errors.map(err => {
|
|
1295
|
+
const path = err.instancePath || '/';
|
|
1296
|
+
const message = err.message || 'validation failed';
|
|
1297
|
+
return ` - ${path}: ${message}`;
|
|
1298
|
+
});
|
|
1299
|
+
|
|
1300
|
+
throw new Error(
|
|
1301
|
+
`Parameter validation failed:\n${errorMessages.join('\n')}\n\nPlease check your parameters and try again.`,
|
|
1302
|
+
);
|
|
1303
|
+
}
|
|
1304
|
+
|
|
1305
|
+
return params ;
|
|
1306
|
+
};
|
|
1307
|
+
|
|
1308
|
+
/**
|
|
1309
|
+
* 从 Commander 解析透传参数
|
|
1310
|
+
* 将 kebab-case 的 CLI 参数转换为 camelCase
|
|
1311
|
+
*
|
|
1312
|
+
* 使用 minimist 解析 process.argv,自动处理类型转换
|
|
1313
|
+
*
|
|
1314
|
+
* @param command - Commander 命令实例
|
|
1315
|
+
* @param knownOptions - 已知的选项集合(不需要透传的选项)
|
|
1316
|
+
* @returns 参数对象
|
|
1317
|
+
*/
|
|
1318
|
+
const parsePassThroughParams = (
|
|
1319
|
+
command,
|
|
1320
|
+
knownOptions = new Set(),
|
|
1321
|
+
) => {
|
|
1322
|
+
// 使用 minimist 解析所有参数
|
|
1323
|
+
// eslint-disable-next-line @typescript-eslint/no-magic-numbers -- slice(2) to skip node and script path
|
|
1324
|
+
const parsed = minimist(process.argv.slice(2));
|
|
1325
|
+
|
|
1326
|
+
// 过滤掉已知选项和位置参数(_)
|
|
1327
|
+
const filtered = Object.entries(parsed).reduce(
|
|
1328
|
+
(params, [key, value]) => {
|
|
1329
|
+
// 跳过 minimist 的位置参数数组
|
|
1330
|
+
if (key === '_') {
|
|
1331
|
+
return params;
|
|
1332
|
+
}
|
|
1333
|
+
|
|
1334
|
+
// 跳过已知选项(支持原始格式和 camelCase 格式)
|
|
1335
|
+
if (knownOptions.has(key) || knownOptions.has(changeCase.camelCase(key))) {
|
|
1336
|
+
return params;
|
|
1337
|
+
}
|
|
1338
|
+
|
|
1339
|
+
// 将 kebab-case 转换为 camelCase
|
|
1340
|
+
const camelKey = changeCase.camelCase(key);
|
|
1341
|
+
// eslint-disable-next-line security/detect-object-injection -- camelKey is sanitized by camelCase
|
|
1342
|
+
params[camelKey] = value;
|
|
1343
|
+
|
|
1344
|
+
return params;
|
|
1345
|
+
},
|
|
1346
|
+
{},
|
|
1347
|
+
);
|
|
1348
|
+
|
|
1349
|
+
return filtered;
|
|
1350
|
+
};
|
|
1351
|
+
|
|
1352
|
+
/**
|
|
1353
|
+
* 渲染 EJS 模板文件
|
|
1354
|
+
*
|
|
1355
|
+
* @param filePath - 模板文件路径
|
|
1356
|
+
* @param context - 模板上下文
|
|
1357
|
+
* @returns 渲染后的内容
|
|
1358
|
+
*/
|
|
1359
|
+
const renderTemplate = async (
|
|
1360
|
+
filePath,
|
|
1361
|
+
context,
|
|
1362
|
+
) => {
|
|
1363
|
+
const content = await fs$1.readFile(filePath, 'utf-8');
|
|
1364
|
+
|
|
1365
|
+
try {
|
|
1366
|
+
return ejs.render(content, context, {
|
|
1367
|
+
filename: filePath,
|
|
1368
|
+
});
|
|
1369
|
+
} catch (error) {
|
|
1370
|
+
throw new Error(
|
|
1371
|
+
`Failed to render template ${filePath}:\n${error instanceof Error ? error.message : String(error)}`,
|
|
1372
|
+
);
|
|
1373
|
+
}
|
|
1374
|
+
};
|
|
1375
|
+
|
|
1376
|
+
/**
|
|
1377
|
+
* 判断文件是否需要渲染(是否为文本文件)
|
|
1378
|
+
*
|
|
1379
|
+
* @param filePath - 文件路径
|
|
1380
|
+
* @returns 是否需要渲染
|
|
1381
|
+
*/
|
|
1382
|
+
const shouldRenderFile = (filePath) => {
|
|
1383
|
+
const textExtensions = new Set([
|
|
1384
|
+
'.js',
|
|
1385
|
+
'.ts',
|
|
1386
|
+
'.jsx',
|
|
1387
|
+
'.tsx',
|
|
1388
|
+
'.json',
|
|
1389
|
+
'.css',
|
|
1390
|
+
'.scss',
|
|
1391
|
+
'.sass',
|
|
1392
|
+
'.less',
|
|
1393
|
+
'.html',
|
|
1394
|
+
'.md',
|
|
1395
|
+
'.txt',
|
|
1396
|
+
'.yaml',
|
|
1397
|
+
'.yml',
|
|
1398
|
+
'.xml',
|
|
1399
|
+
'.svg',
|
|
1400
|
+
'.env',
|
|
1401
|
+
'.gitignore',
|
|
1402
|
+
'.npmrc',
|
|
1403
|
+
'.eslintrc',
|
|
1404
|
+
'.prettierrc',
|
|
1405
|
+
'.babelrc',
|
|
1406
|
+
'.sh',
|
|
1407
|
+
]);
|
|
1408
|
+
|
|
1409
|
+
const ext = path.extname(filePath).toLowerCase();
|
|
1410
|
+
|
|
1411
|
+
// 没有扩展名的文件(如 .coze)也尝试渲染
|
|
1412
|
+
if (ext === '') {
|
|
1413
|
+
return true;
|
|
1414
|
+
}
|
|
1415
|
+
|
|
1416
|
+
return textExtensions.has(ext);
|
|
1417
|
+
};
|
|
1418
|
+
|
|
1419
|
+
/**
|
|
1420
|
+
* 判断文件是否应该被忽略(不复制到目标目录)
|
|
1421
|
+
*
|
|
1422
|
+
* @param filePath - 文件路径
|
|
1423
|
+
* @returns 是否应该忽略
|
|
1424
|
+
*/
|
|
1425
|
+
const shouldIgnoreFile = (filePath) => {
|
|
1426
|
+
const fileName = path.basename(filePath);
|
|
1427
|
+
const pathParts = filePath.split(path.sep);
|
|
1428
|
+
|
|
1429
|
+
// 精确匹配的文件名
|
|
1430
|
+
const exactMatch = ['template.config.ts', 'template.config.js', '.DS_Store'];
|
|
1431
|
+
|
|
1432
|
+
// 目录名(需要检查路径中是否包含)
|
|
1433
|
+
const directoryPatterns = ['node_modules'];
|
|
1434
|
+
|
|
1435
|
+
// 检查精确匹配
|
|
1436
|
+
if (exactMatch.includes(fileName)) {
|
|
1437
|
+
return true;
|
|
1438
|
+
}
|
|
1439
|
+
|
|
1440
|
+
// 检查路径中是否包含需要忽略的目录
|
|
1441
|
+
return directoryPatterns.some(dir => pathParts.includes(dir));
|
|
1442
|
+
};
|
|
1443
|
+
|
|
1444
|
+
// ABOUTME: File system utilities for template file processing
|
|
1445
|
+
// ABOUTME: Provides directory scanning, file path conversion, and node_modules copying
|
|
1446
|
+
|
|
1447
|
+
|
|
1448
|
+
// start_aigc
|
|
1449
|
+
/**
|
|
1450
|
+
* 递归获取目录中的所有文件
|
|
1451
|
+
*
|
|
1452
|
+
* @param dir - 目录路径
|
|
1453
|
+
* @param baseDir - 基础目录(用于计算相对路径)
|
|
1454
|
+
* @returns 文件相对路径数组
|
|
1455
|
+
*/
|
|
1456
|
+
const getAllFiles = async (
|
|
1457
|
+
dir,
|
|
1458
|
+
baseDir = dir,
|
|
1459
|
+
) => {
|
|
1460
|
+
logger.verbose(`Scanning directory: ${dir}`);
|
|
1461
|
+
|
|
1462
|
+
const entries = await fs$1.readdir(dir, { withFileTypes: true });
|
|
1463
|
+
|
|
1464
|
+
logger.verbose(`Found ${entries.length} entries in ${path.basename(dir)}`);
|
|
1465
|
+
|
|
1466
|
+
const results = await Promise.all(
|
|
1467
|
+
entries.map(entry => {
|
|
1468
|
+
const fullPath = path.join(dir, entry.name);
|
|
1469
|
+
const relativePath = path.relative(baseDir, fullPath);
|
|
1470
|
+
|
|
1471
|
+
if (shouldIgnoreFile(relativePath)) {
|
|
1472
|
+
logger.verbose(` - Ignoring: ${entry.name}`);
|
|
1473
|
+
return [];
|
|
1474
|
+
}
|
|
1475
|
+
|
|
1476
|
+
if (entry.isDirectory()) {
|
|
1477
|
+
logger.verbose(` - Entering directory: ${entry.name}`);
|
|
1478
|
+
return getAllFiles(fullPath, baseDir);
|
|
1479
|
+
}
|
|
1480
|
+
|
|
1481
|
+
logger.verbose(` - Found file: ${entry.name}`);
|
|
1482
|
+
return [relativePath];
|
|
1483
|
+
}),
|
|
1484
|
+
);
|
|
1485
|
+
|
|
1486
|
+
return results.flat();
|
|
1487
|
+
};
|
|
1488
|
+
|
|
1489
|
+
/**
|
|
1490
|
+
* 确保目录存在
|
|
1491
|
+
*
|
|
1492
|
+
* @param dir - 目录路径
|
|
1493
|
+
*/
|
|
1494
|
+
const ensureDir = async (dir) => {
|
|
1495
|
+
await fs$1.mkdir(dir, { recursive: true });
|
|
1496
|
+
};
|
|
1497
|
+
|
|
1498
|
+
/**
|
|
1499
|
+
* 转换模板文件名(白名单机制)
|
|
1500
|
+
* 只对特定文件将 _ 开头转换为 . 开头
|
|
1501
|
+
*
|
|
1502
|
+
* @param filePath - 文件相对路径
|
|
1503
|
+
* @returns 转换后的文件路径
|
|
1504
|
+
*/
|
|
1505
|
+
const convertDotfileName = (filePath) => {
|
|
1506
|
+
// 白名单:需要从 _ 开头转换为 . 开头的文件
|
|
1507
|
+
const dotfileWhitelist = ['_gitignore', '_npmrc'];
|
|
1508
|
+
|
|
1509
|
+
const fileName = path.basename(filePath);
|
|
1510
|
+
|
|
1511
|
+
// 只对白名单中的文件进行转换(如 _gitignore -> .gitignore)
|
|
1512
|
+
if (dotfileWhitelist.includes(fileName)) {
|
|
1513
|
+
return filePath.replace(/(^|\/|\\)_([^/\\]+)$/g, '$1.$2');
|
|
1514
|
+
}
|
|
1515
|
+
|
|
1516
|
+
return filePath;
|
|
1517
|
+
};
|
|
1518
|
+
// end_aigc
|
|
1519
|
+
|
|
1520
|
+
// ABOUTME: File rendering utilities for template processing
|
|
1521
|
+
// ABOUTME: Handles file content rendering, hook execution, and file writing
|
|
1522
|
+
|
|
1523
|
+
|
|
1524
|
+
|
|
1525
|
+
|
|
1526
|
+
|
|
1527
|
+
|
|
1528
|
+
|
|
1529
|
+
|
|
1530
|
+
// start_aigc
|
|
1531
|
+
/**
|
|
1532
|
+
* 执行文件渲染钩子
|
|
1533
|
+
*
|
|
1534
|
+
* @param templateConfig - 模板配置
|
|
1535
|
+
* @param fileInfo - 文件渲染信息
|
|
1536
|
+
* @param context - 模板上下文
|
|
1537
|
+
* @returns 处理后的文件信息,或 null 表示跳过该文件
|
|
1538
|
+
*/
|
|
1539
|
+
const executeFileRenderHook = async (
|
|
1540
|
+
templateConfig,
|
|
1541
|
+
fileInfo,
|
|
1542
|
+
context,
|
|
1543
|
+
) => {
|
|
1544
|
+
if (!templateConfig.onFileRender) {
|
|
1545
|
+
return fileInfo;
|
|
1546
|
+
}
|
|
1547
|
+
|
|
1548
|
+
const result = await templateConfig.onFileRender(
|
|
1549
|
+
fileInfo,
|
|
1550
|
+
context,
|
|
1551
|
+
);
|
|
1552
|
+
|
|
1553
|
+
// false: 跳过文件
|
|
1554
|
+
if (result === false) {
|
|
1555
|
+
return null;
|
|
1556
|
+
}
|
|
1557
|
+
|
|
1558
|
+
// undefined/void: 使用默认内容
|
|
1559
|
+
if (result === undefined || result === null) {
|
|
1560
|
+
return fileInfo;
|
|
1561
|
+
}
|
|
1562
|
+
|
|
1563
|
+
// string: 作为 content,其他不变
|
|
1564
|
+
if (typeof result === 'string') {
|
|
1565
|
+
return {
|
|
1566
|
+
...fileInfo,
|
|
1567
|
+
content: result,
|
|
1568
|
+
};
|
|
1569
|
+
}
|
|
1570
|
+
|
|
1571
|
+
// FileRenderInfo: 使用新对象的信息
|
|
1572
|
+
return result;
|
|
1573
|
+
};
|
|
1574
|
+
|
|
1575
|
+
/**
|
|
1576
|
+
* 准备单个文件的渲染信息(不实际写入)
|
|
1577
|
+
* 用于 dry-run 阶段,收集所有将要写入的文件信息
|
|
1578
|
+
*
|
|
1579
|
+
* @param options - 准备选项
|
|
1580
|
+
* @returns 文件渲染信息,或 null 表示该文件被跳过
|
|
1581
|
+
*/
|
|
1582
|
+
const prepareFileInfo = async (options
|
|
1583
|
+
|
|
1584
|
+
|
|
1585
|
+
|
|
1586
|
+
|
|
1587
|
+
) => {
|
|
1588
|
+
const { file, templatePath, context, templateConfig } = options;
|
|
1589
|
+
|
|
1590
|
+
const srcPath = path.join(templatePath, file);
|
|
1591
|
+
const destFile = convertDotfileName(file);
|
|
1592
|
+
|
|
1593
|
+
logger.verbose(
|
|
1594
|
+
` - Preparing: ${file}${destFile !== file ? ` -> ${destFile}` : ''}`,
|
|
1595
|
+
);
|
|
1596
|
+
|
|
1597
|
+
// 判断是否为二进制文件
|
|
1598
|
+
const isBinary = !shouldRenderFile(srcPath);
|
|
1599
|
+
let content;
|
|
1600
|
+
let wasRendered = false;
|
|
1601
|
+
|
|
1602
|
+
if (isBinary) {
|
|
1603
|
+
// 二进制文件,读取为 buffer 然后转为 base64
|
|
1604
|
+
const buffer = await fs$1.readFile(srcPath);
|
|
1605
|
+
content = buffer.toString('base64');
|
|
1606
|
+
} else {
|
|
1607
|
+
// 文本文件,渲染后的内容
|
|
1608
|
+
content = await renderTemplate(srcPath, context);
|
|
1609
|
+
wasRendered = true;
|
|
1610
|
+
}
|
|
1611
|
+
|
|
1612
|
+
// 构造文件信息对象
|
|
1613
|
+
const fileInfo = {
|
|
1614
|
+
path: file,
|
|
1615
|
+
destPath: destFile,
|
|
1616
|
+
content,
|
|
1617
|
+
isBinary,
|
|
1618
|
+
wasRendered,
|
|
1619
|
+
};
|
|
1620
|
+
|
|
1621
|
+
// 执行文件渲染钩子
|
|
1622
|
+
const processedFileInfo = await executeFileRenderHook(
|
|
1623
|
+
templateConfig,
|
|
1624
|
+
fileInfo,
|
|
1625
|
+
context,
|
|
1626
|
+
);
|
|
1627
|
+
|
|
1628
|
+
// 如果返回 null,表示该文件被 hook 跳过
|
|
1629
|
+
if (processedFileInfo === null) {
|
|
1630
|
+
logger.verbose(' ⊘ Skipped by onFileRender hook');
|
|
1631
|
+
return null;
|
|
1632
|
+
}
|
|
1633
|
+
|
|
1634
|
+
return processedFileInfo;
|
|
1635
|
+
};
|
|
1636
|
+
|
|
1637
|
+
/**
|
|
1638
|
+
* 写入渲染后的文件到目标路径
|
|
1639
|
+
*
|
|
1640
|
+
* @param options - 写入选项
|
|
1641
|
+
*/
|
|
1642
|
+
const writeRenderedFile = async (options
|
|
1643
|
+
|
|
1644
|
+
|
|
1645
|
+
|
|
1646
|
+
) => {
|
|
1647
|
+
const { fileInfo, srcPath, destPath } = options;
|
|
1648
|
+
|
|
1649
|
+
logger.verbose(` - Writing: ${fileInfo.destPath}`);
|
|
1650
|
+
|
|
1651
|
+
// 确保目标目录存在
|
|
1652
|
+
await ensureDir(path.dirname(destPath));
|
|
1653
|
+
|
|
1654
|
+
// 写入文件
|
|
1655
|
+
if (fileInfo.isBinary) {
|
|
1656
|
+
// 二进制文件:如果内容是原始 base64(未被 hook 修改),直接复制;否则从 base64 解码写入
|
|
1657
|
+
const buffer = await fs$1.readFile(srcPath);
|
|
1658
|
+
const originalContent = buffer.toString('base64');
|
|
1659
|
+
|
|
1660
|
+
if (fileInfo.content === originalContent) {
|
|
1661
|
+
await fs$1.copyFile(srcPath, destPath);
|
|
1662
|
+
logger.verbose(' ✓ Copied (binary)');
|
|
1663
|
+
} else {
|
|
1664
|
+
const modifiedBuffer = Buffer.from(fileInfo.content, 'base64');
|
|
1665
|
+
await fs$1.writeFile(destPath, modifiedBuffer);
|
|
1666
|
+
logger.verbose(' ✓ Written (binary, modified by hook)');
|
|
1667
|
+
}
|
|
1668
|
+
} else {
|
|
1669
|
+
// 文本文件
|
|
1670
|
+
await fs$1.writeFile(destPath, fileInfo.content, 'utf-8');
|
|
1671
|
+
logger.verbose(' ✓ Rendered and written');
|
|
1672
|
+
}
|
|
1673
|
+
};
|
|
1674
|
+
// end_aigc
|
|
1675
|
+
|
|
1676
|
+
// ABOUTME: File conflict detection utilities for template processing
|
|
1677
|
+
// ABOUTME: Provides dry-run file collection and conflict checking
|
|
1678
|
+
|
|
1679
|
+
|
|
1680
|
+
|
|
1681
|
+
|
|
1682
|
+
|
|
1683
|
+
|
|
1684
|
+
|
|
1685
|
+
// start_aigc
|
|
1686
|
+
/**
|
|
1687
|
+
* 收集所有将要写入的文件路径(dry-run 阶段)
|
|
1688
|
+
*
|
|
1689
|
+
* @param options - 收集选项
|
|
1690
|
+
* @returns 将要写入的文件路径列表
|
|
1691
|
+
*/
|
|
1692
|
+
const collectFilesToRender = async (options
|
|
1693
|
+
|
|
1694
|
+
|
|
1695
|
+
|
|
1696
|
+
|
|
1697
|
+
) => {
|
|
1698
|
+
const { files, templatePath, context, templateConfig } = options;
|
|
1699
|
+
|
|
1700
|
+
logger.verbose('\nDry-run: Collecting files to render...');
|
|
1701
|
+
|
|
1702
|
+
const fileInfos = await Promise.all(
|
|
1703
|
+
files.map(file =>
|
|
1704
|
+
prepareFileInfo({
|
|
1705
|
+
file,
|
|
1706
|
+
templatePath,
|
|
1707
|
+
context,
|
|
1708
|
+
templateConfig,
|
|
1709
|
+
}),
|
|
1710
|
+
),
|
|
1711
|
+
);
|
|
1712
|
+
|
|
1713
|
+
// 过滤掉被 hook 跳过的文件,收集 destPath
|
|
1714
|
+
const filesToWrite = fileInfos
|
|
1715
|
+
.filter((info) => info !== null)
|
|
1716
|
+
.map(info => info.destPath);
|
|
1717
|
+
|
|
1718
|
+
logger.verbose(` - ${filesToWrite.length} files will be written`);
|
|
1719
|
+
logger.verbose(
|
|
1720
|
+
` - ${fileInfos.length - filesToWrite.length} files skipped by hooks`,
|
|
1721
|
+
);
|
|
1722
|
+
|
|
1723
|
+
return filesToWrite;
|
|
1724
|
+
};
|
|
1725
|
+
// end_aigc
|
|
1726
|
+
|
|
1727
|
+
// ABOUTME: Main file processing orchestration for template rendering
|
|
1728
|
+
// ABOUTME: Coordinates file system, rendering, and conflict detection layers
|
|
1729
|
+
|
|
1730
|
+
|
|
1731
|
+
|
|
1732
|
+
// start_aigc
|
|
1733
|
+
/**
|
|
1734
|
+
* 处理单个文件(准备 + 写入)
|
|
1735
|
+
*
|
|
1736
|
+
* @param options - 处理选项
|
|
1737
|
+
*/
|
|
1738
|
+
const processSingleFile = async (options
|
|
1739
|
+
|
|
1740
|
+
|
|
1741
|
+
|
|
1742
|
+
|
|
1743
|
+
|
|
1744
|
+
) => {
|
|
1745
|
+
const { file, templatePath, outputPath, context, templateConfig } = options;
|
|
1746
|
+
|
|
1747
|
+
const srcPath = path.join(templatePath, file);
|
|
1748
|
+
|
|
1749
|
+
// 准备文件信息
|
|
1750
|
+
const processedFileInfo = await prepareFileInfo({
|
|
1751
|
+
file,
|
|
1752
|
+
templatePath,
|
|
1753
|
+
context,
|
|
1754
|
+
templateConfig,
|
|
1755
|
+
});
|
|
1756
|
+
|
|
1757
|
+
// 如果返回 null,跳过该文件
|
|
1758
|
+
if (processedFileInfo === null) {
|
|
1759
|
+
return;
|
|
1760
|
+
}
|
|
1761
|
+
|
|
1762
|
+
// 使用处理后的目标路径
|
|
1763
|
+
const finalDestPath = path.join(outputPath, processedFileInfo.destPath);
|
|
1764
|
+
|
|
1765
|
+
// 写入文件
|
|
1766
|
+
await writeRenderedFile({
|
|
1767
|
+
fileInfo: processedFileInfo,
|
|
1768
|
+
srcPath,
|
|
1769
|
+
destPath: finalDestPath,
|
|
1770
|
+
});
|
|
1771
|
+
};
|
|
1772
|
+
|
|
1773
|
+
/**
|
|
1774
|
+
* 复制并处理模板文件到目标目录
|
|
1775
|
+
*
|
|
1776
|
+
* 流程:
|
|
1777
|
+
* 1. 验证模板目录
|
|
1778
|
+
* 2. 扫描所有模板文件
|
|
1779
|
+
* 3. Dry-run:收集将要写入的文件列表(考虑 hooks 影响)
|
|
1780
|
+
* 4. 冲突检测:检查是否有文件会被覆盖(可通过 force 跳过)
|
|
1781
|
+
* 5. 实际写入:渲染并写入所有文件
|
|
1782
|
+
* 6. 复制 node_modules(如果存在)
|
|
1783
|
+
*
|
|
1784
|
+
* @param options - 处理选项
|
|
1785
|
+
*/
|
|
1786
|
+
const processTemplateFiles = async (options
|
|
1787
|
+
|
|
1788
|
+
|
|
1789
|
+
|
|
1790
|
+
|
|
1791
|
+
|
|
1792
|
+
) => {
|
|
1793
|
+
const { templatePath, outputPath, context, templateConfig} = options;
|
|
1794
|
+
logger.verbose('Processing template files:');
|
|
1795
|
+
logger.verbose(` - Template path: ${templatePath}`);
|
|
1796
|
+
logger.verbose(` - Output path: ${outputPath}`);
|
|
1797
|
+
|
|
1798
|
+
// 阶段 0: 验证模板目录是否存在
|
|
1799
|
+
try {
|
|
1800
|
+
const stat = await fs$1.stat(templatePath);
|
|
1801
|
+
logger.verbose(
|
|
1802
|
+
` - Template path exists: ${stat.isDirectory() ? 'directory' : 'file'}`,
|
|
1803
|
+
);
|
|
1804
|
+
if (!stat.isDirectory()) {
|
|
1805
|
+
throw new Error(`Template path is not a directory: ${templatePath}`);
|
|
1806
|
+
}
|
|
1807
|
+
} catch (error) {
|
|
1808
|
+
logger.error(
|
|
1809
|
+
` - Failed to access template path: ${error instanceof Error ? error.message : String(error)}`,
|
|
1810
|
+
);
|
|
1811
|
+
throw error;
|
|
1812
|
+
}
|
|
1813
|
+
|
|
1814
|
+
// 阶段 1: 扫描所有模板文件
|
|
1815
|
+
const files = await getAllFiles(templatePath);
|
|
1816
|
+
|
|
1817
|
+
logger.verbose(` - Found ${files.length} files to process`);
|
|
1818
|
+
|
|
1819
|
+
if (files.length === 0) {
|
|
1820
|
+
logger.warn(' - No files found in template directory!');
|
|
1821
|
+
return;
|
|
1822
|
+
}
|
|
1823
|
+
|
|
1824
|
+
// 阶段 2: Dry-run - 收集所有将要写入的文件
|
|
1825
|
+
await collectFilesToRender({
|
|
1826
|
+
files,
|
|
1827
|
+
templatePath,
|
|
1828
|
+
context,
|
|
1829
|
+
templateConfig,
|
|
1830
|
+
});
|
|
1831
|
+
|
|
1832
|
+
// 阶段 3: 冲突检测(force 为 true 时跳过)
|
|
1833
|
+
{
|
|
1834
|
+
logger.verbose(
|
|
1835
|
+
' - Force mode enabled, skipping conflict detection. Existing files will be overwritten.',
|
|
1836
|
+
);
|
|
1837
|
+
}
|
|
1838
|
+
|
|
1839
|
+
// 阶段 4: 实际写入文件
|
|
1840
|
+
logger.verbose('\nWriting files...');
|
|
1841
|
+
await Promise.all(
|
|
1842
|
+
files.map(file =>
|
|
1843
|
+
processSingleFile({
|
|
1844
|
+
file,
|
|
1845
|
+
templatePath,
|
|
1846
|
+
outputPath,
|
|
1847
|
+
context,
|
|
1848
|
+
templateConfig,
|
|
1849
|
+
}),
|
|
1850
|
+
),
|
|
1851
|
+
);
|
|
1852
|
+
|
|
1853
|
+
logger.verbose('✓ All files processed successfully');
|
|
1854
|
+
|
|
1855
|
+
// node_modules 将由 pnpm install 处理(利用缓存和硬链接机制)
|
|
1856
|
+
};
|
|
1857
|
+
// end_aigc
|
|
1858
|
+
|
|
1859
|
+
/**
|
|
1860
|
+
* 模板引擎执行选项
|
|
1861
|
+
*/
|
|
1862
|
+
|
|
1863
|
+
|
|
1864
|
+
|
|
1865
|
+
|
|
1866
|
+
|
|
1867
|
+
|
|
1868
|
+
|
|
1869
|
+
/**
|
|
1870
|
+
* 加载模板元数据和路径
|
|
1871
|
+
*/
|
|
1872
|
+
const loadTemplateMetadata = async (templateName) => {
|
|
1873
|
+
const templatesConfigPath = getTemplatesConfigPath();
|
|
1874
|
+
const templatesConfig = await loadTemplatesConfig(templatesConfigPath);
|
|
1875
|
+
const templateMetadata = findTemplate(templatesConfig, templateName);
|
|
1876
|
+
const templatesDir = getTemplatesDir();
|
|
1877
|
+
const templatePath = await getTemplatePath(templatesDir, templateMetadata);
|
|
1878
|
+
|
|
1879
|
+
return { templatePath, templateMetadata };
|
|
1880
|
+
};
|
|
1881
|
+
|
|
1882
|
+
/**
|
|
1883
|
+
* 解析并验证模板参数
|
|
1884
|
+
*/
|
|
1885
|
+
const parseAndValidateParams = (
|
|
1886
|
+
command,
|
|
1887
|
+
templateConfig,
|
|
1888
|
+
) => {
|
|
1889
|
+
const knownOptions = new Set([
|
|
1890
|
+
'template',
|
|
1891
|
+
't',
|
|
1892
|
+
'output',
|
|
1893
|
+
'o',
|
|
1894
|
+
'skipInstall',
|
|
1895
|
+
'skip-install',
|
|
1896
|
+
'skipGit',
|
|
1897
|
+
'skip-git',
|
|
1898
|
+
'skipDev',
|
|
1899
|
+
'skip-dev',
|
|
1900
|
+
]);
|
|
1901
|
+
const userParams = parsePassThroughParams(command, knownOptions);
|
|
1902
|
+
|
|
1903
|
+
// 合并默认参数:defaultParams < schema defaults < userParams
|
|
1904
|
+
const paramsWithDefaults = {
|
|
1905
|
+
...(templateConfig.defaultParams || {}),
|
|
1906
|
+
...userParams,
|
|
1907
|
+
};
|
|
1908
|
+
|
|
1909
|
+
return validateParams(templateConfig.paramsSchema, paramsWithDefaults);
|
|
1910
|
+
};
|
|
1911
|
+
|
|
1912
|
+
/**
|
|
1913
|
+
* 执行生命周期钩子
|
|
1914
|
+
*/
|
|
1915
|
+
const executeBeforeRenderHook = async (
|
|
1916
|
+
templateConfig,
|
|
1917
|
+
context,
|
|
1918
|
+
) => {
|
|
1919
|
+
if (!templateConfig.onBeforeRender) {
|
|
1920
|
+
return context;
|
|
1921
|
+
}
|
|
1922
|
+
return (await templateConfig.onBeforeRender(context)) ;
|
|
1923
|
+
};
|
|
1924
|
+
|
|
1925
|
+
/**
|
|
1926
|
+
* 执行渲染后钩子
|
|
1927
|
+
*/
|
|
1928
|
+
const executeAfterRenderHook = async (
|
|
1929
|
+
templateConfig,
|
|
1930
|
+
context,
|
|
1931
|
+
outputPath,
|
|
1932
|
+
) => {
|
|
1933
|
+
if (templateConfig.onAfterRender) {
|
|
1934
|
+
await templateConfig.onAfterRender(context, outputPath);
|
|
1935
|
+
}
|
|
1936
|
+
};
|
|
1937
|
+
|
|
1938
|
+
/**
|
|
1939
|
+
* 执行完成钩子
|
|
1940
|
+
*/
|
|
1941
|
+
const executeCompleteHook = async (
|
|
1942
|
+
templateConfig,
|
|
1943
|
+
context,
|
|
1944
|
+
outputPath,
|
|
1945
|
+
) => {
|
|
1946
|
+
if (templateConfig.onComplete) {
|
|
1947
|
+
await templateConfig.onComplete(context, outputPath);
|
|
1948
|
+
}
|
|
1949
|
+
};
|
|
1950
|
+
|
|
1951
|
+
/**
|
|
1952
|
+
* 准备输出目录
|
|
1953
|
+
*/
|
|
1954
|
+
const prepareOutputDirectory = (outputPath) => {
|
|
1955
|
+
const absolutePath = path.resolve(process.cwd(), outputPath);
|
|
1956
|
+
// 不再在这里验证目录是否为空,冲突检测已移至 processTemplateFiles 中
|
|
1957
|
+
return absolutePath;
|
|
1958
|
+
};
|
|
1959
|
+
|
|
1960
|
+
/**
|
|
1961
|
+
* 模板引擎执行结果
|
|
1962
|
+
*/
|
|
1963
|
+
|
|
1964
|
+
|
|
1965
|
+
|
|
1966
|
+
|
|
1967
|
+
|
|
1968
|
+
|
|
1969
|
+
|
|
1970
|
+
|
|
1971
|
+
|
|
1972
|
+
/**
|
|
1973
|
+
* 执行完整的模板渲染流程
|
|
1974
|
+
*/
|
|
1975
|
+
const execute = async (
|
|
1976
|
+
options,
|
|
1977
|
+
) => {
|
|
1978
|
+
const { templateName, outputPath, command} = options;
|
|
1979
|
+
|
|
1980
|
+
// 1. 加载模板
|
|
1981
|
+
const { templatePath } = await loadTemplateMetadata(templateName);
|
|
1982
|
+
|
|
1983
|
+
// 2. 加载模板配置
|
|
1984
|
+
const templateConfig = await loadTemplateConfig(templatePath);
|
|
1985
|
+
|
|
1986
|
+
// 3. 解析并验证参数
|
|
1987
|
+
const validatedParams = parseAndValidateParams(command, templateConfig);
|
|
1988
|
+
|
|
1989
|
+
// 4. 执行 onBeforeRender 钩子
|
|
1990
|
+
const context = await executeBeforeRenderHook(templateConfig, {
|
|
1991
|
+
...validatedParams,
|
|
1992
|
+
});
|
|
1993
|
+
|
|
1994
|
+
// 5. 准备输出目录
|
|
1995
|
+
const absoluteOutputPath = prepareOutputDirectory(outputPath);
|
|
1996
|
+
|
|
1997
|
+
// 6. 处理模板文件
|
|
1998
|
+
await processTemplateFiles({
|
|
1999
|
+
templatePath,
|
|
2000
|
+
outputPath: absoluteOutputPath,
|
|
2001
|
+
context,
|
|
2002
|
+
templateConfig});
|
|
2003
|
+
|
|
2004
|
+
// 7. 执行 onAfterRender 钩子
|
|
2005
|
+
await executeAfterRenderHook(templateConfig, context, absoluteOutputPath);
|
|
2006
|
+
|
|
2007
|
+
return {
|
|
2008
|
+
outputPath: absoluteOutputPath,
|
|
2009
|
+
templateConfig,
|
|
2010
|
+
context,
|
|
2011
|
+
};
|
|
2012
|
+
};
|
|
2013
|
+
|
|
2014
|
+
function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } }
|
|
2015
|
+
/**
|
|
2016
|
+
* 运行 pnpm install
|
|
2017
|
+
*/
|
|
2018
|
+
const runPnpmInstall = (projectPath) => {
|
|
2019
|
+
logger.info('\nInstalling dependencies with pnpm...');
|
|
2020
|
+
logger.info(`Executing: pnpm install in ${projectPath}`);
|
|
2021
|
+
|
|
2022
|
+
const result = shelljs.exec('pnpm install', {
|
|
2023
|
+
cwd: projectPath,
|
|
2024
|
+
silent: true, // 使用 silent 来捕获输出
|
|
2025
|
+
});
|
|
2026
|
+
|
|
2027
|
+
// 输出 stdout
|
|
2028
|
+
if (result.stdout) {
|
|
2029
|
+
process.stdout.write(result.stdout);
|
|
2030
|
+
}
|
|
2031
|
+
|
|
2032
|
+
// 输出 stderr
|
|
2033
|
+
if (result.stderr) {
|
|
2034
|
+
process.stderr.write(result.stderr);
|
|
2035
|
+
}
|
|
2036
|
+
|
|
2037
|
+
if (result.code === 0) {
|
|
2038
|
+
logger.success('Dependencies installed successfully!');
|
|
2039
|
+
} else {
|
|
2040
|
+
const errorMessage = [
|
|
2041
|
+
`pnpm install failed with exit code ${result.code}`,
|
|
2042
|
+
result.stderr ? `\nStderr:\n${result.stderr}` : '',
|
|
2043
|
+
result.stdout ? `\nStdout:\n${result.stdout}` : '',
|
|
2044
|
+
]
|
|
2045
|
+
.filter(Boolean)
|
|
2046
|
+
.join('');
|
|
2047
|
+
|
|
2048
|
+
throw new Error(errorMessage);
|
|
2049
|
+
}
|
|
2050
|
+
};
|
|
2051
|
+
|
|
2052
|
+
/**
|
|
2053
|
+
* 运行 git 命令的辅助函数
|
|
2054
|
+
*/
|
|
2055
|
+
const runGitCommand = (command, projectPath) => {
|
|
2056
|
+
logger.info(`Executing: ${command}`);
|
|
2057
|
+
|
|
2058
|
+
const result = shelljs.exec(command, {
|
|
2059
|
+
cwd: projectPath,
|
|
2060
|
+
silent: true,
|
|
2061
|
+
});
|
|
2062
|
+
|
|
2063
|
+
// 输出命令的结果
|
|
2064
|
+
if (result.stdout) {
|
|
2065
|
+
process.stdout.write(result.stdout);
|
|
2066
|
+
}
|
|
2067
|
+
|
|
2068
|
+
if (result.stderr) {
|
|
2069
|
+
process.stderr.write(result.stderr);
|
|
2070
|
+
}
|
|
2071
|
+
|
|
2072
|
+
if (result.code !== 0) {
|
|
2073
|
+
const errorMessage = [
|
|
2074
|
+
`${command} failed with exit code ${result.code}`,
|
|
2075
|
+
result.stderr ? `\nStderr:\n${result.stderr}` : '',
|
|
2076
|
+
result.stdout ? `\nStdout:\n${result.stdout}` : '',
|
|
2077
|
+
]
|
|
2078
|
+
.filter(Boolean)
|
|
2079
|
+
.join('');
|
|
2080
|
+
|
|
2081
|
+
throw new Error(errorMessage);
|
|
2082
|
+
}
|
|
2083
|
+
};
|
|
2084
|
+
|
|
2085
|
+
/**
|
|
2086
|
+
* 初始化 git 仓库
|
|
2087
|
+
* 如果目录中已存在 .git,则跳过初始化
|
|
2088
|
+
*/
|
|
2089
|
+
const runGitInit = (projectPath) => {
|
|
2090
|
+
// 检查是否已存在 .git 目录
|
|
2091
|
+
const gitDir = path.join(projectPath, '.git');
|
|
2092
|
+
if (fs.existsSync(gitDir)) {
|
|
2093
|
+
logger.info(
|
|
2094
|
+
'\n💡 Git repository already exists, skipping git initialization',
|
|
2095
|
+
);
|
|
2096
|
+
return;
|
|
2097
|
+
}
|
|
2098
|
+
|
|
2099
|
+
try {
|
|
2100
|
+
logger.info('\nInitializing git repository...');
|
|
2101
|
+
runGitCommand('git init', projectPath);
|
|
2102
|
+
logger.success('Git repository initialized successfully!');
|
|
2103
|
+
} catch (error) {
|
|
2104
|
+
// Git 初始化失败不应该导致整个流程失败
|
|
2105
|
+
logger.warn(
|
|
2106
|
+
`Git initialization failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
2107
|
+
);
|
|
2108
|
+
logger.info('You can manually initialize git later with: git init');
|
|
2109
|
+
}
|
|
2110
|
+
};
|
|
2111
|
+
|
|
2112
|
+
/**
|
|
2113
|
+
* 提交初始化生成的所有文件
|
|
2114
|
+
*/
|
|
2115
|
+
const commitChanges = (projectPath) => {
|
|
2116
|
+
// 检查是否存在 .git 目录
|
|
2117
|
+
const gitDir = path.join(projectPath, '.git');
|
|
2118
|
+
if (!fs.existsSync(gitDir)) {
|
|
2119
|
+
logger.warn(
|
|
2120
|
+
'\n⚠️ Git repository does not exist, skipping commit. Run git init first.',
|
|
2121
|
+
);
|
|
2122
|
+
return;
|
|
2123
|
+
}
|
|
2124
|
+
|
|
2125
|
+
try {
|
|
2126
|
+
logger.info('\nCommitting initialized files...');
|
|
2127
|
+
runGitCommand('git add --all', projectPath);
|
|
2128
|
+
runGitCommand('git commit -m "chore: init env"', projectPath);
|
|
2129
|
+
logger.success('Changes committed successfully!');
|
|
2130
|
+
} catch (error) {
|
|
2131
|
+
// Commit 失败不应该导致整个流程失败
|
|
2132
|
+
logger.warn(
|
|
2133
|
+
`Git commit failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
2134
|
+
);
|
|
2135
|
+
logger.info(
|
|
2136
|
+
'You can manually commit later with: git add --all && git commit -m "chore: init env"',
|
|
2137
|
+
);
|
|
2138
|
+
}
|
|
2139
|
+
};
|
|
2140
|
+
|
|
2141
|
+
/**
|
|
2142
|
+
* 运行开发服务器(后台模式)
|
|
2143
|
+
* 启动后台子进程运行开发服务器,父进程可以直接退出
|
|
2144
|
+
* 使用 CLI 自己的 dev 命令(定义在 run.ts)而不是直接运行 npm run dev
|
|
2145
|
+
*/
|
|
2146
|
+
const runDev = (projectPath) => {
|
|
2147
|
+
logger.info('\nStarting development server in background...');
|
|
2148
|
+
|
|
2149
|
+
// 获取当前 CLI 的可执行文件路径
|
|
2150
|
+
// process.argv[0] 是 node,process.argv[1] 是 CLI 入口文件
|
|
2151
|
+
const cliPath = process.argv[1];
|
|
2152
|
+
|
|
2153
|
+
logger.info(`Executing: ${cliPath} dev in ${projectPath}`);
|
|
2154
|
+
|
|
2155
|
+
// 使用通用的后台执行函数启动开发服务器
|
|
2156
|
+
// 调用 CLI 自己的 dev 命令
|
|
2157
|
+
const pid = spawnDetached(process.argv[0], [cliPath, 'dev'], {
|
|
2158
|
+
cwd: projectPath,
|
|
2159
|
+
verbose: false, // 不输出额外的进程信息,由 logger 统一处理
|
|
2160
|
+
});
|
|
2161
|
+
|
|
2162
|
+
logger.success('Development server started in background!');
|
|
2163
|
+
if (pid) {
|
|
2164
|
+
logger.info(`Process ID: ${pid}`);
|
|
2165
|
+
logger.info(
|
|
2166
|
+
'\nThe dev server is running independently. You can close this terminal.',
|
|
2167
|
+
);
|
|
2168
|
+
logger.info(`To stop the server later, use: kill ${pid}`);
|
|
2169
|
+
}
|
|
2170
|
+
};
|
|
2171
|
+
|
|
2172
|
+
/**
|
|
2173
|
+
* 执行 init 命令的内部实现
|
|
2174
|
+
*/
|
|
2175
|
+
const executeInit = async (
|
|
2176
|
+
options
|
|
2177
|
+
|
|
2178
|
+
|
|
2179
|
+
|
|
2180
|
+
|
|
2181
|
+
|
|
2182
|
+
|
|
2183
|
+
|
|
2184
|
+
,
|
|
2185
|
+
command,
|
|
2186
|
+
) => {
|
|
2187
|
+
const timer = new TimeTracker();
|
|
2188
|
+
|
|
2189
|
+
try {
|
|
2190
|
+
const {
|
|
2191
|
+
template: templateName,
|
|
2192
|
+
output: outputPath,
|
|
2193
|
+
skipInstall,
|
|
2194
|
+
skipGit,
|
|
2195
|
+
skipCommit,
|
|
2196
|
+
skipDev,
|
|
2197
|
+
force,
|
|
2198
|
+
} = options;
|
|
2199
|
+
|
|
2200
|
+
logger.info(`Initializing project with template: ${templateName}`);
|
|
2201
|
+
timer.logPhase('Initialization');
|
|
2202
|
+
|
|
2203
|
+
// 执行模板引擎,返回结果对象
|
|
2204
|
+
const result = await execute({
|
|
2205
|
+
templateName,
|
|
2206
|
+
outputPath,
|
|
2207
|
+
command,
|
|
2208
|
+
force,
|
|
2209
|
+
});
|
|
2210
|
+
const { outputPath: absoluteOutputPath, templateConfig, context } = result;
|
|
2211
|
+
|
|
2212
|
+
timer.logPhase('Template engine execution');
|
|
2213
|
+
logger.success('Project created successfully!');
|
|
2214
|
+
|
|
2215
|
+
// 检查是否存在 package.json
|
|
2216
|
+
const packageJsonPath = path.join(absoluteOutputPath, 'package.json');
|
|
2217
|
+
const hasPackageJson = fs.existsSync(packageJsonPath);
|
|
2218
|
+
|
|
2219
|
+
// 安装依赖(始终使用 pnpm install,利用缓存机制)
|
|
2220
|
+
if (!skipInstall) {
|
|
2221
|
+
if (hasPackageJson) {
|
|
2222
|
+
runPnpmInstall(absoluteOutputPath);
|
|
2223
|
+
timer.logPhase('Dependencies installation');
|
|
2224
|
+
} else {
|
|
2225
|
+
logger.info(
|
|
2226
|
+
'\n💡 No package.json found, skipping dependency installation',
|
|
2227
|
+
);
|
|
2228
|
+
}
|
|
2229
|
+
}
|
|
2230
|
+
|
|
2231
|
+
// 执行 onComplete 钩子(在 pnpm install 之后)
|
|
2232
|
+
await executeCompleteHook(templateConfig, context, absoluteOutputPath);
|
|
2233
|
+
timer.logPhase('Complete hook execution');
|
|
2234
|
+
|
|
2235
|
+
// 如果没有跳过 git,则初始化 git 仓库
|
|
2236
|
+
if (!skipGit) {
|
|
2237
|
+
runGitInit(absoluteOutputPath);
|
|
2238
|
+
timer.logPhase('Git initialization');
|
|
2239
|
+
}
|
|
2240
|
+
|
|
2241
|
+
// 如果没有跳过 commit,则提交初始化生成的文件
|
|
2242
|
+
if (!skipCommit) {
|
|
2243
|
+
commitChanges(absoluteOutputPath);
|
|
2244
|
+
timer.logPhase('Git commit');
|
|
2245
|
+
}
|
|
2246
|
+
|
|
2247
|
+
// 如果没有跳过 dev,则启动开发服务器
|
|
2248
|
+
if (!skipDev) {
|
|
2249
|
+
runDev(absoluteOutputPath);
|
|
2250
|
+
timer.logPhase('Dev server startup');
|
|
2251
|
+
} else {
|
|
2252
|
+
// 只有跳过 dev 时才显示 Next steps
|
|
2253
|
+
logger.info('\nNext steps:');
|
|
2254
|
+
logger.info(` cd ${outputPath}`);
|
|
2255
|
+
if (skipInstall && hasPackageJson) {
|
|
2256
|
+
logger.info(' pnpm install');
|
|
2257
|
+
}
|
|
2258
|
+
if (skipGit) {
|
|
2259
|
+
logger.info(' git init');
|
|
2260
|
+
}
|
|
2261
|
+
if (skipCommit) {
|
|
2262
|
+
logger.info(' git add --all && git commit -m "chore: init env"');
|
|
2263
|
+
}
|
|
2264
|
+
logger.info(' coze dev');
|
|
2265
|
+
}
|
|
2266
|
+
|
|
2267
|
+
// 输出总耗时
|
|
2268
|
+
timer.logTotal();
|
|
2269
|
+
} catch (error) {
|
|
2270
|
+
timer.logTotal();
|
|
2271
|
+
logger.error('Failed to initialize project:');
|
|
2272
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
2273
|
+
process.exit(1);
|
|
2274
|
+
}
|
|
2275
|
+
};
|
|
2276
|
+
|
|
2277
|
+
/**
|
|
2278
|
+
* 注册 init 命令到 program
|
|
2279
|
+
*/
|
|
2280
|
+
const registerCommand$1 = program => {
|
|
2281
|
+
program
|
|
2282
|
+
.command('init')
|
|
2283
|
+
.description('Initialize a new project from a template')
|
|
2284
|
+
.argument('[directory]', 'Output directory for the project')
|
|
2285
|
+
.requiredOption('-t, --template <name>', 'Template name')
|
|
2286
|
+
.option('-o, --output <path>', 'Output directory', process.cwd())
|
|
2287
|
+
.option('--skip-install', 'Skip automatic pnpm install', false)
|
|
2288
|
+
.option('--skip-git', 'Skip automatic git initialization', false)
|
|
2289
|
+
.option(
|
|
2290
|
+
'--skip-commit',
|
|
2291
|
+
'Skip automatic git commit after initialization',
|
|
2292
|
+
false,
|
|
2293
|
+
)
|
|
2294
|
+
.option('--skip-dev', 'Skip automatic dev server start', false)
|
|
2295
|
+
.allowUnknownOption() // 允许透传参数
|
|
2296
|
+
.action(async (directory, options, command) => {
|
|
2297
|
+
// 位置参数优先级高于 --output 选项
|
|
2298
|
+
const outputPath = _nullishCoalesce(directory, () => ( options.output));
|
|
2299
|
+
// Always use force mode - overwrite existing files without conflict check
|
|
2300
|
+
const force = true;
|
|
2301
|
+
await executeInit({ ...options, output: outputPath, force }, command);
|
|
2302
|
+
});
|
|
2303
|
+
};
|
|
2304
|
+
|
|
2305
|
+
// ABOUTME: This file implements the update command for coze CLI
|
|
2306
|
+
// ABOUTME: It wraps pnpm update/install to update package dependencies with logging support
|
|
2307
|
+
|
|
2308
|
+
|
|
2309
|
+
|
|
2310
|
+
|
|
2311
|
+
/**
|
|
2312
|
+
* 日志文件名常量
|
|
2313
|
+
*/
|
|
2314
|
+
const LOG_FILE_NAME = 'update.log';
|
|
2315
|
+
|
|
2316
|
+
/**
|
|
2317
|
+
* 获取日志目录
|
|
2318
|
+
* 优先使用环境变量 COZE_LOG_DIR,否则使用 ~/.coze-logs
|
|
2319
|
+
*/
|
|
2320
|
+
const getLogDir = () =>
|
|
2321
|
+
process.env.COZE_LOG_DIR || path.join(os.homedir(), '.coze-logs');
|
|
2322
|
+
|
|
2323
|
+
/**
|
|
2324
|
+
* 解析日志文件路径
|
|
2325
|
+
* - 如果是绝对路径,直接使用
|
|
2326
|
+
* - 如果是相对路径,基于 getLogDir() + 相对路径
|
|
2327
|
+
* - 如果为空,使用 getLogDir() + LOG_FILE_NAME
|
|
2328
|
+
*/
|
|
2329
|
+
const resolveLogFilePath = (logFile) => {
|
|
2330
|
+
if (!logFile) {
|
|
2331
|
+
return path.join(getLogDir(), LOG_FILE_NAME);
|
|
2332
|
+
}
|
|
2333
|
+
|
|
2334
|
+
if (path.isAbsolute(logFile)) {
|
|
2335
|
+
return logFile;
|
|
2336
|
+
}
|
|
2337
|
+
|
|
2338
|
+
return path.join(getLogDir(), logFile);
|
|
2339
|
+
};
|
|
2340
|
+
|
|
2341
|
+
/**
|
|
2342
|
+
* 创建日志写入流
|
|
2343
|
+
*/
|
|
2344
|
+
const createLogStream = (logFilePath) => {
|
|
2345
|
+
const logDir = path.dirname(logFilePath);
|
|
2346
|
+
|
|
2347
|
+
// 确保日志目录存在
|
|
2348
|
+
if (!fs.existsSync(logDir)) {
|
|
2349
|
+
fs.mkdirSync(logDir, { recursive: true });
|
|
2350
|
+
}
|
|
2351
|
+
|
|
2352
|
+
// 使用 'w' 标志覆盖之前的日志
|
|
2353
|
+
return fs.createWriteStream(logFilePath, { flags: 'w' });
|
|
2354
|
+
};
|
|
2355
|
+
|
|
2356
|
+
/**
|
|
2357
|
+
* 格式化时间戳
|
|
2358
|
+
*/
|
|
2359
|
+
const formatTimestamp = () => {
|
|
2360
|
+
const now = new Date();
|
|
2361
|
+
return now.toISOString();
|
|
2362
|
+
};
|
|
2363
|
+
|
|
2364
|
+
/**
|
|
2365
|
+
* 写入带时间戳的日志
|
|
2366
|
+
*/
|
|
2367
|
+
const writeLogWithTimestamp = (stream, message) => {
|
|
2368
|
+
const timestamp = formatTimestamp();
|
|
2369
|
+
const lines = message.split('\n');
|
|
2370
|
+
lines.forEach(line => {
|
|
2371
|
+
if (line) {
|
|
2372
|
+
stream.write(`[${timestamp}] ${line}\n`);
|
|
2373
|
+
} else {
|
|
2374
|
+
stream.write('\n');
|
|
2375
|
+
}
|
|
2376
|
+
});
|
|
2377
|
+
// 确保数据写入磁盘
|
|
2378
|
+
stream.uncork();
|
|
2379
|
+
};
|
|
2380
|
+
|
|
2381
|
+
/**
|
|
2382
|
+
* 同时输出到控制台和日志文件
|
|
2383
|
+
*/
|
|
2384
|
+
const logWithFile = (
|
|
2385
|
+
stream,
|
|
2386
|
+
level,
|
|
2387
|
+
message,
|
|
2388
|
+
) => {
|
|
2389
|
+
// 输出到控制台
|
|
2390
|
+
switch (level) {
|
|
2391
|
+
case 'info':
|
|
2392
|
+
logger.info(message);
|
|
2393
|
+
break;
|
|
2394
|
+
case 'success':
|
|
2395
|
+
logger.success(message);
|
|
2396
|
+
break;
|
|
2397
|
+
case 'error':
|
|
2398
|
+
logger.error(message);
|
|
2399
|
+
break;
|
|
2400
|
+
default:
|
|
2401
|
+
logger.info(message);
|
|
2402
|
+
break;
|
|
2403
|
+
}
|
|
2404
|
+
|
|
2405
|
+
// 写入日志文件(带时间戳)
|
|
2406
|
+
writeLogWithTimestamp(stream, `[${level.toUpperCase()}] ${message}`);
|
|
2407
|
+
};
|
|
2408
|
+
|
|
2409
|
+
// start_aigc
|
|
2410
|
+
/**
|
|
2411
|
+
* 构建 pnpm add 命令
|
|
2412
|
+
*/
|
|
2413
|
+
const buildPnpmCommand = (
|
|
2414
|
+
packageName,
|
|
2415
|
+
options
|
|
2416
|
+
|
|
2417
|
+
|
|
2418
|
+
|
|
2419
|
+
|
|
2420
|
+
,
|
|
2421
|
+
) => {
|
|
2422
|
+
const { global, version, registry, extraArgs } = options;
|
|
2423
|
+
|
|
2424
|
+
const parts = ['pnpm', 'add'];
|
|
2425
|
+
|
|
2426
|
+
// 添加全局标记
|
|
2427
|
+
if (global) {
|
|
2428
|
+
parts.push('-g');
|
|
2429
|
+
}
|
|
2430
|
+
|
|
2431
|
+
// 添加包名和版本
|
|
2432
|
+
if (version && version !== 'latest') {
|
|
2433
|
+
parts.push(`${packageName}@${version}`);
|
|
2434
|
+
} else {
|
|
2435
|
+
parts.push(`${packageName}@latest`);
|
|
2436
|
+
}
|
|
2437
|
+
|
|
2438
|
+
// 添加 registry
|
|
2439
|
+
if (registry) {
|
|
2440
|
+
parts.push(`--registry=${registry}`);
|
|
2441
|
+
}
|
|
2442
|
+
|
|
2443
|
+
// 添加额外参数
|
|
2444
|
+
if (extraArgs.length > 0) {
|
|
2445
|
+
parts.push(...extraArgs);
|
|
2446
|
+
}
|
|
2447
|
+
|
|
2448
|
+
return parts.join(' ');
|
|
2449
|
+
};
|
|
2450
|
+
// end_aigc
|
|
2451
|
+
|
|
2452
|
+
// start_aigc
|
|
2453
|
+
/**
|
|
2454
|
+
* 执行 update 命令的内部实现
|
|
2455
|
+
*/
|
|
2456
|
+
const executeUpdate = (
|
|
2457
|
+
packageName,
|
|
2458
|
+
options
|
|
2459
|
+
|
|
2460
|
+
|
|
2461
|
+
|
|
2462
|
+
|
|
2463
|
+
|
|
2464
|
+
|
|
2465
|
+
,
|
|
2466
|
+
) => {
|
|
2467
|
+
let logStream = null;
|
|
2468
|
+
|
|
2469
|
+
try {
|
|
2470
|
+
const { global, cwd, version, registry, logFile, extraArgs } = options;
|
|
2471
|
+
|
|
2472
|
+
// 准备日志
|
|
2473
|
+
const logFilePath = resolveLogFilePath(logFile);
|
|
2474
|
+
|
|
2475
|
+
// 调试:确认日志路径
|
|
2476
|
+
logger.info(`Log file path resolved to: ${logFilePath}`);
|
|
2477
|
+
|
|
2478
|
+
logStream = createLogStream(logFilePath);
|
|
2479
|
+
|
|
2480
|
+
// 调试:确认流已创建
|
|
2481
|
+
logger.info('Log stream created successfully');
|
|
2482
|
+
|
|
2483
|
+
logWithFile(logStream, 'info', `Updating package: ${packageName}`);
|
|
2484
|
+
|
|
2485
|
+
// 构建命令
|
|
2486
|
+
const command = buildPnpmCommand(packageName, {
|
|
2487
|
+
global,
|
|
2488
|
+
version,
|
|
2489
|
+
registry,
|
|
2490
|
+
extraArgs,
|
|
2491
|
+
});
|
|
2492
|
+
|
|
2493
|
+
// 确定工作目录
|
|
2494
|
+
const workingDir = cwd
|
|
2495
|
+
? path.isAbsolute(cwd)
|
|
2496
|
+
? cwd
|
|
2497
|
+
: path.join(process.cwd(), cwd)
|
|
2498
|
+
: process.cwd();
|
|
2499
|
+
|
|
2500
|
+
logWithFile(logStream, 'info', `Executing: ${command}`);
|
|
2501
|
+
logWithFile(logStream, 'info', `Working directory: ${workingDir}`);
|
|
2502
|
+
logWithFile(logStream, 'info', `Log file: ${logFilePath}`);
|
|
2503
|
+
|
|
2504
|
+
// 记录命令开始时间
|
|
2505
|
+
writeLogWithTimestamp(logStream, '--- Command execution started ---');
|
|
2506
|
+
|
|
2507
|
+
// 同步执行命令
|
|
2508
|
+
const result = shelljs.exec(command, {
|
|
2509
|
+
cwd: workingDir,
|
|
2510
|
+
silent: true, // 使用 silent 来捕获输出
|
|
2511
|
+
});
|
|
2512
|
+
|
|
2513
|
+
// 将输出写入控制台和日志文件(带时间戳)
|
|
2514
|
+
if (result.stdout) {
|
|
2515
|
+
process.stdout.write(result.stdout);
|
|
2516
|
+
writeLogWithTimestamp(logStream, result.stdout.trim());
|
|
2517
|
+
}
|
|
2518
|
+
|
|
2519
|
+
if (result.stderr) {
|
|
2520
|
+
process.stderr.write(result.stderr);
|
|
2521
|
+
writeLogWithTimestamp(logStream, result.stderr.trim());
|
|
2522
|
+
}
|
|
2523
|
+
|
|
2524
|
+
// 记录命令结束时间
|
|
2525
|
+
writeLogWithTimestamp(logStream, '--- Command execution ended ---');
|
|
2526
|
+
|
|
2527
|
+
// 检查执行结果并记录到日志
|
|
2528
|
+
if (result.code === 0) {
|
|
2529
|
+
logWithFile(logStream, 'success', 'Package updated successfully');
|
|
2530
|
+
logWithFile(logStream, 'info', `Log file: ${logFilePath}`);
|
|
2531
|
+
} else {
|
|
2532
|
+
logWithFile(
|
|
2533
|
+
logStream,
|
|
2534
|
+
'error',
|
|
2535
|
+
`Command exited with code ${result.code}`,
|
|
2536
|
+
);
|
|
2537
|
+
logWithFile(
|
|
2538
|
+
logStream,
|
|
2539
|
+
'error',
|
|
2540
|
+
`Check log file for details: ${logFilePath}`,
|
|
2541
|
+
);
|
|
2542
|
+
}
|
|
2543
|
+
|
|
2544
|
+
// 关闭日志流并等待写入完成
|
|
2545
|
+
logStream.end(() => {
|
|
2546
|
+
// 流关闭后再退出进程
|
|
2547
|
+
if (result.code !== 0) {
|
|
2548
|
+
process.exit(result.code || 1);
|
|
2549
|
+
}
|
|
2550
|
+
});
|
|
2551
|
+
} catch (error) {
|
|
2552
|
+
logger.error('Failed to update package:');
|
|
2553
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
2554
|
+
|
|
2555
|
+
// 写入错误到日志文件
|
|
2556
|
+
if (logStream) {
|
|
2557
|
+
writeLogWithTimestamp(
|
|
2558
|
+
logStream,
|
|
2559
|
+
`[ERROR] ${error instanceof Error ? error.message : String(error)}`,
|
|
2560
|
+
);
|
|
2561
|
+
// 等待流关闭后再退出
|
|
2562
|
+
logStream.end(() => {
|
|
2563
|
+
process.exit(1);
|
|
2564
|
+
});
|
|
2565
|
+
} else {
|
|
2566
|
+
process.exit(1);
|
|
2567
|
+
}
|
|
2568
|
+
}
|
|
2569
|
+
};
|
|
2570
|
+
// end_aigc
|
|
2571
|
+
|
|
2572
|
+
/**
|
|
2573
|
+
* 注册 update 命令到 program
|
|
2574
|
+
*/
|
|
2575
|
+
const registerCommand = program => {
|
|
2576
|
+
program
|
|
2577
|
+
.command('update <package>')
|
|
2578
|
+
.description('Update a package dependency')
|
|
2579
|
+
.option('-g, --global', 'Update package globally', false)
|
|
2580
|
+
.option('-c, --cwd <path>', 'Working directory for the update')
|
|
2581
|
+
.option(
|
|
2582
|
+
'--to <version>',
|
|
2583
|
+
'Version to update to (default: latest)',
|
|
2584
|
+
'latest',
|
|
2585
|
+
)
|
|
2586
|
+
.option('--registry <url>', 'Registry URL to use for the update')
|
|
2587
|
+
.option('--log-file <path>', 'Log file path')
|
|
2588
|
+
.allowUnknownOption() // 允许透传参数给 pnpm
|
|
2589
|
+
.action((packageName, options, command) => {
|
|
2590
|
+
// 收集所有未知选项作为额外参数
|
|
2591
|
+
const extraArgs = command.args.slice(1);
|
|
2592
|
+
|
|
2593
|
+
executeUpdate(packageName, {
|
|
2594
|
+
...options,
|
|
2595
|
+
version: options.to, // 将 --to 映射到 version
|
|
2596
|
+
extraArgs,
|
|
2597
|
+
});
|
|
2598
|
+
});
|
|
2599
|
+
};
|
|
2600
|
+
|
|
2601
|
+
var version = "0.0.1-alpha.01c0ee";
|
|
2602
|
+
var packageJson = {
|
|
2603
|
+
version: version};
|
|
2604
|
+
|
|
2605
|
+
const commands = [
|
|
2606
|
+
registerCommand$1,
|
|
2607
|
+
registerCommand$2,
|
|
2608
|
+
registerCommand$4,
|
|
2609
|
+
registerCommand$3,
|
|
2610
|
+
registerCommand,
|
|
2611
|
+
];
|
|
2612
|
+
|
|
2613
|
+
const main = () => {
|
|
2614
|
+
const program = new commander.Command();
|
|
2615
|
+
|
|
2616
|
+
program
|
|
2617
|
+
.name('coze')
|
|
2618
|
+
.description(
|
|
2619
|
+
'Coze Coding CLI - Project template engine for frontend stacks',
|
|
2620
|
+
)
|
|
2621
|
+
.version(packageJson.version);
|
|
2622
|
+
|
|
2623
|
+
commands.forEach(initCmd => initCmd(program));
|
|
2624
|
+
|
|
2625
|
+
// 在 help 输出中添加模板信息
|
|
2626
|
+
program.addHelpText('after', generateTemplatesHelpText());
|
|
2627
|
+
|
|
2628
|
+
program.parse(process.argv);
|
|
2629
|
+
};
|
|
2630
|
+
|
|
2631
|
+
main();
|