@coze-arch/cli 0.0.1-alpha.de6c14 → 0.0.1-alpha.deaedf

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (235) hide show
  1. package/README.md +1 -0
  2. package/lib/__templates__/expo/.coze +1 -1
  3. package/lib/__templates__/expo/.cozeproj/scripts/dev_build.sh +19 -82
  4. package/lib/__templates__/expo/.cozeproj/scripts/dev_run.sh +82 -88
  5. package/lib/__templates__/expo/.cozeproj/scripts/prod_build.sh +2 -2
  6. package/lib/__templates__/expo/.cozeproj/scripts/prod_run.sh +3 -4
  7. package/lib/__templates__/expo/.cozeproj/scripts/server_dev_run.sh +46 -0
  8. package/lib/__templates__/expo/README.md +68 -7
  9. package/lib/__templates__/expo/_gitignore +1 -1
  10. package/lib/__templates__/expo/_npmrc +2 -4
  11. package/lib/__templates__/expo/client/app/+not-found.tsx +15 -64
  12. package/lib/__templates__/expo/client/app/_layout.tsx +15 -12
  13. package/lib/__templates__/expo/client/app/index.tsx +1 -0
  14. package/lib/__templates__/expo/client/app.config.ts +76 -0
  15. package/lib/__templates__/expo/client/components/Screen.tsx +3 -19
  16. package/lib/__templates__/expo/client/components/ThemedText.tsx +33 -0
  17. package/lib/__templates__/expo/client/components/ThemedView.tsx +37 -0
  18. package/lib/__templates__/expo/client/constants/theme.ts +116 -67
  19. package/lib/__templates__/expo/client/declarations.d.ts +5 -0
  20. package/lib/__templates__/expo/{eslint.config.mjs → client/eslint.config.mjs} +40 -10
  21. package/lib/__templates__/expo/client/hooks/useColorScheme.tsx +48 -0
  22. package/lib/__templates__/expo/client/hooks/useSafeRouter.ts +152 -0
  23. package/lib/__templates__/expo/client/hooks/useTheme.ts +26 -6
  24. package/lib/__templates__/expo/client/metro.config.js +124 -0
  25. package/lib/__templates__/expo/client/package.json +95 -0
  26. package/lib/__templates__/expo/client/screens/demo/index.tsx +25 -0
  27. package/lib/__templates__/expo/client/screens/demo/styles.ts +28 -0
  28. package/lib/__templates__/expo/client/scripts/install-missing-deps.js +11 -10
  29. package/lib/__templates__/expo/client/tsconfig.json +24 -0
  30. package/lib/__templates__/expo/client/utils/index.ts +22 -0
  31. package/lib/__templates__/expo/eslint-plugins/fontawesome6/index.js +9 -0
  32. package/lib/__templates__/expo/eslint-plugins/fontawesome6/names.js +1889 -0
  33. package/lib/__templates__/expo/eslint-plugins/fontawesome6/rule.js +174 -0
  34. package/lib/__templates__/expo/eslint-plugins/fontawesome6/v5-only-names.js +388 -0
  35. package/lib/__templates__/expo/eslint-plugins/forbid-emoji/index.js +9 -0
  36. package/lib/__templates__/expo/eslint-plugins/forbid-emoji/rule.js +112 -0
  37. package/lib/__templates__/expo/eslint-plugins/forbid-emoji/tech.md +94 -0
  38. package/lib/__templates__/expo/eslint-plugins/react-native/index.js +9 -0
  39. package/lib/__templates__/expo/eslint-plugins/react-native/rule.js +64 -0
  40. package/lib/__templates__/expo/eslint-plugins/reanimated/index.js +9 -0
  41. package/lib/__templates__/expo/eslint-plugins/reanimated/rule.js +88 -0
  42. package/lib/__templates__/expo/eslint-plugins/restrict-linear-gradient/index.js +9 -0
  43. package/lib/__templates__/expo/eslint-plugins/restrict-linear-gradient/rule.js +120 -0
  44. package/lib/__templates__/expo/eslint-plugins/restrict-linear-gradient/tech.md +58 -0
  45. package/lib/__templates__/expo/package.json +11 -107
  46. package/lib/__templates__/expo/patches/expo@54.0.33.patch +45 -0
  47. package/lib/__templates__/expo/pnpm-lock.yaml +1437 -3171
  48. package/lib/__templates__/expo/pnpm-workspace.yaml +3 -0
  49. package/lib/__templates__/expo/server/build.js +21 -0
  50. package/lib/__templates__/expo/server/package.json +34 -0
  51. package/lib/__templates__/expo/server/src/index.ts +20 -0
  52. package/lib/__templates__/expo/server/tsconfig.json +24 -0
  53. package/lib/__templates__/expo/template.config.js +57 -0
  54. package/lib/__templates__/expo/tsconfig.json +1 -24
  55. package/lib/__templates__/native-static/.coze +11 -0
  56. package/lib/__templates__/native-static/index.html +33 -0
  57. package/lib/__templates__/native-static/styles/main.css +136 -0
  58. package/lib/__templates__/native-static/template.config.js +22 -0
  59. package/lib/__templates__/nextjs/.babelrc +15 -0
  60. package/lib/__templates__/nextjs/.coze +1 -0
  61. package/lib/__templates__/nextjs/README.md +5 -0
  62. package/lib/__templates__/nextjs/_npmrc +1 -0
  63. package/lib/__templates__/nextjs/eslint.config.mjs +5 -0
  64. package/lib/__templates__/nextjs/next.config.ts +11 -0
  65. package/lib/__templates__/nextjs/package.json +10 -2
  66. package/lib/__templates__/nextjs/pnpm-lock.yaml +3488 -1569
  67. package/lib/__templates__/nextjs/scripts/build.sh +4 -1
  68. package/lib/__templates__/nextjs/scripts/dev.sh +8 -2
  69. package/lib/__templates__/nextjs/scripts/prepare.sh +9 -0
  70. package/lib/__templates__/nextjs/scripts/start.sh +7 -1
  71. package/lib/__templates__/nextjs/src/app/globals.css +10 -2
  72. package/lib/__templates__/nextjs/src/app/layout.tsx +5 -14
  73. package/lib/__templates__/nextjs/src/app/page.tsx +18 -49
  74. package/lib/__templates__/nextjs/src/components/ui/resizable.tsx +29 -22
  75. package/lib/__templates__/nextjs/src/components/ui/sidebar.tsx +228 -230
  76. package/lib/__templates__/nextjs/src/server.ts +35 -0
  77. package/lib/__templates__/nextjs/template.config.js +67 -2
  78. package/lib/__templates__/nextjs/tsconfig.json +1 -1
  79. package/lib/__templates__/nuxt-vue/.coze +12 -0
  80. package/lib/__templates__/nuxt-vue/README.md +73 -0
  81. package/lib/__templates__/nuxt-vue/_gitignore +24 -0
  82. package/lib/__templates__/nuxt-vue/_npmrc +23 -0
  83. package/lib/__templates__/nuxt-vue/app/app.vue +6 -0
  84. package/lib/__templates__/nuxt-vue/app/pages/index.vue +23 -0
  85. package/lib/__templates__/nuxt-vue/assets/css/main.css +24 -0
  86. package/lib/__templates__/nuxt-vue/nuxt.config.ts +116 -0
  87. package/lib/__templates__/nuxt-vue/package.json +35 -0
  88. package/lib/__templates__/nuxt-vue/pnpm-lock.yaml +8759 -0
  89. package/lib/__templates__/nuxt-vue/postcss.config.mjs +8 -0
  90. package/lib/__templates__/nuxt-vue/public/favicon.ico +0 -0
  91. package/lib/__templates__/nuxt-vue/public/robots.txt +2 -0
  92. package/lib/__templates__/nuxt-vue/scripts/build.sh +14 -0
  93. package/lib/__templates__/nuxt-vue/scripts/dev.sh +39 -0
  94. package/lib/__templates__/nuxt-vue/scripts/prepare.sh +14 -0
  95. package/lib/__templates__/nuxt-vue/scripts/start.sh +21 -0
  96. package/lib/__templates__/nuxt-vue/server/api/hello.ts +10 -0
  97. package/lib/__templates__/nuxt-vue/server/middleware/logger.ts +10 -0
  98. package/lib/__templates__/nuxt-vue/server/routes/health.ts +10 -0
  99. package/lib/__templates__/nuxt-vue/tailwind.config.js +13 -0
  100. package/lib/__templates__/nuxt-vue/template.config.js +87 -0
  101. package/lib/__templates__/nuxt-vue/tsconfig.json +18 -0
  102. package/lib/__templates__/taro/.coze +14 -0
  103. package/lib/__templates__/taro/.cozeproj/scripts/deploy_build.sh +19 -0
  104. package/lib/__templates__/taro/.cozeproj/scripts/deploy_run.sh +14 -0
  105. package/lib/__templates__/taro/.cozeproj/scripts/dev_build.sh +2 -0
  106. package/lib/__templates__/taro/.cozeproj/scripts/dev_run.sh +151 -0
  107. package/lib/__templates__/taro/.cozeproj/scripts/init_env.sh +5 -0
  108. package/lib/__templates__/taro/.cozeproj/scripts/pack.sh +24 -0
  109. package/lib/__templates__/taro/README.md +763 -0
  110. package/lib/__templates__/taro/_gitignore +40 -0
  111. package/lib/__templates__/taro/_npmrc +18 -0
  112. package/lib/__templates__/taro/babel.config.js +12 -0
  113. package/lib/__templates__/taro/config/dev.ts +9 -0
  114. package/lib/__templates__/taro/config/index.ts +237 -0
  115. package/lib/__templates__/taro/config/prod.ts +34 -0
  116. package/lib/__templates__/taro/eslint.config.mjs +135 -0
  117. package/lib/__templates__/taro/key/private.appid.key +0 -0
  118. package/lib/__templates__/taro/package.json +112 -0
  119. package/lib/__templates__/taro/patches/@tarojs__plugin-mini-ci@4.1.9.patch +30 -0
  120. package/lib/__templates__/taro/pnpm-lock.yaml +23412 -0
  121. package/lib/__templates__/taro/pnpm-workspace.yaml +2 -0
  122. package/lib/__templates__/taro/project.config.json +15 -0
  123. package/lib/__templates__/taro/server/nest-cli.json +10 -0
  124. package/lib/__templates__/taro/server/package.json +40 -0
  125. package/lib/__templates__/taro/server/src/app.controller.ts +23 -0
  126. package/lib/__templates__/taro/server/src/app.module.ts +10 -0
  127. package/lib/__templates__/taro/server/src/app.service.ts +8 -0
  128. package/lib/__templates__/taro/server/src/interceptors/http-status.interceptor.ts +23 -0
  129. package/lib/__templates__/taro/server/src/main.ts +49 -0
  130. package/lib/__templates__/taro/server/tsconfig.json +24 -0
  131. package/lib/__templates__/taro/src/app.config.ts +11 -0
  132. package/lib/__templates__/taro/src/app.css +156 -0
  133. package/lib/__templates__/taro/src/app.tsx +9 -0
  134. package/lib/__templates__/taro/src/components/ui/accordion.tsx +159 -0
  135. package/lib/__templates__/taro/src/components/ui/alert-dialog.tsx +260 -0
  136. package/lib/__templates__/taro/src/components/ui/alert.tsx +60 -0
  137. package/lib/__templates__/taro/src/components/ui/aspect-ratio.tsx +36 -0
  138. package/lib/__templates__/taro/src/components/ui/avatar.tsx +84 -0
  139. package/lib/__templates__/taro/src/components/ui/badge.tsx +37 -0
  140. package/lib/__templates__/taro/src/components/ui/breadcrumb.tsx +117 -0
  141. package/lib/__templates__/taro/src/components/ui/button-group.tsx +83 -0
  142. package/lib/__templates__/taro/src/components/ui/button.tsx +67 -0
  143. package/lib/__templates__/taro/src/components/ui/calendar.tsx +394 -0
  144. package/lib/__templates__/taro/src/components/ui/card.tsx +108 -0
  145. package/lib/__templates__/taro/src/components/ui/carousel.tsx +228 -0
  146. package/lib/__templates__/taro/src/components/ui/checkbox.tsx +58 -0
  147. package/lib/__templates__/taro/src/components/ui/code-block.tsx +169 -0
  148. package/lib/__templates__/taro/src/components/ui/collapsible.tsx +71 -0
  149. package/lib/__templates__/taro/src/components/ui/command.tsx +385 -0
  150. package/lib/__templates__/taro/src/components/ui/context-menu.tsx +614 -0
  151. package/lib/__templates__/taro/src/components/ui/dialog.tsx +256 -0
  152. package/lib/__templates__/taro/src/components/ui/drawer.tsx +192 -0
  153. package/lib/__templates__/taro/src/components/ui/dropdown-menu.tsx +561 -0
  154. package/lib/__templates__/taro/src/components/ui/field.tsx +228 -0
  155. package/lib/__templates__/taro/src/components/ui/hover-card.tsx +282 -0
  156. package/lib/__templates__/taro/src/components/ui/input-group.tsx +197 -0
  157. package/lib/__templates__/taro/src/components/ui/input-otp.tsx +136 -0
  158. package/lib/__templates__/taro/src/components/ui/input.tsx +56 -0
  159. package/lib/__templates__/taro/src/components/ui/label.tsx +24 -0
  160. package/lib/__templates__/taro/src/components/ui/menubar.tsx +595 -0
  161. package/lib/__templates__/taro/src/components/ui/navigation-menu.tsx +264 -0
  162. package/lib/__templates__/taro/src/components/ui/pagination.tsx +118 -0
  163. package/lib/__templates__/taro/src/components/ui/popover.tsx +291 -0
  164. package/lib/__templates__/taro/src/components/ui/portal.tsx +19 -0
  165. package/lib/__templates__/taro/src/components/ui/progress.tsx +28 -0
  166. package/lib/__templates__/taro/src/components/ui/radio-group.tsx +64 -0
  167. package/lib/__templates__/taro/src/components/ui/resizable.tsx +346 -0
  168. package/lib/__templates__/taro/src/components/ui/scroll-area.tsx +34 -0
  169. package/lib/__templates__/taro/src/components/ui/select.tsx +438 -0
  170. package/lib/__templates__/taro/src/components/ui/separator.tsx +30 -0
  171. package/lib/__templates__/taro/src/components/ui/sheet.tsx +262 -0
  172. package/lib/__templates__/taro/src/components/ui/skeleton.tsx +17 -0
  173. package/lib/__templates__/taro/src/components/ui/slider.tsx +203 -0
  174. package/lib/__templates__/taro/src/components/ui/sonner.tsx +1 -0
  175. package/lib/__templates__/taro/src/components/ui/switch.tsx +55 -0
  176. package/lib/__templates__/taro/src/components/ui/table.tsx +142 -0
  177. package/lib/__templates__/taro/src/components/ui/tabs.tsx +114 -0
  178. package/lib/__templates__/taro/src/components/ui/textarea.tsx +54 -0
  179. package/lib/__templates__/taro/src/components/ui/toast.tsx +517 -0
  180. package/lib/__templates__/taro/src/components/ui/toggle-group.tsx +120 -0
  181. package/lib/__templates__/taro/src/components/ui/toggle.tsx +77 -0
  182. package/lib/__templates__/taro/src/components/ui/tooltip.tsx +455 -0
  183. package/lib/__templates__/taro/src/index.html +39 -0
  184. package/lib/__templates__/taro/src/lib/hooks/use-keyboard-offset.ts +37 -0
  185. package/lib/__templates__/taro/src/lib/measure.ts +115 -0
  186. package/lib/__templates__/taro/src/lib/platform.ts +12 -0
  187. package/lib/__templates__/taro/src/lib/utils.ts +6 -0
  188. package/lib/__templates__/taro/src/network.ts +39 -0
  189. package/lib/__templates__/taro/src/pages/index/index.config.ts +3 -0
  190. package/lib/__templates__/taro/src/pages/index/index.css +1 -0
  191. package/lib/__templates__/taro/src/pages/index/index.tsx +33 -0
  192. package/lib/__templates__/taro/src/presets/dev-debug.ts +23 -0
  193. package/lib/__templates__/taro/src/presets/h5-container.tsx +15 -0
  194. package/lib/__templates__/taro/src/presets/h5-navbar.tsx +238 -0
  195. package/lib/__templates__/taro/src/presets/h5-styles.ts +220 -0
  196. package/lib/__templates__/taro/src/presets/index.tsx +18 -0
  197. package/lib/__templates__/taro/stylelint.config.mjs +4 -0
  198. package/lib/__templates__/taro/template.config.js +68 -0
  199. package/lib/__templates__/taro/tsconfig.json +29 -0
  200. package/lib/__templates__/taro/types/global.d.ts +32 -0
  201. package/lib/__templates__/templates.json +136 -43
  202. package/lib/__templates__/vite/.coze +1 -0
  203. package/lib/__templates__/vite/README.md +190 -11
  204. package/lib/__templates__/vite/_gitignore +1 -0
  205. package/lib/__templates__/vite/_npmrc +1 -0
  206. package/lib/__templates__/vite/eslint.config.mjs +14 -0
  207. package/lib/__templates__/vite/package.json +22 -2
  208. package/lib/__templates__/vite/pnpm-lock.yaml +2398 -182
  209. package/lib/__templates__/vite/scripts/build.sh +4 -1
  210. package/lib/__templates__/vite/scripts/dev.sh +9 -2
  211. package/lib/__templates__/vite/scripts/prepare.sh +9 -0
  212. package/lib/__templates__/vite/scripts/start.sh +9 -3
  213. package/lib/__templates__/vite/server/routes/index.ts +31 -0
  214. package/lib/__templates__/vite/server/server.ts +65 -0
  215. package/lib/__templates__/vite/server/vite.ts +67 -0
  216. package/lib/__templates__/vite/src/main.ts +17 -48
  217. package/lib/__templates__/vite/template.config.js +67 -6
  218. package/lib/__templates__/vite/tsconfig.json +4 -3
  219. package/lib/__templates__/vite/vite.config.ts +5 -0
  220. package/lib/cli.js +1084 -225
  221. package/package.json +12 -4
  222. package/lib/__templates__/expo/app.json +0 -63
  223. package/lib/__templates__/expo/babel.config.js +0 -19
  224. package/lib/__templates__/expo/client/app/(tabs)/_layout.tsx +0 -43
  225. package/lib/__templates__/expo/client/app/(tabs)/home.tsx +0 -1
  226. package/lib/__templates__/expo/client/app/(tabs)/index.tsx +0 -7
  227. package/lib/__templates__/expo/client/hooks/useColorScheme.ts +0 -1
  228. package/lib/__templates__/expo/client/index.js +0 -12
  229. package/lib/__templates__/expo/client/screens/home/index.tsx +0 -51
  230. package/lib/__templates__/expo/client/screens/home/styles.ts +0 -60
  231. package/lib/__templates__/expo/metro.config.js +0 -51
  232. package/lib/__templates__/expo/src/index.ts +0 -12
  233. package/lib/__templates__/nextjs/.vscode/settings.json +0 -121
  234. package/lib/__templates__/vite/.vscode/settings.json +0 -7
  235. /package/lib/__templates__/expo/{eslint-formatter-simple.mjs → client/eslint-formatter-simple.mjs} +0 -0
package/lib/cli.js CHANGED
@@ -4,13 +4,15 @@
4
4
  var commander = require('commander');
5
5
  var path = require('path');
6
6
  var fs = require('fs');
7
+ var node_path = require('node:path');
8
+ var node_fs = require('node:fs');
7
9
  var shelljs = require('shelljs');
8
10
  var perf_hooks = require('perf_hooks');
9
11
  var fs$1 = require('fs/promises');
10
- var toml = require('@iarna/toml');
12
+ var os = require('os');
11
13
  var jsYaml = require('js-yaml');
14
+ var toml = require('@iarna/toml');
12
15
  var child_process = require('child_process');
13
- var os = require('os');
14
16
  var addFormats = require('ajv-formats');
15
17
  var Ajv = require('ajv');
16
18
  var minimist = require('minimist');
@@ -481,6 +483,14 @@ const warmupTemplate = (templatePath, templateName) => {
481
483
  logger.info(`\nWarming up template: ${templateName}`);
482
484
  logger.info(` Path: ${templatePath}`);
483
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
+
484
494
  const result = shelljs.exec('pnpm install', {
485
495
  cwd: templatePath,
486
496
  silent: true,
@@ -514,13 +524,7 @@ const warmupTemplate = (templatePath, templateName) => {
514
524
  /**
515
525
  * 执行 warmup 命令的内部实现
516
526
  */
517
- const executeWarmup = async (
518
- options
519
-
520
- ,
521
-
522
- command,
523
- ) => {
527
+ const executeWarmup = async (options) => {
524
528
  const timer = new TimeTracker();
525
529
 
526
530
  try {
@@ -585,12 +589,12 @@ const executeWarmup = async (
585
589
  /**
586
590
  * 注册 warmup 命令到 program
587
591
  */
588
- const registerCommand$2 = program => {
592
+ const registerCommand$4 = program => {
589
593
  program
590
594
  .command('warmup')
591
595
  .description('Pre-install dependencies for templates to speed up init')
592
596
  .option('-t, --template <name>', 'Warmup a specific template only')
593
- .action(async (options, command) => {
597
+ .action(async options => {
594
598
  await executeWarmup(options);
595
599
  });
596
600
  };
@@ -727,7 +731,7 @@ const parseConfigContent = (content) => {
727
731
  return config ;
728
732
  } catch (error) {
729
733
  // TOML 解析失败,继续尝试其他格式
730
- // eslint-disable-next-line no-console
734
+
731
735
  console.debug('TOML parse failed:', error);
732
736
  }
733
737
 
@@ -739,7 +743,7 @@ const parseConfigContent = (content) => {
739
743
  }
740
744
  } catch (error) {
741
745
  // YAML 解析失败,继续尝试其他格式
742
- // eslint-disable-next-line no-console
746
+
743
747
  console.debug('YAML parse failed:', error);
744
748
  }
745
749
 
@@ -823,27 +827,276 @@ const getCommandConfig = (
823
827
  return commandConfig;
824
828
  };
825
829
 
826
- 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; }
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
+ };
827
859
 
828
860
  /**
829
- * 创建日志管理器
861
+ * 查找 Next.js 配置文件
830
862
  */
831
- const createLogManager = (logDir = '.coze-logs') => {
832
- const ensureLogDir = () => {
833
- if (!fs.existsSync(logDir)) {
834
- fs.mkdirSync(logDir, { recursive: true });
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;
835
874
  }
836
- };
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);
837
930
 
838
- const getLogPath = (logFile) => path.join(logDir, logFile);
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
+ }
839
954
 
840
955
  return {
841
- createWriteStream: (logFile) => {
842
- ensureLogDir();
843
- return fs.createWriteStream(getLogPath(logFile), { flags: 'a' });
844
- },
956
+ ruleName,
957
+ applied: false,
958
+ message: 'No outputFileTracingRoot config found, skipping',
845
959
  };
846
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
+ };
847
1100
 
848
1101
  /**
849
1102
  * 执行命令的内部实现
@@ -855,21 +1108,32 @@ const executeRun = async (
855
1108
  try {
856
1109
  logger.info(`Running ${commandName} command...`);
857
1110
 
858
- // 1. 加载 .coze 配置
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 配置
859
1124
  const config = await loadCozeConfig();
860
1125
  const commandArgs = getCommandConfig(config, commandName);
861
1126
 
862
- // 2. 准备日志
863
- const logManager = createLogManager();
864
- const logFile = options.logFile || `${commandName}.log`;
865
- const logStream = logManager.createWriteStream(logFile);
1127
+ // 3. 准备日志
1128
+ const logFilePath = resolveLogFilePath$1(options.logFile);
1129
+ const logStream = createLogStream$1(logFilePath);
866
1130
 
867
- // 3. 执行命令
1131
+ // 4. 执行命令
868
1132
  const commandString = commandArgs.join(' ');
869
1133
 
870
1134
  logger.info(`Executing: ${commandString}`);
871
1135
  logger.info(`Working directory: ${process.cwd()}`);
872
- logger.info(`Log file: ${logFile}`);
1136
+ logger.info(`Log file: ${logFilePath}`);
873
1137
 
874
1138
  const childProcess = shelljs.exec(commandString, {
875
1139
  async: true,
@@ -898,11 +1162,11 @@ const executeRun = async (
898
1162
  logger.error(
899
1163
  `Command exited with code ${_nullishCoalesce$1(code, () => ( 'unknown'))}${signal ? ` and signal ${signal}` : ''}`,
900
1164
  );
901
- logger.error(`Check log file for details: ${logFile}`);
1165
+ logger.error(`Check log file for details: ${logFilePath}`);
902
1166
  process.exit(code || 1);
903
1167
  } else {
904
1168
  logger.success('Command completed successfully');
905
- logger.info(`Log file: ${logFile}`);
1169
+ logger.info(`Log file: ${logFilePath}`);
906
1170
  }
907
1171
  });
908
1172
 
@@ -925,7 +1189,7 @@ const executeRun = async (
925
1189
  /**
926
1190
  * 注册 dev/build/start 命令到 program
927
1191
  */
928
- const registerCommand$1 = program => {
1192
+ const registerCommand$2 = program => {
929
1193
  // dev 命令
930
1194
  program
931
1195
  .command('dev')
@@ -1177,6 +1441,11 @@ const shouldIgnoreFile = (filePath) => {
1177
1441
  return directoryPatterns.some(dir => pathParts.includes(dir));
1178
1442
  };
1179
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
1180
1449
  /**
1181
1450
  * 递归获取目录中的所有文件
1182
1451
  *
@@ -1246,173 +1515,376 @@ const convertDotfileName = (filePath) => {
1246
1515
 
1247
1516
  return filePath;
1248
1517
  };
1518
+ // end_aigc
1519
+
1520
+ // ABOUTME: File rendering utilities for template processing
1521
+ // ABOUTME: Handles file content rendering, hook execution, and file writing
1249
1522
 
1523
+
1524
+
1525
+
1526
+
1527
+
1528
+
1529
+
1530
+ // start_aigc
1250
1531
  /**
1251
- * 复制并处理模板文件到目标目录
1532
+ * 执行文件渲染钩子
1252
1533
  *
1253
- * @param templatePath - 模板目录路径
1254
- * @param outputPath - 输出目录路径
1534
+ * @param templateConfig - 模板配置
1535
+ * @param fileInfo - 文件渲染信息
1255
1536
  * @param context - 模板上下文
1537
+ * @returns 处理后的文件信息,或 null 表示跳过该文件
1256
1538
  */
1257
- const processTemplateFiles = async (
1258
- templatePath,
1259
- outputPath,
1539
+ const executeFileRenderHook = async (
1540
+ templateConfig,
1541
+ fileInfo,
1260
1542
  context,
1261
1543
  ) => {
1262
- logger.verbose('Processing template files:');
1263
- logger.verbose(` - Template path: ${templatePath}`);
1264
- logger.verbose(` - Output path: ${outputPath}`);
1265
-
1266
- // 验证模板目录是否存在
1267
- try {
1268
- const stat = await fs$1.stat(templatePath);
1269
- logger.verbose(
1270
- ` - Template path exists: ${stat.isDirectory() ? 'directory' : 'file'}`,
1271
- );
1272
- if (!stat.isDirectory()) {
1273
- throw new Error(`Template path is not a directory: ${templatePath}`);
1274
- }
1275
- } catch (error) {
1276
- logger.error(
1277
- ` - Failed to access template path: ${error instanceof Error ? error.message : String(error)}`,
1278
- );
1279
- throw error;
1544
+ if (!templateConfig.onFileRender) {
1545
+ return fileInfo;
1280
1546
  }
1281
1547
 
1282
- const files = await getAllFiles(templatePath);
1548
+ const result = await templateConfig.onFileRender(
1549
+ fileInfo,
1550
+ context,
1551
+ );
1283
1552
 
1284
- logger.verbose(` - Found ${files.length} files to process`);
1553
+ // false: 跳过文件
1554
+ if (result === false) {
1555
+ return null;
1556
+ }
1285
1557
 
1286
- if (files.length === 0) {
1287
- logger.warn(' - No files found in template directory!');
1288
- return;
1558
+ // undefined/void: 使用默认内容
1559
+ if (result === undefined || result === null) {
1560
+ return fileInfo;
1289
1561
  }
1290
1562
 
1291
- await Promise.all(
1292
- files.map(async file => {
1293
- const srcPath = path.join(templatePath, file);
1294
- const destFile = convertDotfileName(file);
1295
- const destPath = path.join(outputPath, destFile);
1563
+ // string: 作为 content,其他不变
1564
+ if (typeof result === 'string') {
1565
+ return {
1566
+ ...fileInfo,
1567
+ content: result,
1568
+ };
1569
+ }
1296
1570
 
1297
- logger.verbose(
1298
- ` - Processing: ${file}${destFile !== file ? ` -> ${destFile}` : ''}`,
1299
- );
1571
+ // FileRenderInfo: 使用新对象的信息
1572
+ return result;
1573
+ };
1300
1574
 
1301
- // 确保目标目录存在
1302
- await ensureDir(path.dirname(destPath));
1575
+ /**
1576
+ * 准备单个文件的渲染信息(不实际写入)
1577
+ * 用于 dry-run 阶段,收集所有将要写入的文件信息
1578
+ *
1579
+ * @param options - 准备选项
1580
+ * @returns 文件渲染信息,或 null 表示该文件被跳过
1581
+ */
1582
+ const prepareFileInfo = async (options
1303
1583
 
1304
- if (shouldRenderFile(srcPath)) {
1305
- // 渲染文本文件
1306
- const rendered = await renderTemplate(srcPath, context);
1307
- await fs$1.writeFile(destPath, rendered, 'utf-8');
1308
- logger.verbose(' ✓ Rendered and written');
1309
- } else {
1310
- // 直接复制二进制文件
1311
- await fs$1.copyFile(srcPath, destPath);
1312
- logger.verbose(' ✓ Copied');
1313
- }
1314
- }),
1315
- );
1316
1584
 
1317
- logger.verbose('✓ All files processed successfully');
1318
1585
 
1319
- // 单独处理 node_modules 目录(如果存在)
1320
- const sourceNodeModules = path.join(templatePath, 'node_modules');
1321
- const targetNodeModules = path.join(outputPath, 'node_modules');
1322
1586
 
1323
- if (fs.existsSync(sourceNodeModules)) {
1324
- logger.info('\nCopying node_modules from pre-warmed template...');
1325
- logger.verbose(` From: ${sourceNodeModules}`);
1326
- logger.verbose(` To: ${targetNodeModules}`);
1587
+ ) => {
1588
+ const { file, templatePath, context, templateConfig } = options;
1327
1589
 
1328
- const result = shelljs.exec(`cp -R "${sourceNodeModules}" "${targetNodeModules}"`, {
1329
- silent: true,
1330
- });
1590
+ const srcPath = path.join(templatePath, file);
1591
+ const destFile = convertDotfileName(file);
1331
1592
 
1332
- if (result.stdout) {
1333
- process.stdout.write(result.stdout);
1334
- }
1593
+ logger.verbose(
1594
+ ` - Preparing: ${file}${destFile !== file ? ` -> ${destFile}` : ''}`,
1595
+ );
1335
1596
 
1336
- if (result.stderr) {
1337
- process.stderr.write(result.stderr);
1338
- }
1597
+ // 判断是否为二进制文件
1598
+ const isBinary = !shouldRenderFile(srcPath);
1599
+ let content;
1600
+ let wasRendered = false;
1339
1601
 
1340
- if (result.code === 0) {
1341
- logger.success('✓ node_modules copied successfully');
1342
- } else {
1343
- logger.warn(
1344
- `Failed to copy node_modules: ${result.stderr || 'unknown error'}`,
1345
- );
1346
- logger.info('Will need to run pnpm install manually');
1347
- }
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;
1348
1610
  }
1349
- };
1350
1611
 
1351
- /**
1352
- * 检查输出目录是否为空
1353
- * 注意:.git 目录会被忽略,允许在已初始化 git 的目录中创建项目
1354
- *
1355
- * @param outputPath - 输出目录路径
1356
- * @returns 是否为空
1357
- */
1358
- const isOutputDirEmpty = async (
1359
- outputPath,
1360
- ) => {
1361
- try {
1362
- const entries = await fs$1.readdir(outputPath);
1363
- // 过滤掉 .git 目录,允许在已初始化 git 的目录中创建项目
1364
- const filteredEntries = entries.filter(entry => entry !== '.git');
1365
- return filteredEntries.length === 0;
1366
- } catch (e) {
1367
- // 目录不存在,视为空
1368
- return true;
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;
1369
1632
  }
1633
+
1634
+ return processedFileInfo;
1370
1635
  };
1371
1636
 
1372
1637
  /**
1373
- * 验证输出目录
1638
+ * 写入渲染后的文件到目标路径
1374
1639
  *
1375
- * @param outputPath - 输出目录路径
1376
- * @throws 如果目录不为空则抛出错误
1640
+ * @param options - 写入选项
1377
1641
  */
1378
- const validateOutputDir = async (outputPath) => {
1379
- const isEmpty = await isOutputDirEmpty(outputPath);
1380
- if (!isEmpty) {
1381
- throw new Error(
1382
- `Output directory is not empty: ${outputPath}\n` +
1383
- 'Please use an empty directory or remove existing files.',
1384
- );
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');
1385
1672
  }
1386
1673
  };
1674
+ // end_aigc
1387
1675
 
1388
- /**
1389
- * 模板引擎执行选项
1390
- */
1676
+ // ABOUTME: File conflict detection utilities for template processing
1677
+ // ABOUTME: Provides dry-run file collection and conflict checking
1391
1678
 
1392
1679
 
1393
1680
 
1394
1681
 
1395
1682
 
1396
1683
 
1684
+
1685
+ // start_aigc
1397
1686
  /**
1398
- * 加载模板元数据和路径
1687
+ * 收集所有将要写入的文件路径(dry-run 阶段)
1688
+ *
1689
+ * @param options - 收集选项
1690
+ * @returns 将要写入的文件路径列表
1399
1691
  */
1400
- const loadTemplateMetadata = async (templateName) => {
1401
- const templatesConfigPath = getTemplatesConfigPath();
1402
- const templatesConfig = await loadTemplatesConfig(templatesConfigPath);
1403
- const templateMetadata = findTemplate(templatesConfig, templateName);
1404
- const templatesDir = getTemplatesDir();
1405
- const templatePath = await getTemplatePath(templatesDir, templateMetadata);
1692
+ const collectFilesToRender = async (options
1406
1693
 
1407
- return { templatePath, templateMetadata };
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;
1408
1724
  };
1725
+ // end_aigc
1726
+
1727
+ // ABOUTME: Main file processing orchestration for template rendering
1728
+ // ABOUTME: Coordinates file system, rendering, and conflict detection layers
1409
1729
 
1730
+
1731
+
1732
+ // start_aigc
1410
1733
  /**
1411
- * 解析并验证模板参数
1734
+ * 处理单个文件(准备 + 写入)
1735
+ *
1736
+ * @param options - 处理选项
1412
1737
  */
1413
- const parseAndValidateParams = (
1414
- command,
1415
- templateConfig,
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,
1416
1888
  ) => {
1417
1889
  const knownOptions = new Set([
1418
1890
  'template',
@@ -1463,22 +1935,47 @@ const executeAfterRenderHook = async (
1463
1935
  }
1464
1936
  };
1465
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
+
1466
1951
  /**
1467
1952
  * 准备输出目录
1468
1953
  */
1469
- const prepareOutputDirectory = async (outputPath) => {
1954
+ const prepareOutputDirectory = (outputPath) => {
1470
1955
  const absolutePath = path.resolve(process.cwd(), outputPath);
1471
- await validateOutputDir(absolutePath);
1956
+ // 不再在这里验证目录是否为空,冲突检测已移至 processTemplateFiles 中
1472
1957
  return absolutePath;
1473
1958
  };
1474
1959
 
1960
+ /**
1961
+ * 模板引擎执行结果
1962
+ */
1963
+
1964
+
1965
+
1966
+
1967
+
1968
+
1969
+
1970
+
1971
+
1475
1972
  /**
1476
1973
  * 执行完整的模板渲染流程
1477
1974
  */
1478
1975
  const execute = async (
1479
1976
  options,
1480
1977
  ) => {
1481
- const { templateName, outputPath, command } = options;
1978
+ const { templateName, outputPath, command} = options;
1482
1979
 
1483
1980
  // 1. 加载模板
1484
1981
  const { templatePath } = await loadTemplateMetadata(templateName);
@@ -1495,15 +1992,23 @@ const execute = async (
1495
1992
  });
1496
1993
 
1497
1994
  // 5. 准备输出目录
1498
- const absoluteOutputPath = await prepareOutputDirectory(outputPath);
1995
+ const absoluteOutputPath = prepareOutputDirectory(outputPath);
1499
1996
 
1500
1997
  // 6. 处理模板文件
1501
- await processTemplateFiles(templatePath, absoluteOutputPath, context);
1998
+ await processTemplateFiles({
1999
+ templatePath,
2000
+ outputPath: absoluteOutputPath,
2001
+ context,
2002
+ templateConfig});
1502
2003
 
1503
2004
  // 7. 执行 onAfterRender 钩子
1504
2005
  await executeAfterRenderHook(templateConfig, context, absoluteOutputPath);
1505
2006
 
1506
- return absoluteOutputPath;
2007
+ return {
2008
+ outputPath: absoluteOutputPath,
2009
+ templateConfig,
2010
+ context,
2011
+ };
1507
2012
  };
1508
2013
 
1509
2014
  function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } }
@@ -1545,7 +2050,40 @@ const runPnpmInstall = (projectPath) => {
1545
2050
  };
1546
2051
 
1547
2052
  /**
1548
- * 初始化 git 仓库并创建初始提交
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 仓库
1549
2087
  * 如果目录中已存在 .git,则跳过初始化
1550
2088
  */
1551
2089
  const runGitInit = (projectPath) => {
@@ -1558,46 +2096,9 @@ const runGitInit = (projectPath) => {
1558
2096
  return;
1559
2097
  }
1560
2098
 
1561
- const runGitCommand = (command) => {
1562
- logger.info(`Executing: ${command}`);
1563
-
1564
- const result = shelljs.exec(command, {
1565
- cwd: projectPath,
1566
- silent: true,
1567
- });
1568
-
1569
- // 输出命令的结果
1570
- if (result.stdout) {
1571
- process.stdout.write(result.stdout);
1572
- }
1573
-
1574
- if (result.stderr) {
1575
- process.stderr.write(result.stderr);
1576
- }
1577
-
1578
- if (result.code !== 0) {
1579
- const errorMessage = [
1580
- `${command} failed with exit code ${result.code}`,
1581
- result.stderr ? `\nStderr:\n${result.stderr}` : '',
1582
- result.stdout ? `\nStdout:\n${result.stdout}` : '',
1583
- ]
1584
- .filter(Boolean)
1585
- .join('');
1586
-
1587
- throw new Error(errorMessage);
1588
- }
1589
- };
1590
-
1591
2099
  try {
1592
2100
  logger.info('\nInitializing git repository...');
1593
- runGitCommand('git init');
1594
-
1595
- logger.info('Adding files to git...');
1596
- runGitCommand('git add .');
1597
-
1598
- logger.info('Creating initial commit...');
1599
- runGitCommand('git commit -m "chore: initial commit"');
1600
-
2101
+ runGitCommand('git init', projectPath);
1601
2102
  logger.success('Git repository initialized successfully!');
1602
2103
  } catch (error) {
1603
2104
  // Git 初始化失败不应该导致整个流程失败
@@ -1608,16 +2109,52 @@ const runGitInit = (projectPath) => {
1608
2109
  }
1609
2110
  };
1610
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
+
1611
2141
  /**
1612
2142
  * 运行开发服务器(后台模式)
1613
2143
  * 启动后台子进程运行开发服务器,父进程可以直接退出
2144
+ * 使用 CLI 自己的 dev 命令(定义在 run.ts)而不是直接运行 npm run dev
1614
2145
  */
1615
- const runNpmDev = (projectPath) => {
2146
+ const runDev = (projectPath) => {
1616
2147
  logger.info('\nStarting development server in background...');
1617
- logger.info(`Executing: npm run dev in ${projectPath}`);
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}`);
1618
2154
 
1619
2155
  // 使用通用的后台执行函数启动开发服务器
1620
- const pid = spawnDetached('npm', ['run', 'dev'], {
2156
+ // 调用 CLI 自己的 dev 命令
2157
+ const pid = spawnDetached(process.argv[0], [cliPath, 'dev'], {
1621
2158
  cwd: projectPath,
1622
2159
  verbose: false, // 不输出额外的进程信息,由 logger 统一处理
1623
2160
  });
@@ -1642,6 +2179,8 @@ const executeInit = async (
1642
2179
 
1643
2180
 
1644
2181
 
2182
+
2183
+
1645
2184
  ,
1646
2185
  command,
1647
2186
  ) => {
@@ -1653,61 +2192,76 @@ const executeInit = async (
1653
2192
  output: outputPath,
1654
2193
  skipInstall,
1655
2194
  skipGit,
2195
+ skipCommit,
1656
2196
  skipDev,
2197
+ force,
1657
2198
  } = options;
1658
2199
 
1659
2200
  logger.info(`Initializing project with template: ${templateName}`);
1660
2201
  timer.logPhase('Initialization');
1661
2202
 
1662
- // 执行模板引擎,返回绝对路径
1663
- const absoluteOutputPath = await execute({
2203
+ // 执行模板引擎,返回结果对象
2204
+ const result = await execute({
1664
2205
  templateName,
1665
2206
  outputPath,
1666
2207
  command,
2208
+ force,
1667
2209
  });
2210
+ const { outputPath: absoluteOutputPath, templateConfig, context } = result;
1668
2211
 
1669
2212
  timer.logPhase('Template engine execution');
1670
2213
  logger.success('Project created successfully!');
1671
2214
 
1672
- // 如果没有跳过安装,检查是否需要运行 pnpm install
1673
- if (!skipInstall) {
1674
- const nodeModulesPath = path.join(absoluteOutputPath, 'node_modules');
1675
- const hasNodeModules = fs.existsSync(nodeModulesPath);
2215
+ // 检查是否存在 package.json
2216
+ const packageJsonPath = path.join(absoluteOutputPath, 'package.json');
2217
+ const hasPackageJson = fs.existsSync(packageJsonPath);
1676
2218
 
1677
- if (hasNodeModules) {
1678
- logger.info(
1679
- '\n💡 Using pre-warmed node_modules, skipping pnpm install',
1680
- );
1681
- timer.logPhase('Node modules (pre-warmed)');
1682
- } else {
2219
+ // 安装依赖(始终使用 pnpm install,利用缓存机制)
2220
+ if (!skipInstall) {
2221
+ if (hasPackageJson) {
1683
2222
  runPnpmInstall(absoluteOutputPath);
1684
2223
  timer.logPhase('Dependencies installation');
2224
+ } else {
2225
+ logger.info(
2226
+ '\n💡 No package.json found, skipping dependency installation',
2227
+ );
1685
2228
  }
1686
2229
  }
1687
2230
 
2231
+ // 执行 onComplete 钩子(在 pnpm install 之后)
2232
+ await executeCompleteHook(templateConfig, context, absoluteOutputPath);
2233
+ timer.logPhase('Complete hook execution');
2234
+
1688
2235
  // 如果没有跳过 git,则初始化 git 仓库
1689
2236
  if (!skipGit) {
1690
2237
  runGitInit(absoluteOutputPath);
1691
2238
  timer.logPhase('Git initialization');
1692
2239
  }
1693
2240
 
2241
+ // 如果没有跳过 commit,则提交初始化生成的文件
2242
+ if (!skipCommit) {
2243
+ commitChanges(absoluteOutputPath);
2244
+ timer.logPhase('Git commit');
2245
+ }
2246
+
1694
2247
  // 如果没有跳过 dev,则启动开发服务器
1695
2248
  if (!skipDev) {
1696
- runNpmDev(absoluteOutputPath);
2249
+ runDev(absoluteOutputPath);
1697
2250
  timer.logPhase('Dev server startup');
1698
2251
  } else {
1699
2252
  // 只有跳过 dev 时才显示 Next steps
1700
2253
  logger.info('\nNext steps:');
1701
2254
  logger.info(` cd ${outputPath}`);
1702
- if (skipInstall) {
2255
+ if (skipInstall && hasPackageJson) {
1703
2256
  logger.info(' pnpm install');
1704
2257
  }
1705
2258
  if (skipGit) {
1706
- logger.info(
1707
- ' git init && git add . && git commit -m "initial commit"',
1708
- );
2259
+ logger.info(' git init');
1709
2260
  }
1710
- logger.info(' npm run dev');
2261
+ if (skipCommit) {
2262
+ logger.info(' git add --all && git commit -m "chore: init env"');
2263
+ }
2264
+ logger.info(' coze dev');
1711
2265
  }
1712
2266
 
1713
2267
  // 输出总耗时
@@ -1723,7 +2277,7 @@ const executeInit = async (
1723
2277
  /**
1724
2278
  * 注册 init 命令到 program
1725
2279
  */
1726
- const registerCommand = program => {
2280
+ const registerCommand$1 = program => {
1727
2281
  program
1728
2282
  .command('init')
1729
2283
  .description('Initialize a new project from a template')
@@ -1732,23 +2286,328 @@ const registerCommand = program => {
1732
2286
  .option('-o, --output <path>', 'Output directory', process.cwd())
1733
2287
  .option('--skip-install', 'Skip automatic pnpm install', false)
1734
2288
  .option('--skip-git', 'Skip automatic git initialization', false)
2289
+ .option(
2290
+ '--skip-commit',
2291
+ 'Skip automatic git commit after initialization',
2292
+ false,
2293
+ )
1735
2294
  .option('--skip-dev', 'Skip automatic dev server start', false)
1736
2295
  .allowUnknownOption() // 允许透传参数
1737
2296
  .action(async (directory, options, command) => {
1738
2297
  // 位置参数优先级高于 --output 选项
1739
2298
  const outputPath = _nullishCoalesce(directory, () => ( options.output));
1740
- await executeInit({ ...options, output: outputPath }, command);
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
+ }
1741
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
+ }
1742
2569
  };
2570
+ // end_aigc
1743
2571
 
1744
- var version = "0.0.1-alpha.de6c14";
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.deaedf";
1745
2602
  var packageJson = {
1746
2603
  version: version};
1747
2604
 
1748
2605
  const commands = [
1749
- registerCommand,
1750
2606
  registerCommand$1,
1751
2607
  registerCommand$2,
2608
+ registerCommand$4,
2609
+ registerCommand$3,
2610
+ registerCommand,
1752
2611
  ];
1753
2612
 
1754
2613
  const main = () => {