@coze-arch/cli 0.0.1-alpha.e8683e → 0.0.1-alpha.e89608
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -0
- package/lib/__templates__/expo/.cozeproj/scripts/dev_run.sh +25 -16
- package/lib/__templates__/expo/.cozeproj/scripts/server_dev_run.sh +9 -8
- package/lib/__templates__/expo/README.md +2 -2
- package/lib/__templates__/expo/client/app/+not-found.tsx +30 -0
- package/lib/__templates__/expo/client/app.config.ts +2 -2
- package/lib/__templates__/expo/client/components/Screen.tsx +2 -2
- package/lib/__templates__/expo/client/eslint.config.mjs +21 -1
- package/lib/__templates__/expo/client/hooks/useSafeRouter.ts +152 -0
- package/lib/__templates__/expo/client/metro.config.js +3 -0
- package/lib/__templates__/expo/client/package.json +36 -34
- package/lib/__templates__/expo/client/screens/demo/index.tsx +3 -3
- package/lib/__templates__/expo/client/scripts/install-missing-deps.js +10 -10
- package/lib/__templates__/expo/eslint-plugins/forbid-emoji/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/forbid-emoji/rule.js +112 -0
- package/lib/__templates__/expo/eslint-plugins/forbid-emoji/tech.md +94 -0
- package/lib/__templates__/expo/eslint-plugins/react-native/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/react-native/rule.js +64 -0
- package/lib/__templates__/expo/package.json +3 -0
- package/lib/__templates__/expo/patches/expo@54.0.33.patch +45 -0
- package/lib/__templates__/expo/pnpm-lock.yaml +1318 -2636
- package/lib/__templates__/expo/server/package.json +9 -7
- package/lib/__templates__/expo/server/src/index.ts +1 -0
- package/lib/__templates__/expo/template.config.js +56 -0
- package/lib/__templates__/native-static/.coze +11 -0
- package/lib/__templates__/native-static/index.html +33 -0
- package/lib/__templates__/native-static/styles/main.css +136 -0
- package/lib/__templates__/native-static/template.config.js +22 -0
- package/lib/__templates__/nextjs/README.md +5 -0
- package/lib/__templates__/nextjs/eslint.config.mjs +5 -0
- package/lib/__templates__/nextjs/next.config.ts +1 -2
- package/lib/__templates__/nextjs/package.json +5 -6
- package/lib/__templates__/nextjs/pnpm-lock.yaml +2096 -956
- package/lib/__templates__/nextjs/scripts/build.sh +4 -1
- package/lib/__templates__/nextjs/scripts/dev.sh +1 -2
- package/lib/__templates__/nextjs/scripts/start.sh +1 -1
- package/lib/__templates__/nextjs/src/app/layout.tsx +1 -1
- package/lib/__templates__/nextjs/src/app/page.tsx +18 -60
- package/lib/__templates__/nextjs/src/server.ts +35 -0
- package/lib/__templates__/nextjs/template.config.js +49 -14
- package/lib/__templates__/nextjs/tsconfig.json +1 -1
- package/lib/__templates__/nuxt-vue/.coze +12 -0
- package/lib/__templates__/nuxt-vue/README.md +73 -0
- package/lib/__templates__/nuxt-vue/_gitignore +24 -0
- package/lib/__templates__/nuxt-vue/_npmrc +23 -0
- package/lib/__templates__/nuxt-vue/app/app.vue +6 -0
- package/lib/__templates__/nuxt-vue/app/pages/index.vue +23 -0
- package/lib/__templates__/nuxt-vue/assets/css/main.css +24 -0
- package/lib/__templates__/nuxt-vue/nuxt.config.ts +116 -0
- package/lib/__templates__/nuxt-vue/package.json +35 -0
- package/lib/__templates__/nuxt-vue/pnpm-lock.yaml +8759 -0
- package/lib/__templates__/nuxt-vue/postcss.config.mjs +8 -0
- package/lib/__templates__/nuxt-vue/public/favicon.ico +0 -0
- package/lib/__templates__/nuxt-vue/public/robots.txt +2 -0
- package/lib/__templates__/nuxt-vue/scripts/build.sh +14 -0
- package/lib/__templates__/nuxt-vue/scripts/dev.sh +31 -0
- package/lib/__templates__/nuxt-vue/scripts/prepare.sh +14 -0
- package/lib/__templates__/nuxt-vue/scripts/start.sh +15 -0
- package/lib/__templates__/nuxt-vue/server/api/hello.ts +10 -0
- package/lib/__templates__/nuxt-vue/server/middleware/logger.ts +10 -0
- package/lib/__templates__/nuxt-vue/server/routes/health.ts +10 -0
- package/lib/__templates__/nuxt-vue/tailwind.config.js +13 -0
- package/lib/__templates__/nuxt-vue/template.config.js +87 -0
- package/lib/__templates__/nuxt-vue/tsconfig.json +18 -0
- package/lib/__templates__/taro/.coze +14 -0
- package/lib/__templates__/taro/.cozeproj/scripts/deploy_build.sh +19 -0
- package/lib/__templates__/taro/.cozeproj/scripts/deploy_run.sh +14 -0
- package/lib/__templates__/taro/.cozeproj/scripts/dev_build.sh +2 -0
- package/lib/__templates__/taro/.cozeproj/scripts/dev_run.sh +151 -0
- package/lib/__templates__/taro/.cozeproj/scripts/init_env.sh +5 -0
- package/lib/__templates__/taro/.cozeproj/scripts/pack.sh +24 -0
- package/lib/__templates__/taro/README.md +751 -0
- package/lib/__templates__/taro/_gitignore +40 -0
- package/lib/__templates__/taro/_npmrc +18 -0
- package/lib/__templates__/taro/babel.config.js +12 -0
- package/lib/__templates__/taro/config/dev.ts +9 -0
- package/lib/__templates__/taro/config/index.ts +223 -0
- package/lib/__templates__/taro/config/prod.ts +34 -0
- package/lib/__templates__/taro/eslint.config.mjs +80 -0
- package/lib/__templates__/taro/key/private.appid.key +0 -0
- package/lib/__templates__/taro/package.json +107 -0
- package/lib/__templates__/taro/patches/@tarojs__plugin-mini-ci@4.1.9.patch +30 -0
- package/lib/__templates__/taro/pnpm-lock.yaml +23100 -0
- package/lib/__templates__/taro/pnpm-workspace.yaml +2 -0
- package/lib/__templates__/taro/project.config.json +15 -0
- package/lib/__templates__/taro/server/nest-cli.json +10 -0
- package/lib/__templates__/taro/server/package.json +40 -0
- package/lib/__templates__/taro/server/src/app.controller.ts +23 -0
- package/lib/__templates__/taro/server/src/app.module.ts +10 -0
- package/lib/__templates__/taro/server/src/app.service.ts +8 -0
- package/lib/__templates__/taro/server/src/interceptors/http-status.interceptor.ts +23 -0
- package/lib/__templates__/taro/server/src/main.ts +49 -0
- package/lib/__templates__/taro/server/tsconfig.json +24 -0
- package/lib/__templates__/taro/src/app.config.ts +11 -0
- package/lib/__templates__/taro/src/app.css +52 -0
- package/lib/__templates__/taro/src/app.tsx +9 -0
- package/lib/__templates__/taro/src/index.html +39 -0
- package/lib/__templates__/taro/src/network.ts +39 -0
- package/lib/__templates__/taro/src/pages/index/index.config.ts +3 -0
- package/lib/__templates__/taro/src/pages/index/index.css +1 -0
- package/lib/__templates__/taro/src/pages/index/index.tsx +33 -0
- package/lib/__templates__/taro/src/presets/dev-debug.ts +23 -0
- package/lib/__templates__/taro/src/presets/h5-container.tsx +15 -0
- package/lib/__templates__/taro/src/presets/h5-navbar.tsx +201 -0
- package/lib/__templates__/taro/src/presets/h5-styles.ts +142 -0
- package/lib/__templates__/taro/src/presets/index.tsx +18 -0
- package/lib/__templates__/taro/stylelint.config.mjs +4 -0
- package/lib/__templates__/taro/template.config.js +68 -0
- package/lib/__templates__/taro/tsconfig.json +29 -0
- package/lib/__templates__/taro/types/global.d.ts +32 -0
- package/lib/__templates__/templates.json +75 -0
- package/lib/__templates__/vite/README.md +189 -11
- package/lib/__templates__/vite/_gitignore +1 -0
- package/lib/__templates__/vite/eslint.config.mjs +6 -1
- package/lib/__templates__/vite/package.json +20 -3
- package/lib/__templates__/vite/pnpm-lock.yaml +944 -1551
- package/lib/__templates__/vite/scripts/build.sh +4 -1
- package/lib/__templates__/vite/scripts/dev.sh +2 -2
- package/lib/__templates__/vite/scripts/start.sh +3 -3
- package/lib/__templates__/vite/server/index.ts +57 -0
- package/lib/__templates__/vite/server/routes/index.ts +31 -0
- package/lib/__templates__/vite/server/vite.ts +79 -0
- package/lib/__templates__/vite/src/main.ts +17 -47
- package/lib/__templates__/vite/template.config.js +49 -14
- package/lib/__templates__/vite/tsconfig.json +4 -3
- package/lib/__templates__/vite/vite.config.ts +1 -0
- package/lib/cli.js +651 -173
- package/package.json +7 -3
package/lib/cli.js
CHANGED
|
@@ -4,6 +4,8 @@
|
|
|
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');
|
|
@@ -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$
|
|
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
|
|
597
|
+
.action(async options => {
|
|
594
598
|
await executeWarmup(options);
|
|
595
599
|
});
|
|
596
600
|
};
|
|
@@ -1033,7 +1037,7 @@ const executeFix = async (options = {}) => {
|
|
|
1033
1037
|
/**
|
|
1034
1038
|
* 注册 fix 命令到 program
|
|
1035
1039
|
*/
|
|
1036
|
-
const registerCommand$
|
|
1040
|
+
const registerCommand$3 = program => {
|
|
1037
1041
|
program
|
|
1038
1042
|
.command('fix')
|
|
1039
1043
|
.description(
|
|
@@ -1052,13 +1056,13 @@ function _nullishCoalesce$1(lhs, rhsFn) { if (lhs != null) { return lhs; } else
|
|
|
1052
1056
|
/**
|
|
1053
1057
|
* 日志文件名常量
|
|
1054
1058
|
*/
|
|
1055
|
-
const LOG_FILE_NAME = 'dev.log';
|
|
1059
|
+
const LOG_FILE_NAME$1 = 'dev.log';
|
|
1056
1060
|
|
|
1057
1061
|
/**
|
|
1058
1062
|
* 获取日志目录
|
|
1059
1063
|
* 优先使用环境变量 COZE_LOG_DIR,否则使用 ~/.coze-logs
|
|
1060
1064
|
*/
|
|
1061
|
-
const getLogDir = () =>
|
|
1065
|
+
const getLogDir$1 = () =>
|
|
1062
1066
|
process.env.COZE_LOG_DIR || path.join(os.homedir(), '.coze-logs');
|
|
1063
1067
|
|
|
1064
1068
|
/**
|
|
@@ -1067,22 +1071,22 @@ const getLogDir = () =>
|
|
|
1067
1071
|
* - 如果是相对路径,基于 getLogDir() + 相对路径
|
|
1068
1072
|
* - 如果为空,使用 getLogDir() + LOG_FILE_NAME
|
|
1069
1073
|
*/
|
|
1070
|
-
const resolveLogFilePath = (logFile) => {
|
|
1074
|
+
const resolveLogFilePath$1 = (logFile) => {
|
|
1071
1075
|
if (!logFile) {
|
|
1072
|
-
return path.join(getLogDir(), LOG_FILE_NAME);
|
|
1076
|
+
return path.join(getLogDir$1(), LOG_FILE_NAME$1);
|
|
1073
1077
|
}
|
|
1074
1078
|
|
|
1075
1079
|
if (path.isAbsolute(logFile)) {
|
|
1076
1080
|
return logFile;
|
|
1077
1081
|
}
|
|
1078
1082
|
|
|
1079
|
-
return path.join(getLogDir(), logFile);
|
|
1083
|
+
return path.join(getLogDir$1(), logFile);
|
|
1080
1084
|
};
|
|
1081
1085
|
|
|
1082
1086
|
/**
|
|
1083
1087
|
* 创建日志写入流
|
|
1084
1088
|
*/
|
|
1085
|
-
const createLogStream = (logFilePath) => {
|
|
1089
|
+
const createLogStream$1 = (logFilePath) => {
|
|
1086
1090
|
const logDir = path.dirname(logFilePath);
|
|
1087
1091
|
|
|
1088
1092
|
// 确保日志目录存在
|
|
@@ -1105,7 +1109,7 @@ const executeRun = async (
|
|
|
1105
1109
|
logger.info(`Running ${commandName} command...`);
|
|
1106
1110
|
|
|
1107
1111
|
// 1. 对于 build 命令,先执行 fix 以确保项目配置正确
|
|
1108
|
-
if (
|
|
1112
|
+
if (['dev', 'build'].includes(commandName)) {
|
|
1109
1113
|
logger.info('\n🔧 Running fix command before build...\n');
|
|
1110
1114
|
try {
|
|
1111
1115
|
await executeFix();
|
|
@@ -1121,8 +1125,8 @@ const executeRun = async (
|
|
|
1121
1125
|
const commandArgs = getCommandConfig(config, commandName);
|
|
1122
1126
|
|
|
1123
1127
|
// 3. 准备日志
|
|
1124
|
-
const logFilePath = resolveLogFilePath(options.logFile);
|
|
1125
|
-
const logStream = createLogStream(logFilePath);
|
|
1128
|
+
const logFilePath = resolveLogFilePath$1(options.logFile);
|
|
1129
|
+
const logStream = createLogStream$1(logFilePath);
|
|
1126
1130
|
|
|
1127
1131
|
// 4. 执行命令
|
|
1128
1132
|
const commandString = commandArgs.join(' ');
|
|
@@ -1185,7 +1189,7 @@ const executeRun = async (
|
|
|
1185
1189
|
/**
|
|
1186
1190
|
* 注册 dev/build/start 命令到 program
|
|
1187
1191
|
*/
|
|
1188
|
-
const registerCommand$
|
|
1192
|
+
const registerCommand$2 = program => {
|
|
1189
1193
|
// dev 命令
|
|
1190
1194
|
program
|
|
1191
1195
|
.command('dev')
|
|
@@ -1437,6 +1441,11 @@ const shouldIgnoreFile = (filePath) => {
|
|
|
1437
1441
|
return directoryPatterns.some(dir => pathParts.includes(dir));
|
|
1438
1442
|
};
|
|
1439
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
|
|
1440
1449
|
/**
|
|
1441
1450
|
* 递归获取目录中的所有文件
|
|
1442
1451
|
*
|
|
@@ -1506,7 +1515,19 @@ const convertDotfileName = (filePath) => {
|
|
|
1506
1515
|
|
|
1507
1516
|
return filePath;
|
|
1508
1517
|
};
|
|
1518
|
+
// end_aigc
|
|
1519
|
+
|
|
1520
|
+
// ABOUTME: File rendering utilities for template processing
|
|
1521
|
+
// ABOUTME: Handles file content rendering, hook execution, and file writing
|
|
1522
|
+
|
|
1523
|
+
|
|
1524
|
+
|
|
1509
1525
|
|
|
1526
|
+
|
|
1527
|
+
|
|
1528
|
+
|
|
1529
|
+
|
|
1530
|
+
// start_aigc
|
|
1510
1531
|
/**
|
|
1511
1532
|
* 执行文件渲染钩子
|
|
1512
1533
|
*
|
|
@@ -1552,22 +1573,25 @@ const executeFileRenderHook = async (
|
|
|
1552
1573
|
};
|
|
1553
1574
|
|
|
1554
1575
|
/**
|
|
1555
|
-
*
|
|
1576
|
+
* 准备单个文件的渲染信息(不实际写入)
|
|
1577
|
+
* 用于 dry-run 阶段,收集所有将要写入的文件信息
|
|
1578
|
+
*
|
|
1579
|
+
* @param options - 准备选项
|
|
1580
|
+
* @returns 文件渲染信息,或 null 表示该文件被跳过
|
|
1556
1581
|
*/
|
|
1557
|
-
const
|
|
1558
|
-
|
|
1582
|
+
const prepareFileInfo = async (options
|
|
1559
1583
|
|
|
1560
1584
|
|
|
1561
1585
|
|
|
1562
1586
|
|
|
1563
1587
|
) => {
|
|
1564
|
-
const { file, templatePath,
|
|
1588
|
+
const { file, templatePath, context, templateConfig } = options;
|
|
1565
1589
|
|
|
1566
1590
|
const srcPath = path.join(templatePath, file);
|
|
1567
1591
|
const destFile = convertDotfileName(file);
|
|
1568
1592
|
|
|
1569
1593
|
logger.verbose(
|
|
1570
|
-
` -
|
|
1594
|
+
` - Preparing: ${file}${destFile !== file ? ` -> ${destFile}` : ''}`,
|
|
1571
1595
|
);
|
|
1572
1596
|
|
|
1573
1597
|
// 判断是否为二进制文件
|
|
@@ -1601,51 +1625,177 @@ const processSingleFile = async (options
|
|
|
1601
1625
|
context,
|
|
1602
1626
|
);
|
|
1603
1627
|
|
|
1604
|
-
// 如果返回 null
|
|
1628
|
+
// 如果返回 null,表示该文件被 hook 跳过
|
|
1605
1629
|
if (processedFileInfo === null) {
|
|
1606
1630
|
logger.verbose(' ⊘ Skipped by onFileRender hook');
|
|
1607
|
-
return;
|
|
1631
|
+
return null;
|
|
1608
1632
|
}
|
|
1609
1633
|
|
|
1610
|
-
|
|
1611
|
-
|
|
1634
|
+
return processedFileInfo;
|
|
1635
|
+
};
|
|
1636
|
+
|
|
1637
|
+
/**
|
|
1638
|
+
* 写入渲染后的文件到目标路径
|
|
1639
|
+
*
|
|
1640
|
+
* @param options - 写入选项
|
|
1641
|
+
*/
|
|
1642
|
+
const writeRenderedFile = async (options
|
|
1643
|
+
|
|
1644
|
+
|
|
1645
|
+
|
|
1646
|
+
) => {
|
|
1647
|
+
const { fileInfo, srcPath, destPath } = options;
|
|
1648
|
+
|
|
1649
|
+
logger.verbose(` - Writing: ${fileInfo.destPath}`);
|
|
1612
1650
|
|
|
1613
1651
|
// 确保目标目录存在
|
|
1614
|
-
await ensureDir(path.dirname(
|
|
1652
|
+
await ensureDir(path.dirname(destPath));
|
|
1615
1653
|
|
|
1616
1654
|
// 写入文件
|
|
1617
|
-
if (
|
|
1618
|
-
//
|
|
1619
|
-
|
|
1620
|
-
|
|
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);
|
|
1621
1662
|
logger.verbose(' ✓ Copied (binary)');
|
|
1622
1663
|
} else {
|
|
1623
|
-
const
|
|
1624
|
-
await fs$1.writeFile(
|
|
1664
|
+
const modifiedBuffer = Buffer.from(fileInfo.content, 'base64');
|
|
1665
|
+
await fs$1.writeFile(destPath, modifiedBuffer);
|
|
1625
1666
|
logger.verbose(' ✓ Written (binary, modified by hook)');
|
|
1626
1667
|
}
|
|
1627
1668
|
} else {
|
|
1628
1669
|
// 文本文件
|
|
1629
|
-
await fs$1.writeFile(
|
|
1670
|
+
await fs$1.writeFile(destPath, fileInfo.content, 'utf-8');
|
|
1630
1671
|
logger.verbose(' ✓ Rendered and written');
|
|
1631
1672
|
}
|
|
1632
1673
|
};
|
|
1674
|
+
// end_aigc
|
|
1675
|
+
|
|
1676
|
+
// ABOUTME: File conflict detection utilities for template processing
|
|
1677
|
+
// ABOUTME: Provides dry-run file collection and conflict checking
|
|
1678
|
+
|
|
1679
|
+
|
|
1680
|
+
|
|
1681
|
+
|
|
1682
|
+
|
|
1683
|
+
|
|
1684
|
+
|
|
1685
|
+
// start_aigc
|
|
1686
|
+
/**
|
|
1687
|
+
* 收集所有将要写入的文件路径(dry-run 阶段)
|
|
1688
|
+
*
|
|
1689
|
+
* @param options - 收集选项
|
|
1690
|
+
* @returns 将要写入的文件路径列表
|
|
1691
|
+
*/
|
|
1692
|
+
const collectFilesToRender = async (options
|
|
1693
|
+
|
|
1694
|
+
|
|
1695
|
+
|
|
1696
|
+
|
|
1697
|
+
) => {
|
|
1698
|
+
const { files, templatePath, context, templateConfig } = options;
|
|
1699
|
+
|
|
1700
|
+
logger.verbose('\nDry-run: Collecting files to render...');
|
|
1701
|
+
|
|
1702
|
+
const fileInfos = await Promise.all(
|
|
1703
|
+
files.map(file =>
|
|
1704
|
+
prepareFileInfo({
|
|
1705
|
+
file,
|
|
1706
|
+
templatePath,
|
|
1707
|
+
context,
|
|
1708
|
+
templateConfig,
|
|
1709
|
+
}),
|
|
1710
|
+
),
|
|
1711
|
+
);
|
|
1712
|
+
|
|
1713
|
+
// 过滤掉被 hook 跳过的文件,收集 destPath
|
|
1714
|
+
const filesToWrite = fileInfos
|
|
1715
|
+
.filter((info) => info !== null)
|
|
1716
|
+
.map(info => info.destPath);
|
|
1717
|
+
|
|
1718
|
+
logger.verbose(` - ${filesToWrite.length} files will be written`);
|
|
1719
|
+
logger.verbose(
|
|
1720
|
+
` - ${fileInfos.length - filesToWrite.length} files skipped by hooks`,
|
|
1721
|
+
);
|
|
1722
|
+
|
|
1723
|
+
return filesToWrite;
|
|
1724
|
+
};
|
|
1725
|
+
// end_aigc
|
|
1726
|
+
|
|
1727
|
+
// ABOUTME: Main file processing orchestration for template rendering
|
|
1728
|
+
// ABOUTME: Coordinates file system, rendering, and conflict detection layers
|
|
1729
|
+
|
|
1730
|
+
|
|
1731
|
+
|
|
1732
|
+
// start_aigc
|
|
1733
|
+
/**
|
|
1734
|
+
* 处理单个文件(准备 + 写入)
|
|
1735
|
+
*
|
|
1736
|
+
* @param options - 处理选项
|
|
1737
|
+
*/
|
|
1738
|
+
const processSingleFile = async (options
|
|
1739
|
+
|
|
1740
|
+
|
|
1741
|
+
|
|
1742
|
+
|
|
1743
|
+
|
|
1744
|
+
) => {
|
|
1745
|
+
const { file, templatePath, outputPath, context, templateConfig } = options;
|
|
1746
|
+
|
|
1747
|
+
const srcPath = path.join(templatePath, file);
|
|
1748
|
+
|
|
1749
|
+
// 准备文件信息
|
|
1750
|
+
const processedFileInfo = await prepareFileInfo({
|
|
1751
|
+
file,
|
|
1752
|
+
templatePath,
|
|
1753
|
+
context,
|
|
1754
|
+
templateConfig,
|
|
1755
|
+
});
|
|
1756
|
+
|
|
1757
|
+
// 如果返回 null,跳过该文件
|
|
1758
|
+
if (processedFileInfo === null) {
|
|
1759
|
+
return;
|
|
1760
|
+
}
|
|
1761
|
+
|
|
1762
|
+
// 使用处理后的目标路径
|
|
1763
|
+
const finalDestPath = path.join(outputPath, processedFileInfo.destPath);
|
|
1764
|
+
|
|
1765
|
+
// 写入文件
|
|
1766
|
+
await writeRenderedFile({
|
|
1767
|
+
fileInfo: processedFileInfo,
|
|
1768
|
+
srcPath,
|
|
1769
|
+
destPath: finalDestPath,
|
|
1770
|
+
});
|
|
1771
|
+
};
|
|
1633
1772
|
|
|
1634
1773
|
/**
|
|
1635
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 - 处理选项
|
|
1636
1785
|
*/
|
|
1637
1786
|
const processTemplateFiles = async (options
|
|
1638
1787
|
|
|
1639
1788
|
|
|
1640
1789
|
|
|
1641
1790
|
|
|
1791
|
+
|
|
1642
1792
|
) => {
|
|
1643
|
-
const { templatePath, outputPath, context, templateConfig
|
|
1793
|
+
const { templatePath, outputPath, context, templateConfig} = options;
|
|
1644
1794
|
logger.verbose('Processing template files:');
|
|
1645
1795
|
logger.verbose(` - Template path: ${templatePath}`);
|
|
1646
1796
|
logger.verbose(` - Output path: ${outputPath}`);
|
|
1647
1797
|
|
|
1648
|
-
// 验证模板目录是否存在
|
|
1798
|
+
// 阶段 0: 验证模板目录是否存在
|
|
1649
1799
|
try {
|
|
1650
1800
|
const stat = await fs$1.stat(templatePath);
|
|
1651
1801
|
logger.verbose(
|
|
@@ -1661,6 +1811,7 @@ const processTemplateFiles = async (options
|
|
|
1661
1811
|
throw error;
|
|
1662
1812
|
}
|
|
1663
1813
|
|
|
1814
|
+
// 阶段 1: 扫描所有模板文件
|
|
1664
1815
|
const files = await getAllFiles(templatePath);
|
|
1665
1816
|
|
|
1666
1817
|
logger.verbose(` - Found ${files.length} files to process`);
|
|
@@ -1670,6 +1821,23 @@ const processTemplateFiles = async (options
|
|
|
1670
1821
|
return;
|
|
1671
1822
|
}
|
|
1672
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...');
|
|
1673
1841
|
await Promise.all(
|
|
1674
1842
|
files.map(file =>
|
|
1675
1843
|
processSingleFile({
|
|
@@ -1684,74 +1852,9 @@ const processTemplateFiles = async (options
|
|
|
1684
1852
|
|
|
1685
1853
|
logger.verbose('✓ All files processed successfully');
|
|
1686
1854
|
|
|
1687
|
-
//
|
|
1688
|
-
const sourceNodeModules = path.join(templatePath, 'node_modules');
|
|
1689
|
-
const targetNodeModules = path.join(outputPath, 'node_modules');
|
|
1690
|
-
|
|
1691
|
-
if (fs.existsSync(sourceNodeModules)) {
|
|
1692
|
-
logger.info('\nCopying node_modules from pre-warmed template...');
|
|
1693
|
-
logger.verbose(` From: ${sourceNodeModules}`);
|
|
1694
|
-
logger.verbose(` To: ${targetNodeModules}`);
|
|
1695
|
-
|
|
1696
|
-
const result = shelljs.exec(`cp -R "${sourceNodeModules}" "${targetNodeModules}"`, {
|
|
1697
|
-
silent: true,
|
|
1698
|
-
});
|
|
1699
|
-
|
|
1700
|
-
if (result.stdout) {
|
|
1701
|
-
process.stdout.write(result.stdout);
|
|
1702
|
-
}
|
|
1703
|
-
|
|
1704
|
-
if (result.stderr) {
|
|
1705
|
-
process.stderr.write(result.stderr);
|
|
1706
|
-
}
|
|
1707
|
-
|
|
1708
|
-
if (result.code === 0) {
|
|
1709
|
-
logger.success('✓ node_modules copied successfully');
|
|
1710
|
-
} else {
|
|
1711
|
-
logger.warn(
|
|
1712
|
-
`Failed to copy node_modules: ${result.stderr || 'unknown error'}`,
|
|
1713
|
-
);
|
|
1714
|
-
logger.info('Will need to run pnpm install manually');
|
|
1715
|
-
}
|
|
1716
|
-
}
|
|
1717
|
-
};
|
|
1718
|
-
|
|
1719
|
-
/**
|
|
1720
|
-
* 检查输出目录是否为空
|
|
1721
|
-
* 注意:.git 目录会被忽略,允许在已初始化 git 的目录中创建项目
|
|
1722
|
-
*
|
|
1723
|
-
* @param outputPath - 输出目录路径
|
|
1724
|
-
* @returns 是否为空
|
|
1725
|
-
*/
|
|
1726
|
-
const isOutputDirEmpty = async (
|
|
1727
|
-
outputPath,
|
|
1728
|
-
) => {
|
|
1729
|
-
try {
|
|
1730
|
-
const entries = await fs$1.readdir(outputPath);
|
|
1731
|
-
// 过滤掉 .git 目录,允许在已初始化 git 的目录中创建项目
|
|
1732
|
-
const filteredEntries = entries.filter(entry => entry !== '.git');
|
|
1733
|
-
return filteredEntries.length === 0;
|
|
1734
|
-
} catch (e) {
|
|
1735
|
-
// 目录不存在,视为空
|
|
1736
|
-
return true;
|
|
1737
|
-
}
|
|
1738
|
-
};
|
|
1739
|
-
|
|
1740
|
-
/**
|
|
1741
|
-
* 验证输出目录
|
|
1742
|
-
*
|
|
1743
|
-
* @param outputPath - 输出目录路径
|
|
1744
|
-
* @throws 如果目录不为空则抛出错误
|
|
1745
|
-
*/
|
|
1746
|
-
const validateOutputDir = async (outputPath) => {
|
|
1747
|
-
const isEmpty = await isOutputDirEmpty(outputPath);
|
|
1748
|
-
if (!isEmpty) {
|
|
1749
|
-
throw new Error(
|
|
1750
|
-
`Output directory is not empty: ${outputPath}\n` +
|
|
1751
|
-
'Please use an empty directory or remove existing files.',
|
|
1752
|
-
);
|
|
1753
|
-
}
|
|
1855
|
+
// node_modules 将由 pnpm install 处理(利用缓存和硬链接机制)
|
|
1754
1856
|
};
|
|
1857
|
+
// end_aigc
|
|
1755
1858
|
|
|
1756
1859
|
/**
|
|
1757
1860
|
* 模板引擎执行选项
|
|
@@ -1762,6 +1865,7 @@ const validateOutputDir = async (outputPath) => {
|
|
|
1762
1865
|
|
|
1763
1866
|
|
|
1764
1867
|
|
|
1868
|
+
|
|
1765
1869
|
/**
|
|
1766
1870
|
* 加载模板元数据和路径
|
|
1767
1871
|
*/
|
|
@@ -1831,22 +1935,47 @@ const executeAfterRenderHook = async (
|
|
|
1831
1935
|
}
|
|
1832
1936
|
};
|
|
1833
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
|
+
|
|
1834
1951
|
/**
|
|
1835
1952
|
* 准备输出目录
|
|
1836
1953
|
*/
|
|
1837
|
-
const prepareOutputDirectory =
|
|
1954
|
+
const prepareOutputDirectory = (outputPath) => {
|
|
1838
1955
|
const absolutePath = path.resolve(process.cwd(), outputPath);
|
|
1839
|
-
|
|
1956
|
+
// 不再在这里验证目录是否为空,冲突检测已移至 processTemplateFiles 中
|
|
1840
1957
|
return absolutePath;
|
|
1841
1958
|
};
|
|
1842
1959
|
|
|
1960
|
+
/**
|
|
1961
|
+
* 模板引擎执行结果
|
|
1962
|
+
*/
|
|
1963
|
+
|
|
1964
|
+
|
|
1965
|
+
|
|
1966
|
+
|
|
1967
|
+
|
|
1968
|
+
|
|
1969
|
+
|
|
1970
|
+
|
|
1971
|
+
|
|
1843
1972
|
/**
|
|
1844
1973
|
* 执行完整的模板渲染流程
|
|
1845
1974
|
*/
|
|
1846
1975
|
const execute = async (
|
|
1847
1976
|
options,
|
|
1848
1977
|
) => {
|
|
1849
|
-
const { templateName, outputPath, command
|
|
1978
|
+
const { templateName, outputPath, command} = options;
|
|
1850
1979
|
|
|
1851
1980
|
// 1. 加载模板
|
|
1852
1981
|
const { templatePath } = await loadTemplateMetadata(templateName);
|
|
@@ -1863,20 +1992,23 @@ const execute = async (
|
|
|
1863
1992
|
});
|
|
1864
1993
|
|
|
1865
1994
|
// 5. 准备输出目录
|
|
1866
|
-
const absoluteOutputPath =
|
|
1995
|
+
const absoluteOutputPath = prepareOutputDirectory(outputPath);
|
|
1867
1996
|
|
|
1868
1997
|
// 6. 处理模板文件
|
|
1869
1998
|
await processTemplateFiles({
|
|
1870
1999
|
templatePath,
|
|
1871
2000
|
outputPath: absoluteOutputPath,
|
|
1872
2001
|
context,
|
|
1873
|
-
templateConfig
|
|
1874
|
-
});
|
|
2002
|
+
templateConfig});
|
|
1875
2003
|
|
|
1876
2004
|
// 7. 执行 onAfterRender 钩子
|
|
1877
2005
|
await executeAfterRenderHook(templateConfig, context, absoluteOutputPath);
|
|
1878
2006
|
|
|
1879
|
-
return
|
|
2007
|
+
return {
|
|
2008
|
+
outputPath: absoluteOutputPath,
|
|
2009
|
+
templateConfig,
|
|
2010
|
+
context,
|
|
2011
|
+
};
|
|
1880
2012
|
};
|
|
1881
2013
|
|
|
1882
2014
|
function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } }
|
|
@@ -1918,7 +2050,40 @@ const runPnpmInstall = (projectPath) => {
|
|
|
1918
2050
|
};
|
|
1919
2051
|
|
|
1920
2052
|
/**
|
|
1921
|
-
*
|
|
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 仓库
|
|
1922
2087
|
* 如果目录中已存在 .git,则跳过初始化
|
|
1923
2088
|
*/
|
|
1924
2089
|
const runGitInit = (projectPath) => {
|
|
@@ -1931,46 +2096,9 @@ const runGitInit = (projectPath) => {
|
|
|
1931
2096
|
return;
|
|
1932
2097
|
}
|
|
1933
2098
|
|
|
1934
|
-
const runGitCommand = (command) => {
|
|
1935
|
-
logger.info(`Executing: ${command}`);
|
|
1936
|
-
|
|
1937
|
-
const result = shelljs.exec(command, {
|
|
1938
|
-
cwd: projectPath,
|
|
1939
|
-
silent: true,
|
|
1940
|
-
});
|
|
1941
|
-
|
|
1942
|
-
// 输出命令的结果
|
|
1943
|
-
if (result.stdout) {
|
|
1944
|
-
process.stdout.write(result.stdout);
|
|
1945
|
-
}
|
|
1946
|
-
|
|
1947
|
-
if (result.stderr) {
|
|
1948
|
-
process.stderr.write(result.stderr);
|
|
1949
|
-
}
|
|
1950
|
-
|
|
1951
|
-
if (result.code !== 0) {
|
|
1952
|
-
const errorMessage = [
|
|
1953
|
-
`${command} failed with exit code ${result.code}`,
|
|
1954
|
-
result.stderr ? `\nStderr:\n${result.stderr}` : '',
|
|
1955
|
-
result.stdout ? `\nStdout:\n${result.stdout}` : '',
|
|
1956
|
-
]
|
|
1957
|
-
.filter(Boolean)
|
|
1958
|
-
.join('');
|
|
1959
|
-
|
|
1960
|
-
throw new Error(errorMessage);
|
|
1961
|
-
}
|
|
1962
|
-
};
|
|
1963
|
-
|
|
1964
2099
|
try {
|
|
1965
2100
|
logger.info('\nInitializing git repository...');
|
|
1966
|
-
runGitCommand('git init');
|
|
1967
|
-
|
|
1968
|
-
logger.info('Adding files to git...');
|
|
1969
|
-
runGitCommand('git add .');
|
|
1970
|
-
|
|
1971
|
-
logger.info('Creating initial commit...');
|
|
1972
|
-
runGitCommand('git commit -m "chore: initial commit"');
|
|
1973
|
-
|
|
2101
|
+
runGitCommand('git init', projectPath);
|
|
1974
2102
|
logger.success('Git repository initialized successfully!');
|
|
1975
2103
|
} catch (error) {
|
|
1976
2104
|
// Git 初始化失败不应该导致整个流程失败
|
|
@@ -1981,6 +2109,35 @@ const runGitInit = (projectPath) => {
|
|
|
1981
2109
|
}
|
|
1982
2110
|
};
|
|
1983
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
|
+
|
|
1984
2141
|
/**
|
|
1985
2142
|
* 运行开发服务器(后台模式)
|
|
1986
2143
|
* 启动后台子进程运行开发服务器,父进程可以直接退出
|
|
@@ -2022,6 +2179,8 @@ const executeInit = async (
|
|
|
2022
2179
|
|
|
2023
2180
|
|
|
2024
2181
|
|
|
2182
|
+
|
|
2183
|
+
|
|
2025
2184
|
,
|
|
2026
2185
|
command,
|
|
2027
2186
|
) => {
|
|
@@ -2033,44 +2192,58 @@ const executeInit = async (
|
|
|
2033
2192
|
output: outputPath,
|
|
2034
2193
|
skipInstall,
|
|
2035
2194
|
skipGit,
|
|
2195
|
+
skipCommit,
|
|
2036
2196
|
skipDev,
|
|
2197
|
+
force,
|
|
2037
2198
|
} = options;
|
|
2038
2199
|
|
|
2039
2200
|
logger.info(`Initializing project with template: ${templateName}`);
|
|
2040
2201
|
timer.logPhase('Initialization');
|
|
2041
2202
|
|
|
2042
|
-
//
|
|
2043
|
-
const
|
|
2203
|
+
// 执行模板引擎,返回结果对象
|
|
2204
|
+
const result = await execute({
|
|
2044
2205
|
templateName,
|
|
2045
2206
|
outputPath,
|
|
2046
2207
|
command,
|
|
2208
|
+
force,
|
|
2047
2209
|
});
|
|
2210
|
+
const { outputPath: absoluteOutputPath, templateConfig, context } = result;
|
|
2048
2211
|
|
|
2049
2212
|
timer.logPhase('Template engine execution');
|
|
2050
2213
|
logger.success('Project created successfully!');
|
|
2051
2214
|
|
|
2052
|
-
//
|
|
2053
|
-
|
|
2054
|
-
|
|
2055
|
-
const hasNodeModules = fs.existsSync(nodeModulesPath);
|
|
2215
|
+
// 检查是否存在 package.json
|
|
2216
|
+
const packageJsonPath = path.join(absoluteOutputPath, 'package.json');
|
|
2217
|
+
const hasPackageJson = fs.existsSync(packageJsonPath);
|
|
2056
2218
|
|
|
2057
|
-
|
|
2058
|
-
|
|
2059
|
-
|
|
2060
|
-
);
|
|
2061
|
-
timer.logPhase('Node modules (pre-warmed)');
|
|
2062
|
-
} else {
|
|
2219
|
+
// 安装依赖(始终使用 pnpm install,利用缓存机制)
|
|
2220
|
+
if (!skipInstall) {
|
|
2221
|
+
if (hasPackageJson) {
|
|
2063
2222
|
runPnpmInstall(absoluteOutputPath);
|
|
2064
2223
|
timer.logPhase('Dependencies installation');
|
|
2224
|
+
} else {
|
|
2225
|
+
logger.info(
|
|
2226
|
+
'\n💡 No package.json found, skipping dependency installation',
|
|
2227
|
+
);
|
|
2065
2228
|
}
|
|
2066
2229
|
}
|
|
2067
2230
|
|
|
2231
|
+
// 执行 onComplete 钩子(在 pnpm install 之后)
|
|
2232
|
+
await executeCompleteHook(templateConfig, context, absoluteOutputPath);
|
|
2233
|
+
timer.logPhase('Complete hook execution');
|
|
2234
|
+
|
|
2068
2235
|
// 如果没有跳过 git,则初始化 git 仓库
|
|
2069
2236
|
if (!skipGit) {
|
|
2070
2237
|
runGitInit(absoluteOutputPath);
|
|
2071
2238
|
timer.logPhase('Git initialization');
|
|
2072
2239
|
}
|
|
2073
2240
|
|
|
2241
|
+
// 如果没有跳过 commit,则提交初始化生成的文件
|
|
2242
|
+
if (!skipCommit) {
|
|
2243
|
+
commitChanges(absoluteOutputPath);
|
|
2244
|
+
timer.logPhase('Git commit');
|
|
2245
|
+
}
|
|
2246
|
+
|
|
2074
2247
|
// 如果没有跳过 dev,则启动开发服务器
|
|
2075
2248
|
if (!skipDev) {
|
|
2076
2249
|
runDev(absoluteOutputPath);
|
|
@@ -2079,13 +2252,14 @@ const executeInit = async (
|
|
|
2079
2252
|
// 只有跳过 dev 时才显示 Next steps
|
|
2080
2253
|
logger.info('\nNext steps:');
|
|
2081
2254
|
logger.info(` cd ${outputPath}`);
|
|
2082
|
-
if (skipInstall) {
|
|
2255
|
+
if (skipInstall && hasPackageJson) {
|
|
2083
2256
|
logger.info(' pnpm install');
|
|
2084
2257
|
}
|
|
2085
2258
|
if (skipGit) {
|
|
2086
|
-
logger.info(
|
|
2087
|
-
|
|
2088
|
-
|
|
2259
|
+
logger.info(' git init');
|
|
2260
|
+
}
|
|
2261
|
+
if (skipCommit) {
|
|
2262
|
+
logger.info(' git add --all && git commit -m "chore: init env"');
|
|
2089
2263
|
}
|
|
2090
2264
|
logger.info(' coze dev');
|
|
2091
2265
|
}
|
|
@@ -2103,7 +2277,7 @@ const executeInit = async (
|
|
|
2103
2277
|
/**
|
|
2104
2278
|
* 注册 init 命令到 program
|
|
2105
2279
|
*/
|
|
2106
|
-
const registerCommand = program => {
|
|
2280
|
+
const registerCommand$1 = program => {
|
|
2107
2281
|
program
|
|
2108
2282
|
.command('init')
|
|
2109
2283
|
.description('Initialize a new project from a template')
|
|
@@ -2112,24 +2286,328 @@ const registerCommand = program => {
|
|
|
2112
2286
|
.option('-o, --output <path>', 'Output directory', process.cwd())
|
|
2113
2287
|
.option('--skip-install', 'Skip automatic pnpm install', false)
|
|
2114
2288
|
.option('--skip-git', 'Skip automatic git initialization', false)
|
|
2289
|
+
.option(
|
|
2290
|
+
'--skip-commit',
|
|
2291
|
+
'Skip automatic git commit after initialization',
|
|
2292
|
+
false,
|
|
2293
|
+
)
|
|
2115
2294
|
.option('--skip-dev', 'Skip automatic dev server start', false)
|
|
2116
2295
|
.allowUnknownOption() // 允许透传参数
|
|
2117
2296
|
.action(async (directory, options, command) => {
|
|
2118
2297
|
// 位置参数优先级高于 --output 选项
|
|
2119
2298
|
const outputPath = _nullishCoalesce(directory, () => ( options.output));
|
|
2120
|
-
|
|
2299
|
+
// Always use force mode - overwrite existing files without conflict check
|
|
2300
|
+
const force = true;
|
|
2301
|
+
await executeInit({ ...options, output: outputPath, force }, command);
|
|
2302
|
+
});
|
|
2303
|
+
};
|
|
2304
|
+
|
|
2305
|
+
// ABOUTME: This file implements the update command for coze CLI
|
|
2306
|
+
// ABOUTME: It wraps pnpm update/install to update package dependencies with logging support
|
|
2307
|
+
|
|
2308
|
+
|
|
2309
|
+
|
|
2310
|
+
|
|
2311
|
+
/**
|
|
2312
|
+
* 日志文件名常量
|
|
2313
|
+
*/
|
|
2314
|
+
const LOG_FILE_NAME = 'update.log';
|
|
2315
|
+
|
|
2316
|
+
/**
|
|
2317
|
+
* 获取日志目录
|
|
2318
|
+
* 优先使用环境变量 COZE_LOG_DIR,否则使用 ~/.coze-logs
|
|
2319
|
+
*/
|
|
2320
|
+
const getLogDir = () =>
|
|
2321
|
+
process.env.COZE_LOG_DIR || path.join(os.homedir(), '.coze-logs');
|
|
2322
|
+
|
|
2323
|
+
/**
|
|
2324
|
+
* 解析日志文件路径
|
|
2325
|
+
* - 如果是绝对路径,直接使用
|
|
2326
|
+
* - 如果是相对路径,基于 getLogDir() + 相对路径
|
|
2327
|
+
* - 如果为空,使用 getLogDir() + LOG_FILE_NAME
|
|
2328
|
+
*/
|
|
2329
|
+
const resolveLogFilePath = (logFile) => {
|
|
2330
|
+
if (!logFile) {
|
|
2331
|
+
return path.join(getLogDir(), LOG_FILE_NAME);
|
|
2332
|
+
}
|
|
2333
|
+
|
|
2334
|
+
if (path.isAbsolute(logFile)) {
|
|
2335
|
+
return logFile;
|
|
2336
|
+
}
|
|
2337
|
+
|
|
2338
|
+
return path.join(getLogDir(), logFile);
|
|
2339
|
+
};
|
|
2340
|
+
|
|
2341
|
+
/**
|
|
2342
|
+
* 创建日志写入流
|
|
2343
|
+
*/
|
|
2344
|
+
const createLogStream = (logFilePath) => {
|
|
2345
|
+
const logDir = path.dirname(logFilePath);
|
|
2346
|
+
|
|
2347
|
+
// 确保日志目录存在
|
|
2348
|
+
if (!fs.existsSync(logDir)) {
|
|
2349
|
+
fs.mkdirSync(logDir, { recursive: true });
|
|
2350
|
+
}
|
|
2351
|
+
|
|
2352
|
+
// 使用 'w' 标志覆盖之前的日志
|
|
2353
|
+
return fs.createWriteStream(logFilePath, { flags: 'w' });
|
|
2354
|
+
};
|
|
2355
|
+
|
|
2356
|
+
/**
|
|
2357
|
+
* 格式化时间戳
|
|
2358
|
+
*/
|
|
2359
|
+
const formatTimestamp = () => {
|
|
2360
|
+
const now = new Date();
|
|
2361
|
+
return now.toISOString();
|
|
2362
|
+
};
|
|
2363
|
+
|
|
2364
|
+
/**
|
|
2365
|
+
* 写入带时间戳的日志
|
|
2366
|
+
*/
|
|
2367
|
+
const writeLogWithTimestamp = (stream, message) => {
|
|
2368
|
+
const timestamp = formatTimestamp();
|
|
2369
|
+
const lines = message.split('\n');
|
|
2370
|
+
lines.forEach(line => {
|
|
2371
|
+
if (line) {
|
|
2372
|
+
stream.write(`[${timestamp}] ${line}\n`);
|
|
2373
|
+
} else {
|
|
2374
|
+
stream.write('\n');
|
|
2375
|
+
}
|
|
2376
|
+
});
|
|
2377
|
+
// 确保数据写入磁盘
|
|
2378
|
+
stream.uncork();
|
|
2379
|
+
};
|
|
2380
|
+
|
|
2381
|
+
/**
|
|
2382
|
+
* 同时输出到控制台和日志文件
|
|
2383
|
+
*/
|
|
2384
|
+
const logWithFile = (
|
|
2385
|
+
stream,
|
|
2386
|
+
level,
|
|
2387
|
+
message,
|
|
2388
|
+
) => {
|
|
2389
|
+
// 输出到控制台
|
|
2390
|
+
switch (level) {
|
|
2391
|
+
case 'info':
|
|
2392
|
+
logger.info(message);
|
|
2393
|
+
break;
|
|
2394
|
+
case 'success':
|
|
2395
|
+
logger.success(message);
|
|
2396
|
+
break;
|
|
2397
|
+
case 'error':
|
|
2398
|
+
logger.error(message);
|
|
2399
|
+
break;
|
|
2400
|
+
default:
|
|
2401
|
+
logger.info(message);
|
|
2402
|
+
break;
|
|
2403
|
+
}
|
|
2404
|
+
|
|
2405
|
+
// 写入日志文件(带时间戳)
|
|
2406
|
+
writeLogWithTimestamp(stream, `[${level.toUpperCase()}] ${message}`);
|
|
2407
|
+
};
|
|
2408
|
+
|
|
2409
|
+
// start_aigc
|
|
2410
|
+
/**
|
|
2411
|
+
* 构建 pnpm add 命令
|
|
2412
|
+
*/
|
|
2413
|
+
const buildPnpmCommand = (
|
|
2414
|
+
packageName,
|
|
2415
|
+
options
|
|
2416
|
+
|
|
2417
|
+
|
|
2418
|
+
|
|
2419
|
+
|
|
2420
|
+
,
|
|
2421
|
+
) => {
|
|
2422
|
+
const { global, version, registry, extraArgs } = options;
|
|
2423
|
+
|
|
2424
|
+
const parts = ['pnpm', 'add'];
|
|
2425
|
+
|
|
2426
|
+
// 添加全局标记
|
|
2427
|
+
if (global) {
|
|
2428
|
+
parts.push('-g');
|
|
2429
|
+
}
|
|
2430
|
+
|
|
2431
|
+
// 添加包名和版本
|
|
2432
|
+
if (version && version !== 'latest') {
|
|
2433
|
+
parts.push(`${packageName}@${version}`);
|
|
2434
|
+
} else {
|
|
2435
|
+
parts.push(`${packageName}@latest`);
|
|
2436
|
+
}
|
|
2437
|
+
|
|
2438
|
+
// 添加 registry
|
|
2439
|
+
if (registry) {
|
|
2440
|
+
parts.push(`--registry=${registry}`);
|
|
2441
|
+
}
|
|
2442
|
+
|
|
2443
|
+
// 添加额外参数
|
|
2444
|
+
if (extraArgs.length > 0) {
|
|
2445
|
+
parts.push(...extraArgs);
|
|
2446
|
+
}
|
|
2447
|
+
|
|
2448
|
+
return parts.join(' ');
|
|
2449
|
+
};
|
|
2450
|
+
// end_aigc
|
|
2451
|
+
|
|
2452
|
+
// start_aigc
|
|
2453
|
+
/**
|
|
2454
|
+
* 执行 update 命令的内部实现
|
|
2455
|
+
*/
|
|
2456
|
+
const executeUpdate = (
|
|
2457
|
+
packageName,
|
|
2458
|
+
options
|
|
2459
|
+
|
|
2460
|
+
|
|
2461
|
+
|
|
2462
|
+
|
|
2463
|
+
|
|
2464
|
+
|
|
2465
|
+
,
|
|
2466
|
+
) => {
|
|
2467
|
+
let logStream = null;
|
|
2468
|
+
|
|
2469
|
+
try {
|
|
2470
|
+
const { global, cwd, version, registry, logFile, extraArgs } = options;
|
|
2471
|
+
|
|
2472
|
+
// 准备日志
|
|
2473
|
+
const logFilePath = resolveLogFilePath(logFile);
|
|
2474
|
+
|
|
2475
|
+
// 调试:确认日志路径
|
|
2476
|
+
logger.info(`Log file path resolved to: ${logFilePath}`);
|
|
2477
|
+
|
|
2478
|
+
logStream = createLogStream(logFilePath);
|
|
2479
|
+
|
|
2480
|
+
// 调试:确认流已创建
|
|
2481
|
+
logger.info('Log stream created successfully');
|
|
2482
|
+
|
|
2483
|
+
logWithFile(logStream, 'info', `Updating package: ${packageName}`);
|
|
2484
|
+
|
|
2485
|
+
// 构建命令
|
|
2486
|
+
const command = buildPnpmCommand(packageName, {
|
|
2487
|
+
global,
|
|
2488
|
+
version,
|
|
2489
|
+
registry,
|
|
2490
|
+
extraArgs,
|
|
2491
|
+
});
|
|
2492
|
+
|
|
2493
|
+
// 确定工作目录
|
|
2494
|
+
const workingDir = cwd
|
|
2495
|
+
? path.isAbsolute(cwd)
|
|
2496
|
+
? cwd
|
|
2497
|
+
: path.join(process.cwd(), cwd)
|
|
2498
|
+
: process.cwd();
|
|
2499
|
+
|
|
2500
|
+
logWithFile(logStream, 'info', `Executing: ${command}`);
|
|
2501
|
+
logWithFile(logStream, 'info', `Working directory: ${workingDir}`);
|
|
2502
|
+
logWithFile(logStream, 'info', `Log file: ${logFilePath}`);
|
|
2503
|
+
|
|
2504
|
+
// 记录命令开始时间
|
|
2505
|
+
writeLogWithTimestamp(logStream, '--- Command execution started ---');
|
|
2506
|
+
|
|
2507
|
+
// 同步执行命令
|
|
2508
|
+
const result = shelljs.exec(command, {
|
|
2509
|
+
cwd: workingDir,
|
|
2510
|
+
silent: true, // 使用 silent 来捕获输出
|
|
2511
|
+
});
|
|
2512
|
+
|
|
2513
|
+
// 将输出写入控制台和日志文件(带时间戳)
|
|
2514
|
+
if (result.stdout) {
|
|
2515
|
+
process.stdout.write(result.stdout);
|
|
2516
|
+
writeLogWithTimestamp(logStream, result.stdout.trim());
|
|
2517
|
+
}
|
|
2518
|
+
|
|
2519
|
+
if (result.stderr) {
|
|
2520
|
+
process.stderr.write(result.stderr);
|
|
2521
|
+
writeLogWithTimestamp(logStream, result.stderr.trim());
|
|
2522
|
+
}
|
|
2523
|
+
|
|
2524
|
+
// 记录命令结束时间
|
|
2525
|
+
writeLogWithTimestamp(logStream, '--- Command execution ended ---');
|
|
2526
|
+
|
|
2527
|
+
// 检查执行结果并记录到日志
|
|
2528
|
+
if (result.code === 0) {
|
|
2529
|
+
logWithFile(logStream, 'success', 'Package updated successfully');
|
|
2530
|
+
logWithFile(logStream, 'info', `Log file: ${logFilePath}`);
|
|
2531
|
+
} else {
|
|
2532
|
+
logWithFile(
|
|
2533
|
+
logStream,
|
|
2534
|
+
'error',
|
|
2535
|
+
`Command exited with code ${result.code}`,
|
|
2536
|
+
);
|
|
2537
|
+
logWithFile(
|
|
2538
|
+
logStream,
|
|
2539
|
+
'error',
|
|
2540
|
+
`Check log file for details: ${logFilePath}`,
|
|
2541
|
+
);
|
|
2542
|
+
}
|
|
2543
|
+
|
|
2544
|
+
// 关闭日志流并等待写入完成
|
|
2545
|
+
logStream.end(() => {
|
|
2546
|
+
// 流关闭后再退出进程
|
|
2547
|
+
if (result.code !== 0) {
|
|
2548
|
+
process.exit(result.code || 1);
|
|
2549
|
+
}
|
|
2550
|
+
});
|
|
2551
|
+
} catch (error) {
|
|
2552
|
+
logger.error('Failed to update package:');
|
|
2553
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
2554
|
+
|
|
2555
|
+
// 写入错误到日志文件
|
|
2556
|
+
if (logStream) {
|
|
2557
|
+
writeLogWithTimestamp(
|
|
2558
|
+
logStream,
|
|
2559
|
+
`[ERROR] ${error instanceof Error ? error.message : String(error)}`,
|
|
2560
|
+
);
|
|
2561
|
+
// 等待流关闭后再退出
|
|
2562
|
+
logStream.end(() => {
|
|
2563
|
+
process.exit(1);
|
|
2564
|
+
});
|
|
2565
|
+
} else {
|
|
2566
|
+
process.exit(1);
|
|
2567
|
+
}
|
|
2568
|
+
}
|
|
2569
|
+
};
|
|
2570
|
+
// end_aigc
|
|
2571
|
+
|
|
2572
|
+
/**
|
|
2573
|
+
* 注册 update 命令到 program
|
|
2574
|
+
*/
|
|
2575
|
+
const registerCommand = program => {
|
|
2576
|
+
program
|
|
2577
|
+
.command('update <package>')
|
|
2578
|
+
.description('Update a package dependency')
|
|
2579
|
+
.option('-g, --global', 'Update package globally', false)
|
|
2580
|
+
.option('-c, --cwd <path>', 'Working directory for the update')
|
|
2581
|
+
.option(
|
|
2582
|
+
'--to <version>',
|
|
2583
|
+
'Version to update to (default: latest)',
|
|
2584
|
+
'latest',
|
|
2585
|
+
)
|
|
2586
|
+
.option('--registry <url>', 'Registry URL to use for the update')
|
|
2587
|
+
.option('--log-file <path>', 'Log file path')
|
|
2588
|
+
.allowUnknownOption() // 允许透传参数给 pnpm
|
|
2589
|
+
.action((packageName, options, command) => {
|
|
2590
|
+
// 收集所有未知选项作为额外参数
|
|
2591
|
+
const extraArgs = command.args.slice(1);
|
|
2592
|
+
|
|
2593
|
+
executeUpdate(packageName, {
|
|
2594
|
+
...options,
|
|
2595
|
+
version: options.to, // 将 --to 映射到 version
|
|
2596
|
+
extraArgs,
|
|
2597
|
+
});
|
|
2121
2598
|
});
|
|
2122
2599
|
};
|
|
2123
2600
|
|
|
2124
|
-
var version = "0.0.1-alpha.
|
|
2601
|
+
var version = "0.0.1-alpha.e89608";
|
|
2125
2602
|
var packageJson = {
|
|
2126
2603
|
version: version};
|
|
2127
2604
|
|
|
2128
2605
|
const commands = [
|
|
2129
|
-
registerCommand,
|
|
2130
2606
|
registerCommand$1,
|
|
2131
|
-
registerCommand$3,
|
|
2132
2607
|
registerCommand$2,
|
|
2608
|
+
registerCommand$4,
|
|
2609
|
+
registerCommand$3,
|
|
2610
|
+
registerCommand,
|
|
2133
2611
|
];
|
|
2134
2612
|
|
|
2135
2613
|
const main = () => {
|