@coze-arch/cli 0.0.1-alpha.f74941 → 0.0.1-alpha.ff64d9

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 (167) hide show
  1. package/lib/__templates__/expo/.coze +7 -2
  2. package/lib/__templates__/expo/.cozeproj/scripts/dev_build.sh +46 -0
  3. package/lib/__templates__/expo/.cozeproj/scripts/dev_run.sh +220 -0
  4. package/lib/__templates__/expo/.cozeproj/scripts/prod_build.sh +47 -0
  5. package/lib/__templates__/expo/.cozeproj/scripts/prod_run.sh +34 -0
  6. package/lib/__templates__/expo/.cozeproj/scripts/server_dev_run.sh +45 -0
  7. package/lib/__templates__/expo/README.md +66 -7
  8. package/lib/__templates__/expo/_gitignore +1 -1
  9. package/lib/__templates__/expo/_npmrc +3 -5
  10. package/lib/__templates__/expo/client/app/_layout.tsx +14 -14
  11. package/lib/__templates__/expo/client/app/demo.tsx +1 -0
  12. package/lib/__templates__/expo/client/app/index.tsx +1 -0
  13. package/lib/__templates__/expo/client/app.config.ts +75 -0
  14. package/lib/__templates__/expo/client/components/ThemedText.tsx +33 -0
  15. package/lib/__templates__/expo/client/components/ThemedView.tsx +38 -0
  16. package/lib/__templates__/expo/client/constants/theme.ts +786 -50
  17. package/lib/__templates__/expo/client/contexts/AuthContext.tsx +14 -107
  18. package/lib/__templates__/expo/client/declarations.d.ts +5 -0
  19. package/lib/__templates__/expo/client/hooks/useColorScheme.ts +34 -1
  20. package/lib/__templates__/expo/client/hooks/useTheme.ts +1 -1
  21. package/lib/__templates__/expo/client/metro.config.js +121 -0
  22. package/lib/__templates__/expo/client/package.json +93 -0
  23. package/lib/__templates__/expo/client/screens/demo/index.tsx +25 -0
  24. package/lib/__templates__/expo/client/screens/demo/styles.ts +28 -0
  25. package/lib/__templates__/expo/client/scripts/install-missing-deps.js +36 -12
  26. package/lib/__templates__/expo/client/tsconfig.json +24 -0
  27. package/lib/__templates__/expo/client/utils/index.ts +1 -2
  28. package/lib/__templates__/expo/package.json +13 -92
  29. package/lib/__templates__/expo/pnpm-lock.yaml +675 -678
  30. package/lib/__templates__/expo/pnpm-workspace.yaml +3 -0
  31. package/lib/__templates__/expo/server/package.json +32 -0
  32. package/lib/__templates__/expo/{src → server/src}/index.ts +8 -2
  33. package/lib/__templates__/expo/server/tsconfig.json +24 -0
  34. package/lib/__templates__/expo/template.config.js +2 -1
  35. package/lib/__templates__/expo/tsconfig.json +1 -24
  36. package/lib/__templates__/nextjs/.coze +4 -3
  37. package/lib/__templates__/nextjs/README.md +341 -19
  38. package/lib/__templates__/nextjs/_npmrc +2 -1
  39. package/lib/__templates__/nextjs/components.json +21 -0
  40. package/lib/__templates__/nextjs/next.config.ts +12 -0
  41. package/lib/__templates__/nextjs/package.json +56 -2
  42. package/lib/__templates__/nextjs/pnpm-lock.yaml +7951 -1519
  43. package/lib/__templates__/nextjs/scripts/dev.sh +9 -27
  44. package/lib/__templates__/{react-rsbuild/scripts/build.sh → nextjs/scripts/prepare.sh} +0 -5
  45. package/lib/__templates__/nextjs/src/app/globals.css +124 -13
  46. package/lib/__templates__/nextjs/src/app/layout.tsx +19 -30
  47. package/lib/__templates__/nextjs/src/app/page.tsx +35 -23
  48. package/lib/__templates__/nextjs/src/components/ui/accordion.tsx +66 -0
  49. package/lib/__templates__/nextjs/src/components/ui/alert-dialog.tsx +157 -0
  50. package/lib/__templates__/nextjs/src/components/ui/alert.tsx +66 -0
  51. package/lib/__templates__/nextjs/src/components/ui/aspect-ratio.tsx +11 -0
  52. package/lib/__templates__/nextjs/src/components/ui/avatar.tsx +53 -0
  53. package/lib/__templates__/nextjs/src/components/ui/badge.tsx +46 -0
  54. package/lib/__templates__/nextjs/src/components/ui/breadcrumb.tsx +109 -0
  55. package/lib/__templates__/nextjs/src/components/ui/button-group.tsx +83 -0
  56. package/lib/__templates__/nextjs/src/components/ui/button.tsx +62 -0
  57. package/lib/__templates__/nextjs/src/components/ui/calendar.tsx +220 -0
  58. package/lib/__templates__/nextjs/src/components/ui/card.tsx +92 -0
  59. package/lib/__templates__/nextjs/src/components/ui/carousel.tsx +241 -0
  60. package/lib/__templates__/nextjs/src/components/ui/chart.tsx +357 -0
  61. package/lib/__templates__/nextjs/src/components/ui/checkbox.tsx +32 -0
  62. package/lib/__templates__/nextjs/src/components/ui/collapsible.tsx +33 -0
  63. package/lib/__templates__/nextjs/src/components/ui/command.tsx +184 -0
  64. package/lib/__templates__/nextjs/src/components/ui/context-menu.tsx +252 -0
  65. package/lib/__templates__/nextjs/src/components/ui/dialog.tsx +143 -0
  66. package/lib/__templates__/nextjs/src/components/ui/drawer.tsx +135 -0
  67. package/lib/__templates__/nextjs/src/components/ui/dropdown-menu.tsx +257 -0
  68. package/lib/__templates__/nextjs/src/components/ui/empty.tsx +104 -0
  69. package/lib/__templates__/nextjs/src/components/ui/field.tsx +248 -0
  70. package/lib/__templates__/nextjs/src/components/ui/form.tsx +167 -0
  71. package/lib/__templates__/nextjs/src/components/ui/hover-card.tsx +44 -0
  72. package/lib/__templates__/nextjs/src/components/ui/input-group.tsx +170 -0
  73. package/lib/__templates__/nextjs/src/components/ui/input-otp.tsx +77 -0
  74. package/lib/__templates__/nextjs/src/components/ui/input.tsx +21 -0
  75. package/lib/__templates__/nextjs/src/components/ui/item.tsx +193 -0
  76. package/lib/__templates__/nextjs/src/components/ui/kbd.tsx +28 -0
  77. package/lib/__templates__/nextjs/src/components/ui/label.tsx +24 -0
  78. package/lib/__templates__/nextjs/src/components/ui/menubar.tsx +276 -0
  79. package/lib/__templates__/nextjs/src/components/ui/navigation-menu.tsx +168 -0
  80. package/lib/__templates__/nextjs/src/components/ui/pagination.tsx +127 -0
  81. package/lib/__templates__/nextjs/src/components/ui/popover.tsx +48 -0
  82. package/lib/__templates__/nextjs/src/components/ui/progress.tsx +31 -0
  83. package/lib/__templates__/nextjs/src/components/ui/radio-group.tsx +45 -0
  84. package/lib/__templates__/nextjs/src/components/ui/resizable.tsx +63 -0
  85. package/lib/__templates__/nextjs/src/components/ui/scroll-area.tsx +58 -0
  86. package/lib/__templates__/nextjs/src/components/ui/select.tsx +190 -0
  87. package/lib/__templates__/nextjs/src/components/ui/separator.tsx +28 -0
  88. package/lib/__templates__/nextjs/src/components/ui/sheet.tsx +139 -0
  89. package/lib/__templates__/nextjs/src/components/ui/sidebar.tsx +724 -0
  90. package/lib/__templates__/nextjs/src/components/ui/skeleton.tsx +13 -0
  91. package/lib/__templates__/nextjs/src/components/ui/slider.tsx +63 -0
  92. package/lib/__templates__/nextjs/src/components/ui/sonner.tsx +40 -0
  93. package/lib/__templates__/nextjs/src/components/ui/spinner.tsx +16 -0
  94. package/lib/__templates__/nextjs/src/components/ui/switch.tsx +31 -0
  95. package/lib/__templates__/nextjs/src/components/ui/table.tsx +116 -0
  96. package/lib/__templates__/nextjs/src/components/ui/tabs.tsx +66 -0
  97. package/lib/__templates__/nextjs/src/components/ui/textarea.tsx +18 -0
  98. package/lib/__templates__/nextjs/src/components/ui/toggle-group.tsx +83 -0
  99. package/lib/__templates__/nextjs/src/components/ui/toggle.tsx +47 -0
  100. package/lib/__templates__/nextjs/src/components/ui/tooltip.tsx +61 -0
  101. package/lib/__templates__/nextjs/src/hooks/use-mobile.ts +19 -0
  102. package/lib/__templates__/nextjs/src/lib/utils.ts +6 -0
  103. package/lib/__templates__/nextjs/template.config.js +32 -2
  104. package/lib/__templates__/templates.json +61 -74
  105. package/lib/__templates__/vite/.coze +4 -3
  106. package/lib/__templates__/vite/README.md +204 -26
  107. package/lib/__templates__/vite/_npmrc +2 -1
  108. package/lib/__templates__/vite/eslint.config.mjs +9 -0
  109. package/lib/__templates__/vite/package.json +6 -2
  110. package/lib/__templates__/vite/pnpm-lock.yaml +3486 -19
  111. package/lib/__templates__/vite/scripts/dev.sh +7 -26
  112. package/lib/__templates__/{rsbuild/scripts/build.sh → vite/scripts/prepare.sh} +0 -5
  113. package/lib/__templates__/vite/src/main.ts +1 -2
  114. package/lib/__templates__/vite/template.config.js +39 -6
  115. package/lib/__templates__/vite/vite.config.ts +3 -3
  116. package/lib/cli.js +634 -313
  117. package/package.json +11 -4
  118. package/lib/__templates__/expo/.cozeproj/scripts/deploy_build.sh +0 -109
  119. package/lib/__templates__/expo/.cozeproj/scripts/deploy_run.sh +0 -257
  120. package/lib/__templates__/expo/app.json +0 -63
  121. package/lib/__templates__/expo/babel.config.js +0 -9
  122. package/lib/__templates__/expo/client/app/(tabs)/_layout.tsx +0 -43
  123. package/lib/__templates__/expo/client/app/(tabs)/home.tsx +0 -1
  124. package/lib/__templates__/expo/client/app/(tabs)/index.tsx +0 -7
  125. package/lib/__templates__/expo/client/app/+not-found.tsx +0 -79
  126. package/lib/__templates__/expo/client/index.js +0 -11
  127. package/lib/__templates__/expo/client/screens/home/index.tsx +0 -54
  128. package/lib/__templates__/expo/client/screens/home/styles.ts +0 -332
  129. package/lib/__templates__/expo/metro.config.js +0 -53
  130. package/lib/__templates__/nextjs/.vscode/settings.json +0 -121
  131. package/lib/__templates__/react-rsbuild/.coze +0 -11
  132. package/lib/__templates__/react-rsbuild/.vscode/settings.json +0 -121
  133. package/lib/__templates__/react-rsbuild/README.md +0 -61
  134. package/lib/__templates__/react-rsbuild/_gitignore +0 -97
  135. package/lib/__templates__/react-rsbuild/_npmrc +0 -22
  136. package/lib/__templates__/react-rsbuild/package.json +0 -31
  137. package/lib/__templates__/react-rsbuild/pnpm-lock.yaml +0 -997
  138. package/lib/__templates__/react-rsbuild/rsbuild.config.ts +0 -13
  139. package/lib/__templates__/react-rsbuild/scripts/dev.sh +0 -51
  140. package/lib/__templates__/react-rsbuild/scripts/start.sh +0 -15
  141. package/lib/__templates__/react-rsbuild/src/App.tsx +0 -60
  142. package/lib/__templates__/react-rsbuild/src/index.css +0 -21
  143. package/lib/__templates__/react-rsbuild/src/index.html +0 -12
  144. package/lib/__templates__/react-rsbuild/src/index.tsx +0 -16
  145. package/lib/__templates__/react-rsbuild/tailwind.config.js +0 -9
  146. package/lib/__templates__/react-rsbuild/template.config.js +0 -54
  147. package/lib/__templates__/react-rsbuild/tsconfig.json +0 -17
  148. package/lib/__templates__/rsbuild/.coze +0 -11
  149. package/lib/__templates__/rsbuild/.vscode/settings.json +0 -7
  150. package/lib/__templates__/rsbuild/README.md +0 -61
  151. package/lib/__templates__/rsbuild/_gitignore +0 -97
  152. package/lib/__templates__/rsbuild/_npmrc +0 -22
  153. package/lib/__templates__/rsbuild/package.json +0 -24
  154. package/lib/__templates__/rsbuild/pnpm-lock.yaml +0 -888
  155. package/lib/__templates__/rsbuild/rsbuild.config.ts +0 -12
  156. package/lib/__templates__/rsbuild/scripts/dev.sh +0 -51
  157. package/lib/__templates__/rsbuild/scripts/start.sh +0 -15
  158. package/lib/__templates__/rsbuild/src/index.css +0 -21
  159. package/lib/__templates__/rsbuild/src/index.html +0 -12
  160. package/lib/__templates__/rsbuild/src/index.ts +0 -5
  161. package/lib/__templates__/rsbuild/src/main.ts +0 -65
  162. package/lib/__templates__/rsbuild/tailwind.config.js +0 -9
  163. package/lib/__templates__/rsbuild/template.config.js +0 -56
  164. package/lib/__templates__/rsbuild/tsconfig.json +0 -16
  165. package/lib/__templates__/vite/.vscode/settings.json +0 -7
  166. /package/lib/__templates__/expo/{eslint-formatter-simple.mjs → client/eslint-formatter-simple.mjs} +0 -0
  167. /package/lib/__templates__/expo/{eslint.config.mjs → client/eslint.config.mjs} +0 -0
package/lib/cli.js CHANGED
@@ -5,12 +5,15 @@ var commander = require('commander');
5
5
  var path = require('path');
6
6
  var fs = require('fs');
7
7
  var shelljs = require('shelljs');
8
+ var perf_hooks = require('perf_hooks');
8
9
  var fs$1 = require('fs/promises');
10
+ var os = require('os');
9
11
  var toml = require('@iarna/toml');
10
12
  var jsYaml = require('js-yaml');
11
- var perf_hooks = require('perf_hooks');
13
+ var child_process = require('child_process');
12
14
  var addFormats = require('ajv-formats');
13
15
  var Ajv = require('ajv');
16
+ var minimist = require('minimist');
14
17
  var changeCase = require('change-case');
15
18
  var ejs = require('ejs');
16
19
 
@@ -124,7 +127,7 @@ const generateTemplatesHelpText = () => {
124
127
  return lines.join('\n');
125
128
  };
126
129
 
127
- function _nullishCoalesce$2(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } function _optionalChain$4(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) {
130
+ 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) {
128
131
  const ERROR = 0; LogLevel[LogLevel["ERROR"] = ERROR] = "ERROR";
129
132
  const WARN = 1; LogLevel[LogLevel["WARN"] = WARN] = "WARN";
130
133
  const SUCCESS = 2; LogLevel[LogLevel["SUCCESS"] = SUCCESS] = "SUCCESS";
@@ -171,7 +174,7 @@ class Logger {
171
174
  return level;
172
175
  }
173
176
 
174
- const envLevel = _optionalChain$4([process, 'access', _ => _.env, 'access', _2 => _2.LOG_LEVEL, 'optionalAccess', _3 => _3.toLowerCase, 'call', _4 => _4()]);
177
+ const envLevel = _optionalChain$3([process, 'access', _ => _.env, 'access', _2 => _2.LOG_LEVEL, 'optionalAccess', _3 => _3.toLowerCase, 'call', _4 => _4()]);
175
178
  if (envLevel && envLevel in LOG_LEVEL_MAP) {
176
179
  return LOG_LEVEL_MAP[envLevel];
177
180
  }
@@ -183,7 +186,7 @@ class Logger {
183
186
  // 简单检测:Node.js 环境且支持 TTY
184
187
  return (
185
188
  typeof process !== 'undefined' &&
186
- _optionalChain$4([process, 'access', _5 => _5.stdout, 'optionalAccess', _6 => _6.isTTY]) === true &&
189
+ _optionalChain$3([process, 'access', _5 => _5.stdout, 'optionalAccess', _6 => _6.isTTY]) === true &&
187
190
  process.env.NO_COLOR === undefined
188
191
  );
189
192
  }
@@ -269,7 +272,330 @@ const createLogger = (options = {}) =>
269
272
  // 导出默认实例
270
273
  const logger = createLogger();
271
274
 
272
- 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; }/* eslint-disable @typescript-eslint/no-explicit-any */
275
+ /**
276
+ * 时间统计工具
277
+ */
278
+ class TimeTracker {
279
+
280
+
281
+
282
+ constructor() {
283
+ this.startTime = perf_hooks.performance.now();
284
+ this.lastTime = this.startTime;
285
+ }
286
+
287
+ /**
288
+ * 记录阶段耗时
289
+ * @param phaseName 阶段名称
290
+ */
291
+ logPhase(phaseName) {
292
+ const now = perf_hooks.performance.now();
293
+ const phaseTime = now - this.lastTime;
294
+ this.lastTime = now;
295
+
296
+ logger.verbose(`⏱ ${phaseName}: ${phaseTime.toFixed(2)}ms`);
297
+ }
298
+
299
+ /**
300
+ * 记录总耗时
301
+ */
302
+ logTotal() {
303
+ const totalTime = perf_hooks.performance.now() - this.startTime;
304
+ logger.verbose(`⏱ Total time: ${totalTime.toFixed(2)}ms`);
305
+ }
306
+
307
+ /**
308
+ * 获取当前耗时(不输出日志)
309
+ * @returns 从开始到现在的总耗时(毫秒)
310
+ */
311
+ getElapsedTime() {
312
+ return perf_hooks.performance.now() - this.startTime;
313
+ }
314
+ }
315
+
316
+ /**
317
+ * 获取模板配置文件路径
318
+ * @returns templates.json 的绝对路径
319
+ */
320
+ const getTemplatesConfigPath = () => {
321
+ const configPath = path.resolve(getTemplatesDir(), 'templates.json');
322
+ logger.verbose(`Templates config path: ${configPath}`);
323
+ logger.verbose(`Config file exists: ${fs.existsSync(configPath)}`);
324
+ return configPath;
325
+ };
326
+
327
+ /**
328
+ * 获取模板目录路径
329
+ * @returns __templates__ 目录的绝对路径
330
+ */
331
+ const getTemplatesDir = () => {
332
+ const templatesDir = path.resolve(__dirname, './__templates__');
333
+ logger.verbose(`Templates directory: ${templatesDir}`);
334
+ logger.verbose(`Templates directory exists: ${fs.existsSync(templatesDir)}`);
335
+ return templatesDir;
336
+ };
337
+
338
+ /**
339
+ * 加载模板配置文件
340
+ * 支持 .ts 和 .js 文件(通过 sucrase 注册)
341
+ *
342
+ * @param templatePath - 模板目录路径
343
+ * @returns 模板配置对象
344
+ */
345
+
346
+ const loadTemplateConfig = async (
347
+ templatePath,
348
+ ) => {
349
+ logger.verbose(`Loading template config from: ${templatePath}`);
350
+
351
+ const tsConfigPath = path.join(templatePath, 'template.config.ts');
352
+ const jsConfigPath = path.join(templatePath, 'template.config.js');
353
+
354
+ logger.verbose('Checking for config files:');
355
+ logger.verbose(` - TypeScript: ${tsConfigPath}`);
356
+ logger.verbose(` - JavaScript: ${jsConfigPath}`);
357
+
358
+ let configPath;
359
+
360
+ const [tsExists, jsExists] = await Promise.all([
361
+ fs$1.access(tsConfigPath).then(
362
+ () => true,
363
+ () => false,
364
+ ),
365
+ fs$1.access(jsConfigPath).then(
366
+ () => true,
367
+ () => false,
368
+ ),
369
+ ]);
370
+
371
+ logger.verbose('Config file existence check:');
372
+ logger.verbose(` - template.config.ts: ${tsExists}`);
373
+ logger.verbose(` - template.config.js: ${jsExists}`);
374
+
375
+ if (tsExists) {
376
+ configPath = tsConfigPath;
377
+ } else if (jsExists) {
378
+ configPath = jsConfigPath;
379
+ } else {
380
+ throw new Error(
381
+ `Template config not found in ${templatePath}.\n` +
382
+ 'Expected: template.config.ts or template.config.js',
383
+ );
384
+ }
385
+
386
+ logger.verbose(`Using config file: ${configPath}`);
387
+
388
+ // eslint-disable-next-line @typescript-eslint/no-require-imports, security/detect-non-literal-require -- Sucrase handles .ts files at runtime, path is validated above
389
+ const config = require(configPath);
390
+
391
+ logger.verbose('Template config loaded successfully');
392
+
393
+ return config.default || config;
394
+ };
395
+
396
+ /**
397
+ * 加载模板列表配置
398
+ *
399
+ * @param configPath - templates.json 配置文件路径
400
+ * @returns 模板列表配置
401
+ */
402
+ const loadTemplatesConfig = async (
403
+ configPath,
404
+ ) => {
405
+ logger.verbose(`Loading templates config from: ${configPath}`);
406
+
407
+ const content = await fs$1.readFile(configPath, 'utf-8');
408
+ // eslint-disable-next-line no-restricted-syntax -- Static config file loaded at build time, safeJsonParse not needed
409
+ const config = JSON.parse(content) ;
410
+
411
+ logger.verbose(
412
+ `Found ${config.templates.length} templates: ${config.templates.map(t => t.name).join(', ')}`,
413
+ );
414
+
415
+ return config;
416
+ };
417
+
418
+ /**
419
+ * 根据模板名称查找模板元信息
420
+ *
421
+ * @param templatesConfig - 模板列表配置
422
+ * @param templateName - 模板名称
423
+ * @returns 模板元信息
424
+ */
425
+ const findTemplate = (
426
+ templatesConfig,
427
+ templateName,
428
+ ) => {
429
+ const template = templatesConfig.templates.find(t => t.name === templateName);
430
+
431
+ if (!template) {
432
+ const availableTemplates = templatesConfig.templates
433
+ .map(t => t.name)
434
+ .join(', ');
435
+ throw new Error(
436
+ `Template "${templateName}" not found.\n` +
437
+ `Available templates: ${availableTemplates}\n` +
438
+ 'Use --template <name> to specify a template.',
439
+ );
440
+ }
441
+
442
+ return template;
443
+ };
444
+
445
+ /**
446
+ * 获取模板的完整路径
447
+ *
448
+ * @param basePath - 模板目录(templates.json 所在目录)
449
+ * @param templateMetadata - 模板元信息
450
+ * @returns 模板完整路径
451
+ */
452
+ const getTemplatePath = async (
453
+ basePath,
454
+ templateMetadata,
455
+ ) => {
456
+ logger.verbose('Resolving template path:');
457
+ logger.verbose(` - Base path: ${basePath}`);
458
+ logger.verbose(` - Template location: ${templateMetadata.location}`);
459
+
460
+ // location 是相对于 templates.json 文件的路径
461
+ const templatePath = path.join(basePath, templateMetadata.location);
462
+
463
+ logger.verbose(` - Resolved path: ${templatePath}`);
464
+
465
+ try {
466
+ await fs$1.access(templatePath);
467
+ logger.verbose(' - Template directory exists: ✓');
468
+ // eslint-disable-next-line @coze-arch/use-error-in-catch -- Error handling done in throw statement
469
+ } catch (e) {
470
+ logger.error(' - Template directory does not exist: ✗');
471
+ throw new Error(`Template directory not found: ${templatePath}`);
472
+ }
473
+
474
+ return templatePath;
475
+ };
476
+
477
+ /**
478
+ * 对单个模板执行 pnpm install
479
+ */
480
+ const warmupTemplate = (templatePath, templateName) => {
481
+ logger.info(`\nWarming up template: ${templateName}`);
482
+ logger.info(` Path: ${templatePath}`);
483
+
484
+ const result = shelljs.exec('pnpm install', {
485
+ cwd: templatePath,
486
+ silent: true,
487
+ });
488
+
489
+ // 输出 stdout
490
+ if (result.stdout) {
491
+ process.stdout.write(result.stdout);
492
+ }
493
+
494
+ // 输出 stderr
495
+ if (result.stderr) {
496
+ process.stderr.write(result.stderr);
497
+ }
498
+
499
+ if (result.code === 0) {
500
+ logger.success(` ✓ ${templateName} warmed up successfully`);
501
+ } else {
502
+ const errorMessage = [
503
+ `pnpm install failed for ${templateName} with exit code ${result.code}`,
504
+ result.stderr ? `\nStderr:\n${result.stderr}` : '',
505
+ result.stdout ? `\nStdout:\n${result.stdout}` : '',
506
+ ]
507
+ .filter(Boolean)
508
+ .join('');
509
+
510
+ throw new Error(errorMessage);
511
+ }
512
+ };
513
+
514
+ /**
515
+ * 执行 warmup 命令的内部实现
516
+ */
517
+ const executeWarmup = async (
518
+ options
519
+
520
+ ,
521
+
522
+ command,
523
+ ) => {
524
+ const timer = new TimeTracker();
525
+
526
+ try {
527
+ const { template: templateFilter } = options;
528
+
529
+ logger.info('Starting template warmup...');
530
+ timer.logPhase('Initialization');
531
+
532
+ // 加载模板配置
533
+ const configPath = getTemplatesConfigPath();
534
+ const templatesConfig = await loadTemplatesConfig(configPath);
535
+
536
+ logger.verbose(
537
+ `Found ${templatesConfig.templates.length} templates in config`,
538
+ );
539
+
540
+ // 过滤模板
541
+ const templatesToWarmup = templateFilter
542
+ ? templatesConfig.templates.filter(t => t.name === templateFilter)
543
+ : templatesConfig.templates;
544
+
545
+ if (templatesToWarmup.length === 0) {
546
+ if (templateFilter) {
547
+ logger.warn(`Template "${templateFilter}" not found`);
548
+ logger.info(
549
+ `Available templates: ${templatesConfig.templates.map(t => t.name).join(', ')}`,
550
+ );
551
+ } else {
552
+ logger.warn('No templates found');
553
+ }
554
+ return;
555
+ }
556
+
557
+ logger.info(
558
+ `\nWill warm up ${templatesToWarmup.length} template(s): ${templatesToWarmup.map(t => t.name).join(', ')}`,
559
+ );
560
+
561
+ // 获取模板基础路径
562
+ const basePath = configPath.replace(/\/templates\.json$/, '');
563
+
564
+ // 对每个模板执行 pnpm install
565
+ for (const templateMetadata of templatesToWarmup) {
566
+ const templatePath = await getTemplatePath(basePath, templateMetadata);
567
+ warmupTemplate(templatePath, templateMetadata.name);
568
+ timer.logPhase(`Warmup ${templateMetadata.name}`);
569
+ }
570
+
571
+ logger.success('\n✅ All templates warmed up successfully!');
572
+ logger.info(
573
+ '\nNext time you run `coze init`, it will be much faster as node_modules are pre-installed.',
574
+ );
575
+
576
+ timer.logTotal();
577
+ } catch (error) {
578
+ timer.logTotal();
579
+ logger.error('Failed to warmup templates:');
580
+ logger.error(error instanceof Error ? error.message : String(error));
581
+ process.exit(1);
582
+ }
583
+ };
584
+
585
+ /**
586
+ * 注册 warmup 命令到 program
587
+ */
588
+ const registerCommand$2 = program => {
589
+ program
590
+ .command('warmup')
591
+ .description('Pre-install dependencies for templates to speed up init')
592
+ .option('-t, --template <name>', 'Warmup a specific template only')
593
+ .action(async (options, command) => {
594
+ await executeWarmup(options);
595
+ });
596
+ };
597
+
598
+ 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 */
273
599
  // Safe JSON parsing utilities with type safety and error handling
274
600
  // Provides fallback values, validation, and error monitoring capabilities
275
601
 
@@ -360,12 +686,12 @@ function safeJsonParse(
360
686
  const parsed = JSON.parse(String(input));
361
687
 
362
688
  // Optional validation
363
- if (_optionalChain$3([options, 'optionalAccess', _ => _.validate])) {
689
+ if (_optionalChain$2([options, 'optionalAccess', _ => _.validate])) {
364
690
  if (options.validate(parsed)) {
365
691
  return parsed;
366
692
  } else {
367
693
  const validationError = new Error('JSON validation failed');
368
- _optionalChain$3([options, 'access', _2 => _2.onError, 'optionalCall', _3 => _3(validationError, input)]);
694
+ _optionalChain$2([options, 'access', _2 => _2.onError, 'optionalCall', _3 => _3(validationError, input)]);
369
695
 
370
696
  if (options.throwOnValidationError) {
371
697
  throw validationError;
@@ -377,15 +703,15 @@ function safeJsonParse(
377
703
  return parsed;
378
704
  } catch (error) {
379
705
  // Re-throw validation errors when throwOnValidationError is true
380
- if (error instanceof Error && error.message === 'JSON validation failed' && _optionalChain$3([options, 'optionalAccess', _4 => _4.throwOnValidationError])) {
706
+ if (error instanceof Error && error.message === 'JSON validation failed' && _optionalChain$2([options, 'optionalAccess', _4 => _4.throwOnValidationError])) {
381
707
  throw error;
382
708
  }
383
- _optionalChain$3([options, 'optionalAccess', _5 => _5.onError, 'optionalCall', _6 => _6(error , input)]);
709
+ _optionalChain$2([options, 'optionalAccess', _5 => _5.onError, 'optionalCall', _6 => _6(error , input)]);
384
710
  return defaultValue;
385
711
  }
386
712
  }
387
713
 
388
- 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; }
714
+ 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; }
389
715
 
390
716
 
391
717
  /**
@@ -475,13 +801,13 @@ const getCommandConfig = (
475
801
  // 根据命令名称映射到配置路径
476
802
  switch (commandName) {
477
803
  case 'dev':
478
- commandConfig = _optionalChain$2([config, 'access', _ => _.dev, 'optionalAccess', _2 => _2.run]);
804
+ commandConfig = _optionalChain$1([config, 'access', _ => _.dev, 'optionalAccess', _2 => _2.run]);
479
805
  break;
480
806
  case 'build':
481
- commandConfig = _optionalChain$2([config, 'access', _3 => _3.deploy, 'optionalAccess', _4 => _4.build]);
807
+ commandConfig = _optionalChain$1([config, 'access', _3 => _3.deploy, 'optionalAccess', _4 => _4.build]);
482
808
  break;
483
809
  case 'start':
484
- commandConfig = _optionalChain$2([config, 'access', _5 => _5.deploy, 'optionalAccess', _6 => _6.run]);
810
+ commandConfig = _optionalChain$1([config, 'access', _5 => _5.deploy, 'optionalAccess', _6 => _6.run]);
485
811
  break;
486
812
  default:
487
813
  throw new Error(`Unknown command: ${commandName}`);
@@ -497,26 +823,51 @@ const getCommandConfig = (
497
823
  return commandConfig;
498
824
  };
499
825
 
500
- function _nullishCoalesce$1(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } 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; }
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; }
501
827
 
502
828
  /**
503
- * 创建日志管理器
829
+ * 日志文件名常量
504
830
  */
505
- const createLogManager = (logDir = '.coze-logs') => {
506
- const ensureLogDir = () => {
507
- if (!fs.existsSync(logDir)) {
508
- fs.mkdirSync(logDir, { recursive: true });
509
- }
510
- };
831
+ const LOG_FILE_NAME = 'dev.log';
511
832
 
512
- const getLogPath = (logFile) => path.join(logDir, logFile);
833
+ /**
834
+ * 获取日志目录
835
+ * 优先使用环境变量 COZE_LOG_DIR,否则使用 ~/.coze-logs
836
+ */
837
+ const getLogDir = () =>
838
+ process.env.COZE_LOG_DIR || path.join(os.homedir(), '.coze-logs');
513
839
 
514
- return {
515
- createWriteStream: (logFile) => {
516
- ensureLogDir();
517
- return fs.createWriteStream(getLogPath(logFile), { flags: 'a' });
518
- },
519
- };
840
+ /**
841
+ * 解析日志文件路径
842
+ * - 如果是绝对路径,直接使用
843
+ * - 如果是相对路径,基于 getLogDir() + 相对路径
844
+ * - 如果为空,使用 getLogDir() + LOG_FILE_NAME
845
+ */
846
+ const resolveLogFilePath = (logFile) => {
847
+ if (!logFile) {
848
+ return path.join(getLogDir(), LOG_FILE_NAME);
849
+ }
850
+
851
+ if (path.isAbsolute(logFile)) {
852
+ return logFile;
853
+ }
854
+
855
+ return path.join(getLogDir(), logFile);
856
+ };
857
+
858
+ /**
859
+ * 创建日志写入流
860
+ */
861
+ const createLogStream = (logFilePath) => {
862
+ const logDir = path.dirname(logFilePath);
863
+
864
+ // 确保日志目录存在
865
+ if (!fs.existsSync(logDir)) {
866
+ fs.mkdirSync(logDir, { recursive: true });
867
+ }
868
+
869
+ // 使用 'w' 标志覆盖之前的日志
870
+ return fs.createWriteStream(logFilePath, { flags: 'w' });
520
871
  };
521
872
 
522
873
  /**
@@ -534,16 +885,15 @@ const executeRun = async (
534
885
  const commandArgs = getCommandConfig(config, commandName);
535
886
 
536
887
  // 2. 准备日志
537
- const logManager = createLogManager();
538
- const logFile = options.logFile || `${commandName}.log`;
539
- const logStream = logManager.createWriteStream(logFile);
888
+ const logFilePath = resolveLogFilePath(options.logFile);
889
+ const logStream = createLogStream(logFilePath);
540
890
 
541
891
  // 3. 执行命令
542
892
  const commandString = commandArgs.join(' ');
543
893
 
544
894
  logger.info(`Executing: ${commandString}`);
545
895
  logger.info(`Working directory: ${process.cwd()}`);
546
- logger.info(`Log file: ${logFile}`);
896
+ logger.info(`Log file: ${logFilePath}`);
547
897
 
548
898
  const childProcess = shelljs.exec(commandString, {
549
899
  async: true,
@@ -555,12 +905,12 @@ const executeRun = async (
555
905
  }
556
906
 
557
907
  // 将输出同时写入控制台和日志文件
558
- _optionalChain$1([childProcess, 'access', _ => _.stdout, 'optionalAccess', _2 => _2.on, 'call', _3 => _3('data', (data) => {
908
+ _optionalChain([childProcess, 'access', _ => _.stdout, 'optionalAccess', _2 => _2.on, 'call', _3 => _3('data', (data) => {
559
909
  process.stdout.write(data);
560
910
  logStream.write(data);
561
911
  })]);
562
912
 
563
- _optionalChain$1([childProcess, 'access', _4 => _4.stderr, 'optionalAccess', _5 => _5.on, 'call', _6 => _6('data', (data) => {
913
+ _optionalChain([childProcess, 'access', _4 => _4.stderr, 'optionalAccess', _5 => _5.on, 'call', _6 => _6('data', (data) => {
564
914
  process.stderr.write(data);
565
915
  logStream.write(data);
566
916
  })]);
@@ -572,11 +922,11 @@ const executeRun = async (
572
922
  logger.error(
573
923
  `Command exited with code ${_nullishCoalesce$1(code, () => ( 'unknown'))}${signal ? ` and signal ${signal}` : ''}`,
574
924
  );
575
- logger.error(`Check log file for details: ${logFile}`);
925
+ logger.error(`Check log file for details: ${logFilePath}`);
576
926
  process.exit(code || 1);
577
927
  } else {
578
928
  logger.success('Command completed successfully');
579
- logger.info(`Log file: ${logFile}`);
929
+ logger.info(`Log file: ${logFilePath}`);
580
930
  }
581
931
  });
582
932
 
@@ -624,72 +974,48 @@ const registerCommand$1 = program => {
624
974
  .description('Start production server')
625
975
  .option('--log-file <path>', 'Log file path')
626
976
  .action(async options => {
627
- await executeRun('start', options);
628
- });
629
- };
630
-
631
- /**
632
- * 时间统计工具
633
- */
634
- class TimeTracker {
635
-
636
-
637
-
638
- constructor() {
639
- this.startTime = perf_hooks.performance.now();
640
- this.lastTime = this.startTime;
641
- }
642
-
643
- /**
644
- * 记录阶段耗时
645
- * @param phaseName 阶段名称
646
- */
647
- logPhase(phaseName) {
648
- const now = perf_hooks.performance.now();
649
- const phaseTime = now - this.lastTime;
650
- this.lastTime = now;
651
-
652
- logger.verbose(`⏱ ${phaseName}: ${phaseTime.toFixed(2)}ms`);
653
- }
654
-
655
- /**
656
- * 记录总耗时
657
- */
658
- logTotal() {
659
- const totalTime = perf_hooks.performance.now() - this.startTime;
660
- logger.verbose(`⏱ Total time: ${totalTime.toFixed(2)}ms`);
661
- }
662
-
663
- /**
664
- * 获取当前耗时(不输出日志)
665
- * @returns 从开始到现在的总耗时(毫秒)
666
- */
667
- getElapsedTime() {
668
- return perf_hooks.performance.now() - this.startTime;
669
- }
670
- }
671
-
672
- /**
673
- * 获取模板配置文件路径
674
- * @returns templates.json 的绝对路径
675
- */
676
- const getTemplatesConfigPath = () => {
677
- const configPath = path.resolve(getTemplatesDir(), 'templates.json');
678
- logger.verbose(`Templates config path: ${configPath}`);
679
- logger.verbose(`Config file exists: ${fs.existsSync(configPath)}`);
680
- return configPath;
977
+ await executeRun('start', options);
978
+ });
681
979
  };
682
980
 
683
981
  /**
684
- * 获取模板目录路径
685
- * @returns __templates__ 目录的绝对路径
982
+ * 在后台启动一个独立的子进程
983
+ * 类似于 `setsid command args >/dev/null 2>&1 &`
984
+ *
985
+ * @param command - 要执行的命令 (例如: 'npm', 'node', 'bash')
986
+ * @param args - 命令参数数组 (例如: ['run', 'dev'])
987
+ * @param options - 配置选项
988
+ * @returns 子进程的 PID
686
989
  */
687
- const getTemplatesDir = () => {
688
- const templatesDir = path.resolve(__dirname, './__templates__');
689
- logger.verbose(`Templates directory: ${templatesDir}`);
690
- logger.verbose(`Templates directory exists: ${fs.existsSync(templatesDir)}`);
691
- return templatesDir;
692
- };
990
+ function spawnDetached(
991
+ command,
992
+ args,
993
+ options,
994
+ ) {
995
+ const { cwd, verbose = true } = options;
996
+ const isWindows = os.platform() === 'win32';
997
+
998
+ if (verbose) {
999
+ console.log(`Spawning detached process: ${command} ${args.join(' ')}`);
1000
+ console.log(`Working directory: ${cwd}`);
1001
+ }
1002
+
1003
+ // 使用 spawn 创建后台子进程
1004
+ const child = child_process.spawn(command, args, {
1005
+ cwd,
1006
+ detached: !isWindows, // Windows 不完全支持 detached,但仍可以使用
1007
+ stdio: 'ignore', // 忽略所有输入输出,让进程完全独立运行
1008
+ });
1009
+
1010
+ // 分离父子进程引用,允许父进程退出而不等待子进程
1011
+ child.unref();
1012
+
1013
+ if (verbose && child.pid) {
1014
+ console.log(`Process started with PID: ${child.pid}`);
1015
+ }
1016
+
1017
+ return child.pid;
1018
+ }
693
1019
 
694
1020
  /**
695
1021
  * 创建 AJV 验证器实例
@@ -739,149 +1065,12 @@ const validateParams = (
739
1065
  return params ;
740
1066
  };
741
1067
 
742
- /**
743
- * 加载模板配置文件
744
- * 支持 .ts 和 .js 文件(通过 sucrase 注册)
745
- *
746
- * @param templatePath - 模板目录路径
747
- * @returns 模板配置对象
748
- */
749
-
750
- const loadTemplateConfig = async (
751
- templatePath,
752
- ) => {
753
- logger.verbose(`Loading template config from: ${templatePath}`);
754
-
755
- const tsConfigPath = path.join(templatePath, 'template.config.ts');
756
- const jsConfigPath = path.join(templatePath, 'template.config.js');
757
-
758
- logger.verbose('Checking for config files:');
759
- logger.verbose(` - TypeScript: ${tsConfigPath}`);
760
- logger.verbose(` - JavaScript: ${jsConfigPath}`);
761
-
762
- let configPath;
763
-
764
- const [tsExists, jsExists] = await Promise.all([
765
- fs$1.access(tsConfigPath).then(
766
- () => true,
767
- () => false,
768
- ),
769
- fs$1.access(jsConfigPath).then(
770
- () => true,
771
- () => false,
772
- ),
773
- ]);
774
-
775
- logger.verbose('Config file existence check:');
776
- logger.verbose(` - template.config.ts: ${tsExists}`);
777
- logger.verbose(` - template.config.js: ${jsExists}`);
778
-
779
- if (tsExists) {
780
- configPath = tsConfigPath;
781
- } else if (jsExists) {
782
- configPath = jsConfigPath;
783
- } else {
784
- throw new Error(
785
- `Template config not found in ${templatePath}.\n` +
786
- 'Expected: template.config.ts or template.config.js',
787
- );
788
- }
789
-
790
- logger.verbose(`Using config file: ${configPath}`);
791
-
792
- // eslint-disable-next-line @typescript-eslint/no-require-imports, security/detect-non-literal-require -- Sucrase handles .ts files at runtime, path is validated above
793
- const config = require(configPath);
794
-
795
- logger.verbose('Template config loaded successfully');
796
-
797
- return config.default || config;
798
- };
799
-
800
- /**
801
- * 加载模板列表配置
802
- *
803
- * @param configPath - templates.json 配置文件路径
804
- * @returns 模板列表配置
805
- */
806
- const loadTemplatesConfig = async (
807
- configPath,
808
- ) => {
809
- logger.verbose(`Loading templates config from: ${configPath}`);
810
-
811
- const content = await fs$1.readFile(configPath, 'utf-8');
812
- // eslint-disable-next-line no-restricted-syntax -- Static config file loaded at build time, safeJsonParse not needed
813
- const config = JSON.parse(content) ;
814
-
815
- logger.verbose(
816
- `Found ${config.templates.length} templates: ${config.templates.map(t => t.name).join(', ')}`,
817
- );
818
-
819
- return config;
820
- };
821
-
822
- /**
823
- * 根据模板名称查找模板元信息
824
- *
825
- * @param templatesConfig - 模板列表配置
826
- * @param templateName - 模板名称
827
- * @returns 模板元信息
828
- */
829
- const findTemplate = (
830
- templatesConfig,
831
- templateName,
832
- ) => {
833
- const template = templatesConfig.templates.find(t => t.name === templateName);
834
-
835
- if (!template) {
836
- const availableTemplates = templatesConfig.templates
837
- .map(t => t.name)
838
- .join(', ');
839
- throw new Error(
840
- `Template "${templateName}" not found.\n` +
841
- `Available templates: ${availableTemplates}\n` +
842
- 'Use --template <name> to specify a template.',
843
- );
844
- }
845
-
846
- return template;
847
- };
848
-
849
- /**
850
- * 获取模板的完整路径
851
- *
852
- * @param basePath - 模板目录(templates.json 所在目录)
853
- * @param templateMetadata - 模板元信息
854
- * @returns 模板完整路径
855
- */
856
- const getTemplatePath = async (
857
- basePath,
858
- templateMetadata,
859
- ) => {
860
- logger.verbose('Resolving template path:');
861
- logger.verbose(` - Base path: ${basePath}`);
862
- logger.verbose(` - Template location: ${templateMetadata.location}`);
863
-
864
- // location 是相对于 templates.json 文件的路径
865
- const templatePath = path.join(basePath, templateMetadata.location);
866
-
867
- logger.verbose(` - Resolved path: ${templatePath}`);
868
-
869
- try {
870
- await fs$1.access(templatePath);
871
- logger.verbose(' - Template directory exists: ✓');
872
- // eslint-disable-next-line @coze-arch/use-error-in-catch -- Error handling done in throw statement
873
- } catch (e) {
874
- logger.error(' - Template directory does not exist: ✗');
875
- throw new Error(`Template directory not found: ${templatePath}`);
876
- }
877
-
878
- return templatePath;
879
- };
880
-
881
1068
  /**
882
1069
  * 从 Commander 解析透传参数
883
1070
  * 将 kebab-case 的 CLI 参数转换为 camelCase
884
1071
  *
1072
+ * 使用 minimist 解析 process.argv,自动处理类型转换
1073
+ *
885
1074
  * @param command - Commander 命令实例
886
1075
  * @param knownOptions - 已知的选项集合(不需要透传的选项)
887
1076
  * @returns 参数对象
@@ -890,20 +1079,34 @@ const parsePassThroughParams = (
890
1079
  command,
891
1080
  knownOptions = new Set(),
892
1081
  ) => {
893
- const rawOptions = command.opts();
1082
+ // 使用 minimist 解析所有参数
1083
+ // eslint-disable-next-line @typescript-eslint/no-magic-numbers -- slice(2) to skip node and script path
1084
+ const parsed = minimist(process.argv.slice(2));
894
1085
 
895
- const initial = {};
896
- return Object.entries(rawOptions).reduce(
1086
+ // 过滤掉已知选项和位置参数(_)
1087
+ const filtered = Object.entries(parsed).reduce(
897
1088
  (params, [key, value]) => {
898
- if (knownOptions.has(key)) {
1089
+ // 跳过 minimist 的位置参数数组
1090
+ if (key === '_') {
1091
+ return params;
1092
+ }
1093
+
1094
+ // 跳过已知选项(支持原始格式和 camelCase 格式)
1095
+ if (knownOptions.has(key) || knownOptions.has(changeCase.camelCase(key))) {
899
1096
  return params;
900
1097
  }
901
1098
 
1099
+ // 将 kebab-case 转换为 camelCase
902
1100
  const camelKey = changeCase.camelCase(key);
903
- return { ...params, [camelKey]: value };
1101
+ // eslint-disable-next-line security/detect-object-injection -- camelKey is sanitized by camelCase
1102
+ params[camelKey] = value;
1103
+
1104
+ return params;
904
1105
  },
905
- initial,
1106
+ {},
906
1107
  );
1108
+
1109
+ return filtered;
907
1110
  };
908
1111
 
909
1112
  /**
@@ -1069,17 +1272,139 @@ const convertDotfileName = (filePath) => {
1069
1272
  };
1070
1273
 
1071
1274
  /**
1072
- * 复制并处理模板文件到目标目录
1275
+ * 执行文件渲染钩子
1073
1276
  *
1074
- * @param templatePath - 模板目录路径
1075
- * @param outputPath - 输出目录路径
1277
+ * @param templateConfig - 模板配置
1278
+ * @param fileInfo - 文件渲染信息
1076
1279
  * @param context - 模板上下文
1280
+ * @returns 处理后的文件信息,或 null 表示跳过该文件
1077
1281
  */
1078
- const processTemplateFiles = async (
1079
- templatePath,
1080
- outputPath,
1282
+ const executeFileRenderHook = async (
1283
+ templateConfig,
1284
+ fileInfo,
1081
1285
  context,
1082
1286
  ) => {
1287
+ if (!templateConfig.onFileRender) {
1288
+ return fileInfo;
1289
+ }
1290
+
1291
+ const result = await templateConfig.onFileRender(
1292
+ fileInfo,
1293
+ context,
1294
+ );
1295
+
1296
+ // false: 跳过文件
1297
+ if (result === false) {
1298
+ return null;
1299
+ }
1300
+
1301
+ // undefined/void: 使用默认内容
1302
+ if (result === undefined || result === null) {
1303
+ return fileInfo;
1304
+ }
1305
+
1306
+ // string: 作为 content,其他不变
1307
+ if (typeof result === 'string') {
1308
+ return {
1309
+ ...fileInfo,
1310
+ content: result,
1311
+ };
1312
+ }
1313
+
1314
+ // FileRenderInfo: 使用新对象的信息
1315
+ return result;
1316
+ };
1317
+
1318
+ /**
1319
+ * 处理单个文件
1320
+ */
1321
+ const processSingleFile = async (options
1322
+
1323
+
1324
+
1325
+
1326
+
1327
+ ) => {
1328
+ const { file, templatePath, outputPath, context, templateConfig } = options;
1329
+
1330
+ const srcPath = path.join(templatePath, file);
1331
+ const destFile = convertDotfileName(file);
1332
+
1333
+ logger.verbose(
1334
+ ` - Processing: ${file}${destFile !== file ? ` -> ${destFile}` : ''}`,
1335
+ );
1336
+
1337
+ // 判断是否为二进制文件
1338
+ const isBinary = !shouldRenderFile(srcPath);
1339
+ let content;
1340
+ let wasRendered = false;
1341
+
1342
+ if (isBinary) {
1343
+ // 二进制文件,读取为 buffer 然后转为 base64
1344
+ const buffer = await fs$1.readFile(srcPath);
1345
+ content = buffer.toString('base64');
1346
+ } else {
1347
+ // 文本文件,渲染后的内容
1348
+ content = await renderTemplate(srcPath, context);
1349
+ wasRendered = true;
1350
+ }
1351
+
1352
+ // 构造文件信息对象
1353
+ const fileInfo = {
1354
+ path: file,
1355
+ destPath: destFile,
1356
+ content,
1357
+ isBinary,
1358
+ wasRendered,
1359
+ };
1360
+
1361
+ // 执行文件渲染钩子
1362
+ const processedFileInfo = await executeFileRenderHook(
1363
+ templateConfig,
1364
+ fileInfo,
1365
+ context,
1366
+ );
1367
+
1368
+ // 如果返回 null,跳过该文件
1369
+ if (processedFileInfo === null) {
1370
+ logger.verbose(' ⊘ Skipped by onFileRender hook');
1371
+ return;
1372
+ }
1373
+
1374
+ // 使用处理后的目标路径
1375
+ const finalDestPath = path.join(outputPath, processedFileInfo.destPath);
1376
+
1377
+ // 确保目标目录存在
1378
+ await ensureDir(path.dirname(finalDestPath));
1379
+
1380
+ // 写入文件
1381
+ if (processedFileInfo.isBinary) {
1382
+ // 二进制文件:如果内容没变,直接复制;否则从 base64 解码写入
1383
+ if (processedFileInfo.content === content) {
1384
+ await fs$1.copyFile(srcPath, finalDestPath);
1385
+ logger.verbose(' ✓ Copied (binary)');
1386
+ } else {
1387
+ const buffer = Buffer.from(processedFileInfo.content, 'base64');
1388
+ await fs$1.writeFile(finalDestPath, buffer);
1389
+ logger.verbose(' ✓ Written (binary, modified by hook)');
1390
+ }
1391
+ } else {
1392
+ // 文本文件
1393
+ await fs$1.writeFile(finalDestPath, processedFileInfo.content, 'utf-8');
1394
+ logger.verbose(' ✓ Rendered and written');
1395
+ }
1396
+ };
1397
+
1398
+ /**
1399
+ * 复制并处理模板文件到目标目录
1400
+ */
1401
+ const processTemplateFiles = async (options
1402
+
1403
+
1404
+
1405
+
1406
+ ) => {
1407
+ const { templatePath, outputPath, context, templateConfig } = options;
1083
1408
  logger.verbose('Processing template files:');
1084
1409
  logger.verbose(` - Template path: ${templatePath}`);
1085
1410
  logger.verbose(` - Output path: ${outputPath}`);
@@ -1110,29 +1435,15 @@ const processTemplateFiles = async (
1110
1435
  }
1111
1436
 
1112
1437
  await Promise.all(
1113
- files.map(async file => {
1114
- const srcPath = path.join(templatePath, file);
1115
- const destFile = convertDotfileName(file);
1116
- const destPath = path.join(outputPath, destFile);
1117
-
1118
- logger.verbose(
1119
- ` - Processing: ${file}${destFile !== file ? ` -> ${destFile}` : ''}`,
1120
- );
1121
-
1122
- // 确保目标目录存在
1123
- await ensureDir(path.dirname(destPath));
1124
-
1125
- if (shouldRenderFile(srcPath)) {
1126
- // 渲染文本文件
1127
- const rendered = await renderTemplate(srcPath, context);
1128
- await fs$1.writeFile(destPath, rendered, 'utf-8');
1129
- logger.verbose(' ✓ Rendered and written');
1130
- } else {
1131
- // 直接复制二进制文件
1132
- await fs$1.copyFile(srcPath, destPath);
1133
- logger.verbose(' ✓ Copied');
1134
- }
1135
- }),
1438
+ files.map(file =>
1439
+ processSingleFile({
1440
+ file,
1441
+ templatePath,
1442
+ outputPath,
1443
+ context,
1444
+ templateConfig,
1445
+ }),
1446
+ ),
1136
1447
  );
1137
1448
 
1138
1449
  logger.verbose('✓ All files processed successfully');
@@ -1171,6 +1482,7 @@ const processTemplateFiles = async (
1171
1482
 
1172
1483
  /**
1173
1484
  * 检查输出目录是否为空
1485
+ * 注意:.git 目录会被忽略,允许在已初始化 git 的目录中创建项目
1174
1486
  *
1175
1487
  * @param outputPath - 输出目录路径
1176
1488
  * @returns 是否为空
@@ -1180,7 +1492,9 @@ const isOutputDirEmpty = async (
1180
1492
  ) => {
1181
1493
  try {
1182
1494
  const entries = await fs$1.readdir(outputPath);
1183
- return entries.length === 0;
1495
+ // 过滤掉 .git 目录,允许在已初始化 git 的目录中创建项目
1496
+ const filteredEntries = entries.filter(entry => entry !== '.git');
1497
+ return filteredEntries.length === 0;
1184
1498
  } catch (e) {
1185
1499
  // 目录不存在,视为空
1186
1500
  return true;
@@ -1316,7 +1630,12 @@ const execute = async (
1316
1630
  const absoluteOutputPath = await prepareOutputDirectory(outputPath);
1317
1631
 
1318
1632
  // 6. 处理模板文件
1319
- await processTemplateFiles(templatePath, absoluteOutputPath, context);
1633
+ await processTemplateFiles({
1634
+ templatePath,
1635
+ outputPath: absoluteOutputPath,
1636
+ context,
1637
+ templateConfig,
1638
+ });
1320
1639
 
1321
1640
  // 7. 执行 onAfterRender 钩子
1322
1641
  await executeAfterRenderHook(templateConfig, context, absoluteOutputPath);
@@ -1324,7 +1643,7 @@ const execute = async (
1324
1643
  return absoluteOutputPath;
1325
1644
  };
1326
1645
 
1327
- function _nullishCoalesce(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; }
1646
+ function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } }
1328
1647
  /**
1329
1648
  * 运行 pnpm install
1330
1649
  */
@@ -1364,8 +1683,18 @@ const runPnpmInstall = (projectPath) => {
1364
1683
 
1365
1684
  /**
1366
1685
  * 初始化 git 仓库并创建初始提交
1686
+ * 如果目录中已存在 .git,则跳过初始化
1367
1687
  */
1368
1688
  const runGitInit = (projectPath) => {
1689
+ // 检查是否已存在 .git 目录
1690
+ const gitDir = path.join(projectPath, '.git');
1691
+ if (fs.existsSync(gitDir)) {
1692
+ logger.info(
1693
+ '\n💡 Git repository already exists, skipping git initialization',
1694
+ );
1695
+ return;
1696
+ }
1697
+
1369
1698
  const runGitCommand = (command) => {
1370
1699
  logger.info(`Executing: ${command}`);
1371
1700
 
@@ -1417,45 +1746,33 @@ const runGitInit = (projectPath) => {
1417
1746
  };
1418
1747
 
1419
1748
  /**
1420
- * 运行开发服务器
1749
+ * 运行开发服务器(后台模式)
1750
+ * 启动后台子进程运行开发服务器,父进程可以直接退出
1751
+ * 使用 CLI 自己的 dev 命令(定义在 run.ts)而不是直接运行 npm run dev
1421
1752
  */
1422
- const runNpmDev = (projectPath) => {
1423
- logger.info('\nStarting development server...');
1424
- logger.info(`Executing: npm run dev in ${projectPath}`);
1425
- logger.info('Press Ctrl+C to stop the server\n');
1753
+ const runDev = (projectPath) => {
1754
+ logger.info('\nStarting development server in background...');
1426
1755
 
1427
- // 使用 async: true 异步执行,不阻塞进程
1428
- const child = shelljs.exec('npm run dev', {
1429
- cwd: projectPath,
1430
- async: true,
1431
- silent: true, // 手动处理输出以便显示详细信息
1432
- });
1433
-
1434
- if (child) {
1435
- // 输出 stdout
1436
- _optionalChain([child, 'access', _ => _.stdout, 'optionalAccess', _2 => _2.on, 'call', _3 => _3('data', (data) => {
1437
- process.stdout.write(data);
1438
- })]);
1756
+ // 获取当前 CLI 的可执行文件路径
1757
+ // process.argv[0] node,process.argv[1] CLI 入口文件
1758
+ const cliPath = process.argv[1];
1439
1759
 
1440
- // 输出 stderr
1441
- _optionalChain([child, 'access', _4 => _4.stderr, 'optionalAccess', _5 => _5.on, 'call', _6 => _6('data', (data) => {
1442
- process.stderr.write(data);
1443
- })]);
1760
+ logger.info(`Executing: ${cliPath} dev in ${projectPath}`);
1444
1761
 
1445
- // 监听错误
1446
- child.on('error', (error) => {
1447
- logger.error(`Failed to start dev server: ${error.message}`);
1448
- logger.error(`Error stack: ${error.stack}`);
1449
- });
1762
+ // 使用通用的后台执行函数启动开发服务器
1763
+ // 调用 CLI 自己的 dev 命令
1764
+ const pid = spawnDetached(process.argv[0], [cliPath, 'dev'], {
1765
+ cwd: projectPath,
1766
+ verbose: false, // 不输出额外的进程信息,由 logger 统一处理
1767
+ });
1450
1768
 
1451
- // 监听退出
1452
- child.on('exit', (code, signal) => {
1453
- if (code !== 0 && code !== null) {
1454
- logger.error(
1455
- `Dev server exited with code ${code}${signal ? ` and signal ${signal}` : ''}`,
1456
- );
1457
- }
1458
- });
1769
+ logger.success('Development server started in background!');
1770
+ if (pid) {
1771
+ logger.info(`Process ID: ${pid}`);
1772
+ logger.info(
1773
+ '\nThe dev server is running independently. You can close this terminal.',
1774
+ );
1775
+ logger.info(`To stop the server later, use: kill ${pid}`);
1459
1776
  }
1460
1777
  };
1461
1778
 
@@ -1520,7 +1837,7 @@ const executeInit = async (
1520
1837
 
1521
1838
  // 如果没有跳过 dev,则启动开发服务器
1522
1839
  if (!skipDev) {
1523
- runNpmDev(absoluteOutputPath);
1840
+ runDev(absoluteOutputPath);
1524
1841
  timer.logPhase('Dev server startup');
1525
1842
  } else {
1526
1843
  // 只有跳过 dev 时才显示 Next steps
@@ -1534,7 +1851,7 @@ const executeInit = async (
1534
1851
  ' git init && git add . && git commit -m "initial commit"',
1535
1852
  );
1536
1853
  }
1537
- logger.info(' npm run dev');
1854
+ logger.info(' coze dev');
1538
1855
  }
1539
1856
 
1540
1857
  // 输出总耗时
@@ -1568,10 +1885,14 @@ const registerCommand = program => {
1568
1885
  });
1569
1886
  };
1570
1887
 
1888
+ var version = "0.0.1-alpha.ff64d9";
1889
+ var packageJson = {
1890
+ version: version};
1891
+
1571
1892
  const commands = [
1572
1893
  registerCommand,
1573
1894
  registerCommand$1,
1574
- // registerWarmupCommand,
1895
+ registerCommand$2,
1575
1896
  ];
1576
1897
 
1577
1898
  const main = () => {
@@ -1582,7 +1903,7 @@ const main = () => {
1582
1903
  .description(
1583
1904
  'Coze Coding CLI - Project template engine for frontend stacks',
1584
1905
  )
1585
- .version('1.0.0');
1906
+ .version(packageJson.version);
1586
1907
 
1587
1908
  commands.forEach(initCmd => initCmd(program));
1588
1909