@coze-arch/cli 0.0.1-alpha.ccd993 → 0.0.1-alpha.cf9a3b
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/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/_layout.tsx +11 -8
- package/lib/__templates__/expo/client/app.config.ts +2 -2
- package/lib/__templates__/expo/client/components/Screen.tsx +1 -17
- package/lib/__templates__/expo/client/components/ThemedView.tsx +1 -2
- package/lib/__templates__/expo/client/constants/theme.ts +21 -698
- package/lib/__templates__/expo/client/eslint.config.mjs +20 -0
- package/lib/__templates__/expo/client/hooks/{useColorScheme.ts → useColorScheme.tsx} +20 -6
- 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/eslint-plugins/fontawesome6/names.js +1886 -2483
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/rule.js +20 -1
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/v5-only-names.js +388 -0
- package/lib/__templates__/expo/eslint-plugins/react-native/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/react-native/rule.js +64 -0
- package/lib/__templates__/expo/eslint-plugins/reanimated/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/reanimated/rule.js +88 -0
- package/lib/__templates__/expo/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__/nextjs/.babelrc +15 -0
- package/lib/__templates__/nextjs/package.json +11 -1
- package/lib/__templates__/nextjs/pnpm-lock.yaml +2701 -1813
- package/lib/__templates__/nextjs/src/app/layout.tsx +5 -3
- package/lib/__templates__/nextjs/src/app/page.tsx +18 -60
- package/lib/__templates__/nextjs/template.config.js +47 -12
- 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 +749 -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 +173 -0
- package/lib/__templates__/taro/config/prod.ts +35 -0
- package/lib/__templates__/taro/eslint.config.mjs +79 -0
- package/lib/__templates__/taro/key/private.appid.key +0 -0
- package/lib/__templates__/taro/package.json +97 -0
- package/lib/__templates__/taro/pnpm-lock.yaml +22694 -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/h5-navbar.tsx +171 -0
- package/lib/__templates__/taro/src/presets/h5-styles.ts +33 -0
- package/lib/__templates__/taro/src/presets/index.tsx +18 -0
- package/lib/__templates__/taro/src/presets/wx-debug.ts +23 -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 +32 -0
- package/lib/__templates__/vite/package.json +10 -1
- package/lib/__templates__/vite/pnpm-lock.yaml +350 -2341
- package/lib/__templates__/vite/src/main.ts +17 -47
- package/lib/__templates__/vite/template.config.js +47 -10
- package/lib/cli.js +818 -124
- package/package.json +1 -1
package/lib/cli.js
CHANGED
|
@@ -8,8 +8,8 @@ var shelljs = require('shelljs');
|
|
|
8
8
|
var perf_hooks = require('perf_hooks');
|
|
9
9
|
var fs$1 = require('fs/promises');
|
|
10
10
|
var os = require('os');
|
|
11
|
-
var toml = require('@iarna/toml');
|
|
12
11
|
var jsYaml = require('js-yaml');
|
|
12
|
+
var toml = require('@iarna/toml');
|
|
13
13
|
var child_process = require('child_process');
|
|
14
14
|
var addFormats = require('ajv-formats');
|
|
15
15
|
var Ajv = require('ajv');
|
|
@@ -585,7 +585,7 @@ const executeWarmup = async (
|
|
|
585
585
|
/**
|
|
586
586
|
* 注册 warmup 命令到 program
|
|
587
587
|
*/
|
|
588
|
-
const registerCommand$
|
|
588
|
+
const registerCommand$4 = program => {
|
|
589
589
|
program
|
|
590
590
|
.command('warmup')
|
|
591
591
|
.description('Pre-install dependencies for templates to speed up init')
|
|
@@ -727,7 +727,7 @@ const parseConfigContent = (content) => {
|
|
|
727
727
|
return config ;
|
|
728
728
|
} catch (error) {
|
|
729
729
|
// TOML 解析失败,继续尝试其他格式
|
|
730
|
-
|
|
730
|
+
|
|
731
731
|
console.debug('TOML parse failed:', error);
|
|
732
732
|
}
|
|
733
733
|
|
|
@@ -739,7 +739,7 @@ const parseConfigContent = (content) => {
|
|
|
739
739
|
}
|
|
740
740
|
} catch (error) {
|
|
741
741
|
// YAML 解析失败,继续尝试其他格式
|
|
742
|
-
|
|
742
|
+
|
|
743
743
|
console.debug('YAML parse failed:', error);
|
|
744
744
|
}
|
|
745
745
|
|
|
@@ -823,18 +823,242 @@ const getCommandConfig = (
|
|
|
823
823
|
return commandConfig;
|
|
824
824
|
};
|
|
825
825
|
|
|
826
|
-
|
|
826
|
+
// ABOUTME: Fix rule to comment out problematic outputFileTracingRoot config in Next.js projects
|
|
827
|
+
// ABOUTME: This config can cause issues in monorepo environments and should be removed
|
|
828
|
+
|
|
829
|
+
|
|
830
|
+
|
|
831
|
+
|
|
832
|
+
/**
|
|
833
|
+
* 检查是否为 Next.js 项目
|
|
834
|
+
*/
|
|
835
|
+
const isNextProject = (projectFolder) => {
|
|
836
|
+
const packageJsonPath = path.join(projectFolder, 'package.json');
|
|
837
|
+
|
|
838
|
+
if (!fs.existsSync(packageJsonPath)) {
|
|
839
|
+
return false;
|
|
840
|
+
}
|
|
841
|
+
|
|
842
|
+
try {
|
|
843
|
+
// eslint-disable-next-line no-restricted-syntax
|
|
844
|
+
const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf-8'));
|
|
845
|
+
const deps = {
|
|
846
|
+
...packageJson.dependencies,
|
|
847
|
+
...packageJson.devDependencies,
|
|
848
|
+
};
|
|
849
|
+
|
|
850
|
+
return 'next' in deps;
|
|
851
|
+
} catch (e) {
|
|
852
|
+
return false;
|
|
853
|
+
}
|
|
854
|
+
};
|
|
855
|
+
|
|
856
|
+
/**
|
|
857
|
+
* 查找 Next.js 配置文件
|
|
858
|
+
*/
|
|
859
|
+
const findNextConfigFile = (projectFolder) => {
|
|
860
|
+
const possibleConfigs = [
|
|
861
|
+
'next.config.ts',
|
|
862
|
+
'next.config.js',
|
|
863
|
+
'next.config.mjs',
|
|
864
|
+
];
|
|
865
|
+
|
|
866
|
+
for (const configFile of possibleConfigs) {
|
|
867
|
+
const configPath = path.join(projectFolder, configFile);
|
|
868
|
+
if (fs.existsSync(configPath)) {
|
|
869
|
+
return configPath;
|
|
870
|
+
}
|
|
871
|
+
}
|
|
872
|
+
|
|
873
|
+
return null;
|
|
874
|
+
};
|
|
827
875
|
|
|
876
|
+
/**
|
|
877
|
+
* 注释掉 outputFileTracingRoot 配置
|
|
878
|
+
*/
|
|
879
|
+
const commentOutOutputTracingRoot = (
|
|
880
|
+
configPath,
|
|
881
|
+
) => {
|
|
882
|
+
let content = fs.readFileSync(configPath, 'utf-8');
|
|
883
|
+
let modified = false;
|
|
884
|
+
let originalLine = null;
|
|
885
|
+
|
|
886
|
+
// 匹配包含 outputFileTracingRoot 的行(尚未被注释的)
|
|
887
|
+
// 支持 path.resolve(...) 后面有空格,然后可选逗号
|
|
888
|
+
// 只匹配单行配置,不匹配跨多行的配置
|
|
889
|
+
const pattern =
|
|
890
|
+
/^(\s*)(outputFileTracingRoot:\s*path\.resolve\([^\n\r)]+\)\s*,?)\s*$/gm;
|
|
891
|
+
|
|
892
|
+
const matches = content.match(pattern);
|
|
893
|
+
|
|
894
|
+
if (matches && matches.length > 0) {
|
|
895
|
+
originalLine = matches[0].trim();
|
|
896
|
+
|
|
897
|
+
// 在匹配的行前添加 // 注释
|
|
898
|
+
content = content.replace(pattern, '$1// $2');
|
|
899
|
+
modified = true;
|
|
900
|
+
|
|
901
|
+
fs.writeFileSync(configPath, content, 'utf-8');
|
|
902
|
+
}
|
|
903
|
+
|
|
904
|
+
return { modified, originalLine };
|
|
905
|
+
};
|
|
906
|
+
|
|
907
|
+
// start_aigc
|
|
908
|
+
/**
|
|
909
|
+
* Fix 规则:注释掉 Next.js 项目中的 outputFileTracingRoot 配置
|
|
910
|
+
* 这个配置在 monorepo 环境中可能会导致问题
|
|
911
|
+
*/
|
|
912
|
+
const fixNextOutputTracingRoot = context => {
|
|
913
|
+
const ruleName = 'next-output-tracing-root';
|
|
914
|
+
|
|
915
|
+
// 1. 检查是否为 Next.js 项目
|
|
916
|
+
if (!isNextProject(context.projectFolder)) {
|
|
917
|
+
return {
|
|
918
|
+
ruleName,
|
|
919
|
+
applied: false,
|
|
920
|
+
message: 'Not a Next.js project, skipping',
|
|
921
|
+
};
|
|
922
|
+
}
|
|
923
|
+
|
|
924
|
+
// 2. 查找 Next.js 配置文件
|
|
925
|
+
const configPath = findNextConfigFile(context.projectFolder);
|
|
926
|
+
|
|
927
|
+
if (!configPath) {
|
|
928
|
+
return {
|
|
929
|
+
ruleName,
|
|
930
|
+
applied: false,
|
|
931
|
+
message: 'Next.js config file not found, skipping',
|
|
932
|
+
};
|
|
933
|
+
}
|
|
934
|
+
|
|
935
|
+
// 3. 注释掉 outputFileTracingRoot 配置
|
|
936
|
+
const { modified, originalLine } = commentOutOutputTracingRoot(configPath);
|
|
937
|
+
|
|
938
|
+
if (modified && originalLine) {
|
|
939
|
+
logger.success(
|
|
940
|
+
`Commented out outputFileTracingRoot in ${configPath.split('/').pop()}`,
|
|
941
|
+
);
|
|
942
|
+
logger.info(` Original: ${originalLine}`);
|
|
943
|
+
|
|
944
|
+
return {
|
|
945
|
+
ruleName,
|
|
946
|
+
applied: true,
|
|
947
|
+
message: `Successfully commented out: ${originalLine}`,
|
|
948
|
+
};
|
|
949
|
+
}
|
|
950
|
+
|
|
951
|
+
return {
|
|
952
|
+
ruleName,
|
|
953
|
+
applied: false,
|
|
954
|
+
message: 'No outputFileTracingRoot config found, skipping',
|
|
955
|
+
};
|
|
956
|
+
};
|
|
957
|
+
// end_aigc
|
|
958
|
+
|
|
959
|
+
/**
|
|
960
|
+
* 所有修复规则的数组
|
|
961
|
+
* 按顺序执行,新增规则直接添加到数组中
|
|
962
|
+
*/
|
|
963
|
+
const rules = [
|
|
964
|
+
// Next.js related fixes
|
|
965
|
+
fixNextOutputTracingRoot,
|
|
966
|
+
|
|
967
|
+
// Add more rules here
|
|
968
|
+
] ;
|
|
969
|
+
|
|
970
|
+
// ABOUTME: Fix command for resolving legacy issues from previous project versions
|
|
971
|
+
// ABOUTME: Applies a series of fix rules defined in the fix-rules directory
|
|
972
|
+
|
|
973
|
+
|
|
974
|
+
// start_aigc
|
|
975
|
+
/**
|
|
976
|
+
* 执行 fix 命令的内部实现
|
|
977
|
+
*/
|
|
978
|
+
const executeFix = async (options = {}) => {
|
|
979
|
+
try {
|
|
980
|
+
const cwd = process.cwd();
|
|
981
|
+
const projectFolder = options.directory
|
|
982
|
+
? path.resolve(cwd, options.directory)
|
|
983
|
+
: cwd;
|
|
984
|
+
|
|
985
|
+
logger.info(`Running fix command on: ${projectFolder}`);
|
|
986
|
+
logger.info(`Found ${rules.length} fix rule(s) to apply\n`);
|
|
987
|
+
|
|
988
|
+
const context = {
|
|
989
|
+
cwd,
|
|
990
|
+
projectFolder,
|
|
991
|
+
};
|
|
992
|
+
|
|
993
|
+
let appliedCount = 0;
|
|
994
|
+
let skippedCount = 0;
|
|
995
|
+
|
|
996
|
+
// 依次执行所有修复规则
|
|
997
|
+
for (const rule of rules) {
|
|
998
|
+
try {
|
|
999
|
+
const result = await Promise.resolve(rule(context));
|
|
1000
|
+
|
|
1001
|
+
if (result.applied) {
|
|
1002
|
+
appliedCount++;
|
|
1003
|
+
logger.success(`✓ ${result.ruleName}: ${result.message}`);
|
|
1004
|
+
} else {
|
|
1005
|
+
skippedCount++;
|
|
1006
|
+
logger.info(`○ ${result.ruleName}: ${result.message}`);
|
|
1007
|
+
}
|
|
1008
|
+
} catch (error) {
|
|
1009
|
+
logger.error(
|
|
1010
|
+
`✗ Rule execution failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
1011
|
+
);
|
|
1012
|
+
}
|
|
1013
|
+
}
|
|
1014
|
+
|
|
1015
|
+
// 输出汇总信息
|
|
1016
|
+
logger.info(
|
|
1017
|
+
`\nSummary: ${appliedCount} fixed, ${skippedCount} skipped, ${rules.length} total`,
|
|
1018
|
+
);
|
|
1019
|
+
|
|
1020
|
+
if (appliedCount > 0) {
|
|
1021
|
+
logger.success('\nFixes applied successfully!');
|
|
1022
|
+
} else {
|
|
1023
|
+
logger.info('\nNo fixes needed');
|
|
1024
|
+
}
|
|
1025
|
+
} catch (error) {
|
|
1026
|
+
logger.error('Failed to run fix command:');
|
|
1027
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
1028
|
+
process.exit(1);
|
|
1029
|
+
}
|
|
1030
|
+
};
|
|
1031
|
+
// end_aigc
|
|
1032
|
+
|
|
1033
|
+
/**
|
|
1034
|
+
* 注册 fix 命令到 program
|
|
1035
|
+
*/
|
|
1036
|
+
const registerCommand$3 = program => {
|
|
1037
|
+
program
|
|
1038
|
+
.command('fix')
|
|
1039
|
+
.description(
|
|
1040
|
+
'Fix legacy issues from previous versions (e.g., problematic configs)',
|
|
1041
|
+
)
|
|
1042
|
+
.argument(
|
|
1043
|
+
'[directory]',
|
|
1044
|
+
'Target directory to fix (defaults to current directory)',
|
|
1045
|
+
)
|
|
1046
|
+
.action(async (directory) => {
|
|
1047
|
+
await executeFix({ directory });
|
|
1048
|
+
});
|
|
1049
|
+
};
|
|
1050
|
+
|
|
1051
|
+
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; }
|
|
828
1052
|
/**
|
|
829
1053
|
* 日志文件名常量
|
|
830
1054
|
*/
|
|
831
|
-
const LOG_FILE_NAME = 'dev.log';
|
|
1055
|
+
const LOG_FILE_NAME$1 = 'dev.log';
|
|
832
1056
|
|
|
833
1057
|
/**
|
|
834
1058
|
* 获取日志目录
|
|
835
1059
|
* 优先使用环境变量 COZE_LOG_DIR,否则使用 ~/.coze-logs
|
|
836
1060
|
*/
|
|
837
|
-
const getLogDir = () =>
|
|
1061
|
+
const getLogDir$1 = () =>
|
|
838
1062
|
process.env.COZE_LOG_DIR || path.join(os.homedir(), '.coze-logs');
|
|
839
1063
|
|
|
840
1064
|
/**
|
|
@@ -843,22 +1067,22 @@ const getLogDir = () =>
|
|
|
843
1067
|
* - 如果是相对路径,基于 getLogDir() + 相对路径
|
|
844
1068
|
* - 如果为空,使用 getLogDir() + LOG_FILE_NAME
|
|
845
1069
|
*/
|
|
846
|
-
const resolveLogFilePath = (logFile) => {
|
|
1070
|
+
const resolveLogFilePath$1 = (logFile) => {
|
|
847
1071
|
if (!logFile) {
|
|
848
|
-
return path.join(getLogDir(), LOG_FILE_NAME);
|
|
1072
|
+
return path.join(getLogDir$1(), LOG_FILE_NAME$1);
|
|
849
1073
|
}
|
|
850
1074
|
|
|
851
1075
|
if (path.isAbsolute(logFile)) {
|
|
852
1076
|
return logFile;
|
|
853
1077
|
}
|
|
854
1078
|
|
|
855
|
-
return path.join(getLogDir(), logFile);
|
|
1079
|
+
return path.join(getLogDir$1(), logFile);
|
|
856
1080
|
};
|
|
857
1081
|
|
|
858
1082
|
/**
|
|
859
1083
|
* 创建日志写入流
|
|
860
1084
|
*/
|
|
861
|
-
const createLogStream = (logFilePath) => {
|
|
1085
|
+
const createLogStream$1 = (logFilePath) => {
|
|
862
1086
|
const logDir = path.dirname(logFilePath);
|
|
863
1087
|
|
|
864
1088
|
// 确保日志目录存在
|
|
@@ -880,15 +1104,27 @@ const executeRun = async (
|
|
|
880
1104
|
try {
|
|
881
1105
|
logger.info(`Running ${commandName} command...`);
|
|
882
1106
|
|
|
883
|
-
// 1.
|
|
1107
|
+
// 1. 对于 build 命令,先执行 fix 以确保项目配置正确
|
|
1108
|
+
if (['dev', 'build'].includes(commandName)) {
|
|
1109
|
+
logger.info('\n🔧 Running fix command before build...\n');
|
|
1110
|
+
try {
|
|
1111
|
+
await executeFix();
|
|
1112
|
+
// eslint-disable-next-line @coze-arch/no-empty-catch
|
|
1113
|
+
} catch (e) {
|
|
1114
|
+
// just ignore
|
|
1115
|
+
}
|
|
1116
|
+
logger.info('');
|
|
1117
|
+
}
|
|
1118
|
+
|
|
1119
|
+
// 2. 加载 .coze 配置
|
|
884
1120
|
const config = await loadCozeConfig();
|
|
885
1121
|
const commandArgs = getCommandConfig(config, commandName);
|
|
886
1122
|
|
|
887
|
-
//
|
|
888
|
-
const logFilePath = resolveLogFilePath(options.logFile);
|
|
889
|
-
const logStream = createLogStream(logFilePath);
|
|
1123
|
+
// 3. 准备日志
|
|
1124
|
+
const logFilePath = resolveLogFilePath$1(options.logFile);
|
|
1125
|
+
const logStream = createLogStream$1(logFilePath);
|
|
890
1126
|
|
|
891
|
-
//
|
|
1127
|
+
// 4. 执行命令
|
|
892
1128
|
const commandString = commandArgs.join(' ');
|
|
893
1129
|
|
|
894
1130
|
logger.info(`Executing: ${commandString}`);
|
|
@@ -949,7 +1185,7 @@ const executeRun = async (
|
|
|
949
1185
|
/**
|
|
950
1186
|
* 注册 dev/build/start 命令到 program
|
|
951
1187
|
*/
|
|
952
|
-
const registerCommand$
|
|
1188
|
+
const registerCommand$2 = program => {
|
|
953
1189
|
// dev 命令
|
|
954
1190
|
program
|
|
955
1191
|
.command('dev')
|
|
@@ -1201,6 +1437,11 @@ const shouldIgnoreFile = (filePath) => {
|
|
|
1201
1437
|
return directoryPatterns.some(dir => pathParts.includes(dir));
|
|
1202
1438
|
};
|
|
1203
1439
|
|
|
1440
|
+
// ABOUTME: File system utilities for template file processing
|
|
1441
|
+
// ABOUTME: Provides directory scanning, file path conversion, and node_modules copying
|
|
1442
|
+
|
|
1443
|
+
|
|
1444
|
+
// start_aigc
|
|
1204
1445
|
/**
|
|
1205
1446
|
* 递归获取目录中的所有文件
|
|
1206
1447
|
*
|
|
@@ -1270,7 +1511,19 @@ const convertDotfileName = (filePath) => {
|
|
|
1270
1511
|
|
|
1271
1512
|
return filePath;
|
|
1272
1513
|
};
|
|
1514
|
+
// end_aigc
|
|
1515
|
+
|
|
1516
|
+
// ABOUTME: File rendering utilities for template processing
|
|
1517
|
+
// ABOUTME: Handles file content rendering, hook execution, and file writing
|
|
1518
|
+
|
|
1519
|
+
|
|
1520
|
+
|
|
1521
|
+
|
|
1522
|
+
|
|
1523
|
+
|
|
1273
1524
|
|
|
1525
|
+
|
|
1526
|
+
// start_aigc
|
|
1274
1527
|
/**
|
|
1275
1528
|
* 执行文件渲染钩子
|
|
1276
1529
|
*
|
|
@@ -1316,22 +1569,25 @@ const executeFileRenderHook = async (
|
|
|
1316
1569
|
};
|
|
1317
1570
|
|
|
1318
1571
|
/**
|
|
1319
|
-
*
|
|
1572
|
+
* 准备单个文件的渲染信息(不实际写入)
|
|
1573
|
+
* 用于 dry-run 阶段,收集所有将要写入的文件信息
|
|
1574
|
+
*
|
|
1575
|
+
* @param options - 准备选项
|
|
1576
|
+
* @returns 文件渲染信息,或 null 表示该文件被跳过
|
|
1320
1577
|
*/
|
|
1321
|
-
const
|
|
1322
|
-
|
|
1578
|
+
const prepareFileInfo = async (options
|
|
1323
1579
|
|
|
1324
1580
|
|
|
1325
1581
|
|
|
1326
1582
|
|
|
1327
1583
|
) => {
|
|
1328
|
-
const { file, templatePath,
|
|
1584
|
+
const { file, templatePath, context, templateConfig } = options;
|
|
1329
1585
|
|
|
1330
1586
|
const srcPath = path.join(templatePath, file);
|
|
1331
1587
|
const destFile = convertDotfileName(file);
|
|
1332
1588
|
|
|
1333
1589
|
logger.verbose(
|
|
1334
|
-
` -
|
|
1590
|
+
` - Preparing: ${file}${destFile !== file ? ` -> ${destFile}` : ''}`,
|
|
1335
1591
|
);
|
|
1336
1592
|
|
|
1337
1593
|
// 判断是否为二进制文件
|
|
@@ -1365,38 +1621,195 @@ const processSingleFile = async (options
|
|
|
1365
1621
|
context,
|
|
1366
1622
|
);
|
|
1367
1623
|
|
|
1368
|
-
// 如果返回 null
|
|
1624
|
+
// 如果返回 null,表示该文件被 hook 跳过
|
|
1369
1625
|
if (processedFileInfo === null) {
|
|
1370
1626
|
logger.verbose(' ⊘ Skipped by onFileRender hook');
|
|
1371
|
-
return;
|
|
1627
|
+
return null;
|
|
1372
1628
|
}
|
|
1373
1629
|
|
|
1374
|
-
|
|
1375
|
-
|
|
1630
|
+
return processedFileInfo;
|
|
1631
|
+
};
|
|
1632
|
+
|
|
1633
|
+
/**
|
|
1634
|
+
* 写入渲染后的文件到目标路径
|
|
1635
|
+
*
|
|
1636
|
+
* @param options - 写入选项
|
|
1637
|
+
*/
|
|
1638
|
+
const writeRenderedFile = async (options
|
|
1639
|
+
|
|
1640
|
+
|
|
1641
|
+
|
|
1642
|
+
) => {
|
|
1643
|
+
const { fileInfo, srcPath, destPath } = options;
|
|
1644
|
+
|
|
1645
|
+
logger.verbose(` - Writing: ${fileInfo.destPath}`);
|
|
1376
1646
|
|
|
1377
1647
|
// 确保目标目录存在
|
|
1378
|
-
await ensureDir(path.dirname(
|
|
1648
|
+
await ensureDir(path.dirname(destPath));
|
|
1379
1649
|
|
|
1380
1650
|
// 写入文件
|
|
1381
|
-
if (
|
|
1382
|
-
//
|
|
1383
|
-
|
|
1384
|
-
|
|
1651
|
+
if (fileInfo.isBinary) {
|
|
1652
|
+
// 二进制文件:如果内容是原始 base64(未被 hook 修改),直接复制;否则从 base64 解码写入
|
|
1653
|
+
const buffer = await fs$1.readFile(srcPath);
|
|
1654
|
+
const originalContent = buffer.toString('base64');
|
|
1655
|
+
|
|
1656
|
+
if (fileInfo.content === originalContent) {
|
|
1657
|
+
await fs$1.copyFile(srcPath, destPath);
|
|
1385
1658
|
logger.verbose(' ✓ Copied (binary)');
|
|
1386
1659
|
} else {
|
|
1387
|
-
const
|
|
1388
|
-
await fs$1.writeFile(
|
|
1660
|
+
const modifiedBuffer = Buffer.from(fileInfo.content, 'base64');
|
|
1661
|
+
await fs$1.writeFile(destPath, modifiedBuffer);
|
|
1389
1662
|
logger.verbose(' ✓ Written (binary, modified by hook)');
|
|
1390
1663
|
}
|
|
1391
1664
|
} else {
|
|
1392
1665
|
// 文本文件
|
|
1393
|
-
await fs$1.writeFile(
|
|
1666
|
+
await fs$1.writeFile(destPath, fileInfo.content, 'utf-8');
|
|
1394
1667
|
logger.verbose(' ✓ Rendered and written');
|
|
1395
1668
|
}
|
|
1396
1669
|
};
|
|
1670
|
+
// end_aigc
|
|
1671
|
+
|
|
1672
|
+
// ABOUTME: File conflict detection utilities for template processing
|
|
1673
|
+
// ABOUTME: Provides dry-run file collection and conflict checking
|
|
1674
|
+
|
|
1675
|
+
|
|
1676
|
+
|
|
1677
|
+
|
|
1678
|
+
|
|
1679
|
+
|
|
1680
|
+
|
|
1681
|
+
// start_aigc
|
|
1682
|
+
/**
|
|
1683
|
+
* 收集所有将要写入的文件路径(dry-run 阶段)
|
|
1684
|
+
*
|
|
1685
|
+
* @param options - 收集选项
|
|
1686
|
+
* @returns 将要写入的文件路径列表
|
|
1687
|
+
*/
|
|
1688
|
+
const collectFilesToRender = async (options
|
|
1689
|
+
|
|
1690
|
+
|
|
1691
|
+
|
|
1692
|
+
|
|
1693
|
+
) => {
|
|
1694
|
+
const { files, templatePath, context, templateConfig } = options;
|
|
1695
|
+
|
|
1696
|
+
logger.verbose('\nDry-run: Collecting files to render...');
|
|
1697
|
+
|
|
1698
|
+
const fileInfos = await Promise.all(
|
|
1699
|
+
files.map(file =>
|
|
1700
|
+
prepareFileInfo({
|
|
1701
|
+
file,
|
|
1702
|
+
templatePath,
|
|
1703
|
+
context,
|
|
1704
|
+
templateConfig,
|
|
1705
|
+
}),
|
|
1706
|
+
),
|
|
1707
|
+
);
|
|
1708
|
+
|
|
1709
|
+
// 过滤掉被 hook 跳过的文件,收集 destPath
|
|
1710
|
+
const filesToWrite = fileInfos
|
|
1711
|
+
.filter((info) => info !== null)
|
|
1712
|
+
.map(info => info.destPath);
|
|
1713
|
+
|
|
1714
|
+
logger.verbose(` - ${filesToWrite.length} files will be written`);
|
|
1715
|
+
logger.verbose(
|
|
1716
|
+
` - ${fileInfos.length - filesToWrite.length} files skipped by hooks`,
|
|
1717
|
+
);
|
|
1718
|
+
|
|
1719
|
+
return filesToWrite;
|
|
1720
|
+
};
|
|
1721
|
+
|
|
1722
|
+
/**
|
|
1723
|
+
* 检测文件冲突
|
|
1724
|
+
*
|
|
1725
|
+
* @param outputPath - 输出目录路径
|
|
1726
|
+
* @param filesToWrite - 将要写入的文件路径列表
|
|
1727
|
+
* @returns 冲突的文件路径列表
|
|
1728
|
+
*/
|
|
1729
|
+
const detectFileConflicts = (
|
|
1730
|
+
outputPath,
|
|
1731
|
+
filesToWrite,
|
|
1732
|
+
) => {
|
|
1733
|
+
logger.verbose('\nChecking for file conflicts...');
|
|
1734
|
+
|
|
1735
|
+
const conflicts = [];
|
|
1736
|
+
|
|
1737
|
+
for (const file of filesToWrite) {
|
|
1738
|
+
const fullPath = path.join(outputPath, file);
|
|
1739
|
+
if (fs.existsSync(fullPath)) {
|
|
1740
|
+
conflicts.push(file);
|
|
1741
|
+
logger.verbose(` ⚠ Conflict detected: ${file}`);
|
|
1742
|
+
}
|
|
1743
|
+
}
|
|
1744
|
+
|
|
1745
|
+
if (conflicts.length === 0) {
|
|
1746
|
+
logger.verbose(' ✓ No conflicts detected');
|
|
1747
|
+
} else {
|
|
1748
|
+
logger.verbose(` ⚠ ${conflicts.length} conflicts detected`);
|
|
1749
|
+
}
|
|
1750
|
+
|
|
1751
|
+
return conflicts;
|
|
1752
|
+
};
|
|
1753
|
+
// end_aigc
|
|
1754
|
+
|
|
1755
|
+
// ABOUTME: Main file processing orchestration for template rendering
|
|
1756
|
+
// ABOUTME: Coordinates file system, rendering, and conflict detection layers
|
|
1757
|
+
|
|
1758
|
+
|
|
1759
|
+
|
|
1760
|
+
// start_aigc
|
|
1761
|
+
/**
|
|
1762
|
+
* 处理单个文件(准备 + 写入)
|
|
1763
|
+
*
|
|
1764
|
+
* @param options - 处理选项
|
|
1765
|
+
*/
|
|
1766
|
+
const processSingleFile = async (options
|
|
1767
|
+
|
|
1768
|
+
|
|
1769
|
+
|
|
1770
|
+
|
|
1771
|
+
|
|
1772
|
+
) => {
|
|
1773
|
+
const { file, templatePath, outputPath, context, templateConfig } = options;
|
|
1774
|
+
|
|
1775
|
+
const srcPath = path.join(templatePath, file);
|
|
1776
|
+
|
|
1777
|
+
// 准备文件信息
|
|
1778
|
+
const processedFileInfo = await prepareFileInfo({
|
|
1779
|
+
file,
|
|
1780
|
+
templatePath,
|
|
1781
|
+
context,
|
|
1782
|
+
templateConfig,
|
|
1783
|
+
});
|
|
1784
|
+
|
|
1785
|
+
// 如果返回 null,跳过该文件
|
|
1786
|
+
if (processedFileInfo === null) {
|
|
1787
|
+
return;
|
|
1788
|
+
}
|
|
1789
|
+
|
|
1790
|
+
// 使用处理后的目标路径
|
|
1791
|
+
const finalDestPath = path.join(outputPath, processedFileInfo.destPath);
|
|
1792
|
+
|
|
1793
|
+
// 写入文件
|
|
1794
|
+
await writeRenderedFile({
|
|
1795
|
+
fileInfo: processedFileInfo,
|
|
1796
|
+
srcPath,
|
|
1797
|
+
destPath: finalDestPath,
|
|
1798
|
+
});
|
|
1799
|
+
};
|
|
1397
1800
|
|
|
1398
1801
|
/**
|
|
1399
1802
|
* 复制并处理模板文件到目标目录
|
|
1803
|
+
*
|
|
1804
|
+
* 流程:
|
|
1805
|
+
* 1. 验证模板目录
|
|
1806
|
+
* 2. 扫描所有模板文件
|
|
1807
|
+
* 3. Dry-run:收集将要写入的文件列表(考虑 hooks 影响)
|
|
1808
|
+
* 4. 冲突检测:检查是否有文件会被覆盖
|
|
1809
|
+
* 5. 实际写入:渲染并写入所有文件
|
|
1810
|
+
* 6. 复制 node_modules(如果存在)
|
|
1811
|
+
*
|
|
1812
|
+
* @param options - 处理选项
|
|
1400
1813
|
*/
|
|
1401
1814
|
const processTemplateFiles = async (options
|
|
1402
1815
|
|
|
@@ -1409,7 +1822,7 @@ const processTemplateFiles = async (options
|
|
|
1409
1822
|
logger.verbose(` - Template path: ${templatePath}`);
|
|
1410
1823
|
logger.verbose(` - Output path: ${outputPath}`);
|
|
1411
1824
|
|
|
1412
|
-
// 验证模板目录是否存在
|
|
1825
|
+
// 阶段 0: 验证模板目录是否存在
|
|
1413
1826
|
try {
|
|
1414
1827
|
const stat = await fs$1.stat(templatePath);
|
|
1415
1828
|
logger.verbose(
|
|
@@ -1425,6 +1838,7 @@ const processTemplateFiles = async (options
|
|
|
1425
1838
|
throw error;
|
|
1426
1839
|
}
|
|
1427
1840
|
|
|
1841
|
+
// 阶段 1: 扫描所有模板文件
|
|
1428
1842
|
const files = await getAllFiles(templatePath);
|
|
1429
1843
|
|
|
1430
1844
|
logger.verbose(` - Found ${files.length} files to process`);
|
|
@@ -1434,6 +1848,29 @@ const processTemplateFiles = async (options
|
|
|
1434
1848
|
return;
|
|
1435
1849
|
}
|
|
1436
1850
|
|
|
1851
|
+
// 阶段 2: Dry-run - 收集所有将要写入的文件
|
|
1852
|
+
const filesToWrite = await collectFilesToRender({
|
|
1853
|
+
files,
|
|
1854
|
+
templatePath,
|
|
1855
|
+
context,
|
|
1856
|
+
templateConfig,
|
|
1857
|
+
});
|
|
1858
|
+
|
|
1859
|
+
// 阶段 3: 冲突检测
|
|
1860
|
+
const conflicts = detectFileConflicts(outputPath, filesToWrite);
|
|
1861
|
+
|
|
1862
|
+
if (conflicts.length > 0) {
|
|
1863
|
+
// 有冲突,抛出详细的错误信息
|
|
1864
|
+
const conflictList = conflicts.map(f => ` - ${f}`).join('\n');
|
|
1865
|
+
throw new Error(
|
|
1866
|
+
`File conflicts detected in output directory: ${outputPath}\n\n` +
|
|
1867
|
+
`The following files already exist and would be overwritten:\n${conflictList}\n\n` +
|
|
1868
|
+
'Please remove these files or use a different output directory.',
|
|
1869
|
+
);
|
|
1870
|
+
}
|
|
1871
|
+
|
|
1872
|
+
// 阶段 4: 实际写入文件
|
|
1873
|
+
logger.verbose('\nWriting files...');
|
|
1437
1874
|
await Promise.all(
|
|
1438
1875
|
files.map(file =>
|
|
1439
1876
|
processSingleFile({
|
|
@@ -1448,74 +1885,9 @@ const processTemplateFiles = async (options
|
|
|
1448
1885
|
|
|
1449
1886
|
logger.verbose('✓ All files processed successfully');
|
|
1450
1887
|
|
|
1451
|
-
//
|
|
1452
|
-
const sourceNodeModules = path.join(templatePath, 'node_modules');
|
|
1453
|
-
const targetNodeModules = path.join(outputPath, 'node_modules');
|
|
1454
|
-
|
|
1455
|
-
if (fs.existsSync(sourceNodeModules)) {
|
|
1456
|
-
logger.info('\nCopying node_modules from pre-warmed template...');
|
|
1457
|
-
logger.verbose(` From: ${sourceNodeModules}`);
|
|
1458
|
-
logger.verbose(` To: ${targetNodeModules}`);
|
|
1459
|
-
|
|
1460
|
-
const result = shelljs.exec(`cp -R "${sourceNodeModules}" "${targetNodeModules}"`, {
|
|
1461
|
-
silent: true,
|
|
1462
|
-
});
|
|
1463
|
-
|
|
1464
|
-
if (result.stdout) {
|
|
1465
|
-
process.stdout.write(result.stdout);
|
|
1466
|
-
}
|
|
1467
|
-
|
|
1468
|
-
if (result.stderr) {
|
|
1469
|
-
process.stderr.write(result.stderr);
|
|
1470
|
-
}
|
|
1471
|
-
|
|
1472
|
-
if (result.code === 0) {
|
|
1473
|
-
logger.success('✓ node_modules copied successfully');
|
|
1474
|
-
} else {
|
|
1475
|
-
logger.warn(
|
|
1476
|
-
`Failed to copy node_modules: ${result.stderr || 'unknown error'}`,
|
|
1477
|
-
);
|
|
1478
|
-
logger.info('Will need to run pnpm install manually');
|
|
1479
|
-
}
|
|
1480
|
-
}
|
|
1481
|
-
};
|
|
1482
|
-
|
|
1483
|
-
/**
|
|
1484
|
-
* 检查输出目录是否为空
|
|
1485
|
-
* 注意:.git 目录会被忽略,允许在已初始化 git 的目录中创建项目
|
|
1486
|
-
*
|
|
1487
|
-
* @param outputPath - 输出目录路径
|
|
1488
|
-
* @returns 是否为空
|
|
1489
|
-
*/
|
|
1490
|
-
const isOutputDirEmpty = async (
|
|
1491
|
-
outputPath,
|
|
1492
|
-
) => {
|
|
1493
|
-
try {
|
|
1494
|
-
const entries = await fs$1.readdir(outputPath);
|
|
1495
|
-
// 过滤掉 .git 目录,允许在已初始化 git 的目录中创建项目
|
|
1496
|
-
const filteredEntries = entries.filter(entry => entry !== '.git');
|
|
1497
|
-
return filteredEntries.length === 0;
|
|
1498
|
-
} catch (e) {
|
|
1499
|
-
// 目录不存在,视为空
|
|
1500
|
-
return true;
|
|
1501
|
-
}
|
|
1502
|
-
};
|
|
1503
|
-
|
|
1504
|
-
/**
|
|
1505
|
-
* 验证输出目录
|
|
1506
|
-
*
|
|
1507
|
-
* @param outputPath - 输出目录路径
|
|
1508
|
-
* @throws 如果目录不为空则抛出错误
|
|
1509
|
-
*/
|
|
1510
|
-
const validateOutputDir = async (outputPath) => {
|
|
1511
|
-
const isEmpty = await isOutputDirEmpty(outputPath);
|
|
1512
|
-
if (!isEmpty) {
|
|
1513
|
-
throw new Error(
|
|
1514
|
-
`Output directory is not empty: ${outputPath}\n` +
|
|
1515
|
-
'Please use an empty directory or remove existing files.',
|
|
1516
|
-
);
|
|
1517
|
-
}
|
|
1888
|
+
// node_modules 将由 pnpm install 处理(利用缓存和硬链接机制)
|
|
1518
1889
|
};
|
|
1890
|
+
// end_aigc
|
|
1519
1891
|
|
|
1520
1892
|
/**
|
|
1521
1893
|
* 模板引擎执行选项
|
|
@@ -1595,15 +1967,40 @@ const executeAfterRenderHook = async (
|
|
|
1595
1967
|
}
|
|
1596
1968
|
};
|
|
1597
1969
|
|
|
1970
|
+
/**
|
|
1971
|
+
* 执行完成钩子
|
|
1972
|
+
*/
|
|
1973
|
+
const executeCompleteHook = async (
|
|
1974
|
+
templateConfig,
|
|
1975
|
+
context,
|
|
1976
|
+
outputPath,
|
|
1977
|
+
) => {
|
|
1978
|
+
if (templateConfig.onComplete) {
|
|
1979
|
+
await templateConfig.onComplete(context, outputPath);
|
|
1980
|
+
}
|
|
1981
|
+
};
|
|
1982
|
+
|
|
1598
1983
|
/**
|
|
1599
1984
|
* 准备输出目录
|
|
1600
1985
|
*/
|
|
1601
|
-
const prepareOutputDirectory =
|
|
1986
|
+
const prepareOutputDirectory = (outputPath) => {
|
|
1602
1987
|
const absolutePath = path.resolve(process.cwd(), outputPath);
|
|
1603
|
-
|
|
1988
|
+
// 不再在这里验证目录是否为空,冲突检测已移至 processTemplateFiles 中
|
|
1604
1989
|
return absolutePath;
|
|
1605
1990
|
};
|
|
1606
1991
|
|
|
1992
|
+
/**
|
|
1993
|
+
* 模板引擎执行结果
|
|
1994
|
+
*/
|
|
1995
|
+
|
|
1996
|
+
|
|
1997
|
+
|
|
1998
|
+
|
|
1999
|
+
|
|
2000
|
+
|
|
2001
|
+
|
|
2002
|
+
|
|
2003
|
+
|
|
1607
2004
|
/**
|
|
1608
2005
|
* 执行完整的模板渲染流程
|
|
1609
2006
|
*/
|
|
@@ -1627,7 +2024,7 @@ const execute = async (
|
|
|
1627
2024
|
});
|
|
1628
2025
|
|
|
1629
2026
|
// 5. 准备输出目录
|
|
1630
|
-
const absoluteOutputPath =
|
|
2027
|
+
const absoluteOutputPath = prepareOutputDirectory(outputPath);
|
|
1631
2028
|
|
|
1632
2029
|
// 6. 处理模板文件
|
|
1633
2030
|
await processTemplateFiles({
|
|
@@ -1640,7 +2037,11 @@ const execute = async (
|
|
|
1640
2037
|
// 7. 执行 onAfterRender 钩子
|
|
1641
2038
|
await executeAfterRenderHook(templateConfig, context, absoluteOutputPath);
|
|
1642
2039
|
|
|
1643
|
-
return
|
|
2040
|
+
return {
|
|
2041
|
+
outputPath: absoluteOutputPath,
|
|
2042
|
+
templateConfig,
|
|
2043
|
+
context,
|
|
2044
|
+
};
|
|
1644
2045
|
};
|
|
1645
2046
|
|
|
1646
2047
|
function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } }
|
|
@@ -1803,32 +2204,27 @@ const executeInit = async (
|
|
|
1803
2204
|
logger.info(`Initializing project with template: ${templateName}`);
|
|
1804
2205
|
timer.logPhase('Initialization');
|
|
1805
2206
|
|
|
1806
|
-
//
|
|
1807
|
-
const
|
|
2207
|
+
// 执行模板引擎,返回结果对象
|
|
2208
|
+
const result = await execute({
|
|
1808
2209
|
templateName,
|
|
1809
2210
|
outputPath,
|
|
1810
2211
|
command,
|
|
1811
2212
|
});
|
|
2213
|
+
const { outputPath: absoluteOutputPath, templateConfig, context } = result;
|
|
1812
2214
|
|
|
1813
2215
|
timer.logPhase('Template engine execution');
|
|
1814
2216
|
logger.success('Project created successfully!');
|
|
1815
2217
|
|
|
1816
|
-
//
|
|
2218
|
+
// 安装依赖(始终使用 pnpm install,利用缓存机制)
|
|
1817
2219
|
if (!skipInstall) {
|
|
1818
|
-
|
|
1819
|
-
|
|
1820
|
-
|
|
1821
|
-
if (hasNodeModules) {
|
|
1822
|
-
logger.info(
|
|
1823
|
-
'\n💡 Using pre-warmed node_modules, skipping pnpm install',
|
|
1824
|
-
);
|
|
1825
|
-
timer.logPhase('Node modules (pre-warmed)');
|
|
1826
|
-
} else {
|
|
1827
|
-
runPnpmInstall(absoluteOutputPath);
|
|
1828
|
-
timer.logPhase('Dependencies installation');
|
|
1829
|
-
}
|
|
2220
|
+
runPnpmInstall(absoluteOutputPath);
|
|
2221
|
+
timer.logPhase('Dependencies installation');
|
|
1830
2222
|
}
|
|
1831
2223
|
|
|
2224
|
+
// 执行 onComplete 钩子(在 pnpm install 之后)
|
|
2225
|
+
await executeCompleteHook(templateConfig, context, absoluteOutputPath);
|
|
2226
|
+
timer.logPhase('Complete hook execution');
|
|
2227
|
+
|
|
1832
2228
|
// 如果没有跳过 git,则初始化 git 仓库
|
|
1833
2229
|
if (!skipGit) {
|
|
1834
2230
|
runGitInit(absoluteOutputPath);
|
|
@@ -1867,7 +2263,7 @@ const executeInit = async (
|
|
|
1867
2263
|
/**
|
|
1868
2264
|
* 注册 init 命令到 program
|
|
1869
2265
|
*/
|
|
1870
|
-
const registerCommand = program => {
|
|
2266
|
+
const registerCommand$1 = program => {
|
|
1871
2267
|
program
|
|
1872
2268
|
.command('init')
|
|
1873
2269
|
.description('Initialize a new project from a template')
|
|
@@ -1885,14 +2281,312 @@ const registerCommand = program => {
|
|
|
1885
2281
|
});
|
|
1886
2282
|
};
|
|
1887
2283
|
|
|
1888
|
-
|
|
2284
|
+
// ABOUTME: This file implements the update command for coze CLI
|
|
2285
|
+
// ABOUTME: It wraps pnpm update/install to update package dependencies with logging support
|
|
2286
|
+
|
|
2287
|
+
|
|
2288
|
+
|
|
2289
|
+
|
|
2290
|
+
/**
|
|
2291
|
+
* 日志文件名常量
|
|
2292
|
+
*/
|
|
2293
|
+
const LOG_FILE_NAME = 'update.log';
|
|
2294
|
+
|
|
2295
|
+
/**
|
|
2296
|
+
* 获取日志目录
|
|
2297
|
+
* 优先使用环境变量 COZE_LOG_DIR,否则使用 ~/.coze-logs
|
|
2298
|
+
*/
|
|
2299
|
+
const getLogDir = () =>
|
|
2300
|
+
process.env.COZE_LOG_DIR || path.join(os.homedir(), '.coze-logs');
|
|
2301
|
+
|
|
2302
|
+
/**
|
|
2303
|
+
* 解析日志文件路径
|
|
2304
|
+
* - 如果是绝对路径,直接使用
|
|
2305
|
+
* - 如果是相对路径,基于 getLogDir() + 相对路径
|
|
2306
|
+
* - 如果为空,使用 getLogDir() + LOG_FILE_NAME
|
|
2307
|
+
*/
|
|
2308
|
+
const resolveLogFilePath = (logFile) => {
|
|
2309
|
+
if (!logFile) {
|
|
2310
|
+
return path.join(getLogDir(), LOG_FILE_NAME);
|
|
2311
|
+
}
|
|
2312
|
+
|
|
2313
|
+
if (path.isAbsolute(logFile)) {
|
|
2314
|
+
return logFile;
|
|
2315
|
+
}
|
|
2316
|
+
|
|
2317
|
+
return path.join(getLogDir(), logFile);
|
|
2318
|
+
};
|
|
2319
|
+
|
|
2320
|
+
/**
|
|
2321
|
+
* 创建日志写入流
|
|
2322
|
+
*/
|
|
2323
|
+
const createLogStream = (logFilePath) => {
|
|
2324
|
+
const logDir = path.dirname(logFilePath);
|
|
2325
|
+
|
|
2326
|
+
// 确保日志目录存在
|
|
2327
|
+
if (!fs.existsSync(logDir)) {
|
|
2328
|
+
fs.mkdirSync(logDir, { recursive: true });
|
|
2329
|
+
}
|
|
2330
|
+
|
|
2331
|
+
// 使用 'w' 标志覆盖之前的日志
|
|
2332
|
+
return fs.createWriteStream(logFilePath, { flags: 'w' });
|
|
2333
|
+
};
|
|
2334
|
+
|
|
2335
|
+
/**
|
|
2336
|
+
* 格式化时间戳
|
|
2337
|
+
*/
|
|
2338
|
+
const formatTimestamp = () => {
|
|
2339
|
+
const now = new Date();
|
|
2340
|
+
return now.toISOString();
|
|
2341
|
+
};
|
|
2342
|
+
|
|
2343
|
+
/**
|
|
2344
|
+
* 写入带时间戳的日志
|
|
2345
|
+
*/
|
|
2346
|
+
const writeLogWithTimestamp = (stream, message) => {
|
|
2347
|
+
const timestamp = formatTimestamp();
|
|
2348
|
+
const lines = message.split('\n');
|
|
2349
|
+
lines.forEach(line => {
|
|
2350
|
+
if (line) {
|
|
2351
|
+
stream.write(`[${timestamp}] ${line}\n`);
|
|
2352
|
+
} else {
|
|
2353
|
+
stream.write('\n');
|
|
2354
|
+
}
|
|
2355
|
+
});
|
|
2356
|
+
// 确保数据写入磁盘
|
|
2357
|
+
stream.uncork();
|
|
2358
|
+
};
|
|
2359
|
+
|
|
2360
|
+
/**
|
|
2361
|
+
* 同时输出到控制台和日志文件
|
|
2362
|
+
*/
|
|
2363
|
+
const logWithFile = (
|
|
2364
|
+
stream,
|
|
2365
|
+
level,
|
|
2366
|
+
message,
|
|
2367
|
+
) => {
|
|
2368
|
+
// 输出到控制台
|
|
2369
|
+
switch (level) {
|
|
2370
|
+
case 'info':
|
|
2371
|
+
logger.info(message);
|
|
2372
|
+
break;
|
|
2373
|
+
case 'success':
|
|
2374
|
+
logger.success(message);
|
|
2375
|
+
break;
|
|
2376
|
+
case 'error':
|
|
2377
|
+
logger.error(message);
|
|
2378
|
+
break;
|
|
2379
|
+
default:
|
|
2380
|
+
logger.info(message);
|
|
2381
|
+
break;
|
|
2382
|
+
}
|
|
2383
|
+
|
|
2384
|
+
// 写入日志文件(带时间戳)
|
|
2385
|
+
writeLogWithTimestamp(stream, `[${level.toUpperCase()}] ${message}`);
|
|
2386
|
+
};
|
|
2387
|
+
|
|
2388
|
+
// start_aigc
|
|
2389
|
+
/**
|
|
2390
|
+
* 构建 pnpm add 命令
|
|
2391
|
+
*/
|
|
2392
|
+
const buildPnpmCommand = (
|
|
2393
|
+
packageName,
|
|
2394
|
+
options
|
|
2395
|
+
|
|
2396
|
+
|
|
2397
|
+
|
|
2398
|
+
|
|
2399
|
+
,
|
|
2400
|
+
) => {
|
|
2401
|
+
const { global, version, registry, extraArgs } = options;
|
|
2402
|
+
|
|
2403
|
+
const parts = ['pnpm', 'add'];
|
|
2404
|
+
|
|
2405
|
+
// 添加全局标记
|
|
2406
|
+
if (global) {
|
|
2407
|
+
parts.push('-g');
|
|
2408
|
+
}
|
|
2409
|
+
|
|
2410
|
+
// 添加包名和版本
|
|
2411
|
+
if (version && version !== 'latest') {
|
|
2412
|
+
parts.push(`${packageName}@${version}`);
|
|
2413
|
+
} else {
|
|
2414
|
+
parts.push(`${packageName}@latest`);
|
|
2415
|
+
}
|
|
2416
|
+
|
|
2417
|
+
// 添加 registry
|
|
2418
|
+
if (registry) {
|
|
2419
|
+
parts.push(`--registry=${registry}`);
|
|
2420
|
+
}
|
|
2421
|
+
|
|
2422
|
+
// 添加额外参数
|
|
2423
|
+
if (extraArgs.length > 0) {
|
|
2424
|
+
parts.push(...extraArgs);
|
|
2425
|
+
}
|
|
2426
|
+
|
|
2427
|
+
return parts.join(' ');
|
|
2428
|
+
};
|
|
2429
|
+
// end_aigc
|
|
2430
|
+
|
|
2431
|
+
// start_aigc
|
|
2432
|
+
/**
|
|
2433
|
+
* 执行 update 命令的内部实现
|
|
2434
|
+
*/
|
|
2435
|
+
const executeUpdate = (
|
|
2436
|
+
packageName,
|
|
2437
|
+
options
|
|
2438
|
+
|
|
2439
|
+
|
|
2440
|
+
|
|
2441
|
+
|
|
2442
|
+
|
|
2443
|
+
|
|
2444
|
+
,
|
|
2445
|
+
) => {
|
|
2446
|
+
let logStream = null;
|
|
2447
|
+
|
|
2448
|
+
try {
|
|
2449
|
+
const { global, cwd, version, registry, logFile, extraArgs } = options;
|
|
2450
|
+
|
|
2451
|
+
// 准备日志
|
|
2452
|
+
const logFilePath = resolveLogFilePath(logFile);
|
|
2453
|
+
|
|
2454
|
+
// 调试:确认日志路径
|
|
2455
|
+
logger.info(`Log file path resolved to: ${logFilePath}`);
|
|
2456
|
+
|
|
2457
|
+
logStream = createLogStream(logFilePath);
|
|
2458
|
+
|
|
2459
|
+
// 调试:确认流已创建
|
|
2460
|
+
logger.info('Log stream created successfully');
|
|
2461
|
+
|
|
2462
|
+
logWithFile(logStream, 'info', `Updating package: ${packageName}`);
|
|
2463
|
+
|
|
2464
|
+
// 构建命令
|
|
2465
|
+
const command = buildPnpmCommand(packageName, {
|
|
2466
|
+
global,
|
|
2467
|
+
version,
|
|
2468
|
+
registry,
|
|
2469
|
+
extraArgs,
|
|
2470
|
+
});
|
|
2471
|
+
|
|
2472
|
+
// 确定工作目录
|
|
2473
|
+
const workingDir = cwd
|
|
2474
|
+
? path.isAbsolute(cwd)
|
|
2475
|
+
? cwd
|
|
2476
|
+
: path.join(process.cwd(), cwd)
|
|
2477
|
+
: process.cwd();
|
|
2478
|
+
|
|
2479
|
+
logWithFile(logStream, 'info', `Executing: ${command}`);
|
|
2480
|
+
logWithFile(logStream, 'info', `Working directory: ${workingDir}`);
|
|
2481
|
+
logWithFile(logStream, 'info', `Log file: ${logFilePath}`);
|
|
2482
|
+
|
|
2483
|
+
// 记录命令开始时间
|
|
2484
|
+
writeLogWithTimestamp(logStream, '--- Command execution started ---');
|
|
2485
|
+
|
|
2486
|
+
// 同步执行命令
|
|
2487
|
+
const result = shelljs.exec(command, {
|
|
2488
|
+
cwd: workingDir,
|
|
2489
|
+
silent: true, // 使用 silent 来捕获输出
|
|
2490
|
+
});
|
|
2491
|
+
|
|
2492
|
+
// 将输出写入控制台和日志文件(带时间戳)
|
|
2493
|
+
if (result.stdout) {
|
|
2494
|
+
process.stdout.write(result.stdout);
|
|
2495
|
+
writeLogWithTimestamp(logStream, result.stdout.trim());
|
|
2496
|
+
}
|
|
2497
|
+
|
|
2498
|
+
if (result.stderr) {
|
|
2499
|
+
process.stderr.write(result.stderr);
|
|
2500
|
+
writeLogWithTimestamp(logStream, result.stderr.trim());
|
|
2501
|
+
}
|
|
2502
|
+
|
|
2503
|
+
// 记录命令结束时间
|
|
2504
|
+
writeLogWithTimestamp(logStream, '--- Command execution ended ---');
|
|
2505
|
+
|
|
2506
|
+
// 检查执行结果并记录到日志
|
|
2507
|
+
if (result.code === 0) {
|
|
2508
|
+
logWithFile(logStream, 'success', 'Package updated successfully');
|
|
2509
|
+
logWithFile(logStream, 'info', `Log file: ${logFilePath}`);
|
|
2510
|
+
} else {
|
|
2511
|
+
logWithFile(
|
|
2512
|
+
logStream,
|
|
2513
|
+
'error',
|
|
2514
|
+
`Command exited with code ${result.code}`,
|
|
2515
|
+
);
|
|
2516
|
+
logWithFile(
|
|
2517
|
+
logStream,
|
|
2518
|
+
'error',
|
|
2519
|
+
`Check log file for details: ${logFilePath}`,
|
|
2520
|
+
);
|
|
2521
|
+
}
|
|
2522
|
+
|
|
2523
|
+
// 关闭日志流并等待写入完成
|
|
2524
|
+
logStream.end(() => {
|
|
2525
|
+
// 流关闭后再退出进程
|
|
2526
|
+
if (result.code !== 0) {
|
|
2527
|
+
process.exit(result.code || 1);
|
|
2528
|
+
}
|
|
2529
|
+
});
|
|
2530
|
+
} catch (error) {
|
|
2531
|
+
logger.error('Failed to update package:');
|
|
2532
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
2533
|
+
|
|
2534
|
+
// 写入错误到日志文件
|
|
2535
|
+
if (logStream) {
|
|
2536
|
+
writeLogWithTimestamp(
|
|
2537
|
+
logStream,
|
|
2538
|
+
`[ERROR] ${error instanceof Error ? error.message : String(error)}`,
|
|
2539
|
+
);
|
|
2540
|
+
// 等待流关闭后再退出
|
|
2541
|
+
logStream.end(() => {
|
|
2542
|
+
process.exit(1);
|
|
2543
|
+
});
|
|
2544
|
+
} else {
|
|
2545
|
+
process.exit(1);
|
|
2546
|
+
}
|
|
2547
|
+
}
|
|
2548
|
+
};
|
|
2549
|
+
// end_aigc
|
|
2550
|
+
|
|
2551
|
+
/**
|
|
2552
|
+
* 注册 update 命令到 program
|
|
2553
|
+
*/
|
|
2554
|
+
const registerCommand = program => {
|
|
2555
|
+
program
|
|
2556
|
+
.command('update <package>')
|
|
2557
|
+
.description('Update a package dependency')
|
|
2558
|
+
.option('-g, --global', 'Update package globally', false)
|
|
2559
|
+
.option('-c, --cwd <path>', 'Working directory for the update')
|
|
2560
|
+
.option(
|
|
2561
|
+
'--to <version>',
|
|
2562
|
+
'Version to update to (default: latest)',
|
|
2563
|
+
'latest',
|
|
2564
|
+
)
|
|
2565
|
+
.option('--registry <url>', 'Registry URL to use for the update')
|
|
2566
|
+
.option('--log-file <path>', 'Log file path')
|
|
2567
|
+
.allowUnknownOption() // 允许透传参数给 pnpm
|
|
2568
|
+
.action((packageName, options, command) => {
|
|
2569
|
+
// 收集所有未知选项作为额外参数
|
|
2570
|
+
const extraArgs = command.args.slice(1);
|
|
2571
|
+
|
|
2572
|
+
executeUpdate(packageName, {
|
|
2573
|
+
...options,
|
|
2574
|
+
version: options.to, // 将 --to 映射到 version
|
|
2575
|
+
extraArgs,
|
|
2576
|
+
});
|
|
2577
|
+
});
|
|
2578
|
+
};
|
|
2579
|
+
|
|
2580
|
+
var version = "0.0.1-alpha.cf9a3b";
|
|
1889
2581
|
var packageJson = {
|
|
1890
2582
|
version: version};
|
|
1891
2583
|
|
|
1892
2584
|
const commands = [
|
|
1893
|
-
registerCommand,
|
|
1894
2585
|
registerCommand$1,
|
|
1895
2586
|
registerCommand$2,
|
|
2587
|
+
registerCommand$4,
|
|
2588
|
+
registerCommand$3,
|
|
2589
|
+
registerCommand,
|
|
1896
2590
|
];
|
|
1897
2591
|
|
|
1898
2592
|
const main = () => {
|