@coze-arch/cli 0.0.1-alpha.afefde → 0.0.1-alpha.b287c1
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 +30 -21
- package/lib/__templates__/expo/.cozeproj/scripts/prod_build.sh +2 -2
- package/lib/__templates__/expo/.cozeproj/scripts/prod_run.sh +2 -2
- package/lib/__templates__/expo/.cozeproj/scripts/server_dev_run.sh +46 -0
- package/lib/__templates__/expo/AGENTS.md +74 -0
- package/lib/__templates__/expo/README.md +0 -71
- package/lib/__templates__/expo/_npmrc +1 -0
- package/lib/__templates__/expo/client/app/+not-found.tsx +30 -0
- package/lib/__templates__/expo/client/app/_layout.tsx +15 -12
- package/lib/__templates__/expo/client/app/index.tsx +1 -1
- package/lib/__templates__/expo/client/app.config.ts +4 -3
- package/lib/__templates__/expo/client/components/Screen.tsx +3 -19
- 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/declarations.d.ts +5 -0
- package/lib/__templates__/expo/client/eslint.config.mjs +40 -10
- 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/hooks/useTheme.ts +26 -6
- 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/{home → demo}/index.tsx +4 -4
- package/lib/__templates__/expo/client/scripts/install-missing-deps.js +11 -10
- package/lib/__templates__/expo/client/utils/index.ts +22 -0
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/names.js +1889 -0
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/rule.js +174 -0
- package/lib/__templates__/expo/eslint-plugins/fontawesome6/v5-only-names.js +388 -0
- package/lib/__templates__/expo/eslint-plugins/forbid-emoji/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/forbid-emoji/rule.js +112 -0
- package/lib/__templates__/expo/eslint-plugins/forbid-emoji/tech.md +94 -0
- package/lib/__templates__/expo/eslint-plugins/react-native/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/react-native/rule.js +64 -0
- package/lib/__templates__/expo/eslint-plugins/reanimated/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/reanimated/rule.js +88 -0
- package/lib/__templates__/expo/eslint-plugins/restrict-linear-gradient/index.js +9 -0
- package/lib/__templates__/expo/eslint-plugins/restrict-linear-gradient/rule.js +120 -0
- package/lib/__templates__/expo/eslint-plugins/restrict-linear-gradient/tech.md +58 -0
- package/lib/__templates__/expo/package.json +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/build.js +21 -0
- package/lib/__templates__/expo/server/package.json +9 -7
- package/lib/__templates__/expo/server/src/index.ts +3 -1
- 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/.babelrc +15 -0
- package/lib/__templates__/nextjs/AGENTS.md +54 -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 +2 -3
- package/lib/__templates__/nextjs/package.json +8 -1
- package/lib/__templates__/nextjs/pnpm-lock.yaml +3472 -1561
- package/lib/__templates__/nextjs/scripts/build.sh +4 -1
- package/lib/__templates__/nextjs/scripts/dev.sh +8 -2
- package/lib/__templates__/nextjs/scripts/start.sh +7 -1
- package/lib/__templates__/nextjs/src/app/layout.tsx +5 -3
- package/lib/__templates__/nextjs/src/app/page.tsx +17 -60
- package/lib/__templates__/nextjs/src/server.ts +35 -0
- package/lib/__templates__/nextjs/template.config.js +47 -12
- package/lib/__templates__/nextjs/tsconfig.json +1 -1
- package/lib/__templates__/nuxt-vue/.coze +12 -0
- package/lib/__templates__/nuxt-vue/AGENTS.md +42 -0
- package/lib/__templates__/nuxt-vue/README.md +73 -0
- package/lib/__templates__/nuxt-vue/_gitignore +24 -0
- package/lib/__templates__/nuxt-vue/_npmrc +23 -0
- package/lib/__templates__/nuxt-vue/app/app.vue +6 -0
- package/lib/__templates__/nuxt-vue/app/pages/index.vue +23 -0
- package/lib/__templates__/nuxt-vue/assets/css/main.css +24 -0
- package/lib/__templates__/nuxt-vue/nuxt.config.ts +116 -0
- package/lib/__templates__/nuxt-vue/package.json +35 -0
- package/lib/__templates__/nuxt-vue/pnpm-lock.yaml +8759 -0
- package/lib/__templates__/nuxt-vue/postcss.config.mjs +8 -0
- package/lib/__templates__/nuxt-vue/public/favicon.ico +0 -0
- package/lib/__templates__/nuxt-vue/public/robots.txt +2 -0
- package/lib/__templates__/nuxt-vue/scripts/build.sh +14 -0
- package/lib/__templates__/nuxt-vue/scripts/dev.sh +39 -0
- package/lib/__templates__/nuxt-vue/scripts/prepare.sh +14 -0
- package/lib/__templates__/nuxt-vue/scripts/start.sh +21 -0
- package/lib/__templates__/nuxt-vue/server/api/hello.ts +10 -0
- package/lib/__templates__/nuxt-vue/server/middleware/logger.ts +10 -0
- package/lib/__templates__/nuxt-vue/server/routes/health.ts +10 -0
- package/lib/__templates__/nuxt-vue/tailwind.config.js +13 -0
- package/lib/__templates__/nuxt-vue/template.config.js +87 -0
- package/lib/__templates__/nuxt-vue/tsconfig.json +18 -0
- package/lib/__templates__/taro/.coze +14 -0
- package/lib/__templates__/taro/.cozeproj/scripts/deploy_build.sh +19 -0
- package/lib/__templates__/taro/.cozeproj/scripts/deploy_run.sh +14 -0
- package/lib/__templates__/taro/.cozeproj/scripts/dev_build.sh +2 -0
- package/lib/__templates__/taro/.cozeproj/scripts/dev_run.sh +151 -0
- package/lib/__templates__/taro/.cozeproj/scripts/init_env.sh +5 -0
- package/lib/__templates__/taro/.cozeproj/scripts/pack.sh +24 -0
- package/lib/__templates__/taro/README.md +763 -0
- package/lib/__templates__/taro/_gitignore +40 -0
- package/lib/__templates__/taro/_npmrc +18 -0
- package/lib/__templates__/taro/babel.config.js +12 -0
- package/lib/__templates__/taro/config/dev.ts +9 -0
- package/lib/__templates__/taro/config/index.ts +238 -0
- package/lib/__templates__/taro/config/prod.ts +34 -0
- package/lib/__templates__/taro/eslint.config.mjs +135 -0
- package/lib/__templates__/taro/key/private.appid.key +0 -0
- package/lib/__templates__/taro/package.json +112 -0
- package/lib/__templates__/taro/patches/@tarojs__plugin-mini-ci@4.1.9.patch +30 -0
- package/lib/__templates__/taro/pnpm-lock.yaml +23412 -0
- package/lib/__templates__/taro/pnpm-workspace.yaml +2 -0
- package/lib/__templates__/taro/project.config.json +15 -0
- package/lib/__templates__/taro/server/nest-cli.json +10 -0
- package/lib/__templates__/taro/server/package.json +40 -0
- package/lib/__templates__/taro/server/src/app.controller.ts +23 -0
- package/lib/__templates__/taro/server/src/app.module.ts +10 -0
- package/lib/__templates__/taro/server/src/app.service.ts +8 -0
- package/lib/__templates__/taro/server/src/interceptors/http-status.interceptor.ts +23 -0
- package/lib/__templates__/taro/server/src/main.ts +49 -0
- package/lib/__templates__/taro/server/tsconfig.json +24 -0
- package/lib/__templates__/taro/src/app.config.ts +11 -0
- package/lib/__templates__/taro/src/app.css +156 -0
- package/lib/__templates__/taro/src/app.tsx +9 -0
- package/lib/__templates__/taro/src/components/ui/accordion.tsx +159 -0
- package/lib/__templates__/taro/src/components/ui/alert-dialog.tsx +260 -0
- package/lib/__templates__/taro/src/components/ui/alert.tsx +60 -0
- package/lib/__templates__/taro/src/components/ui/aspect-ratio.tsx +36 -0
- package/lib/__templates__/taro/src/components/ui/avatar.tsx +84 -0
- package/lib/__templates__/taro/src/components/ui/badge.tsx +37 -0
- package/lib/__templates__/taro/src/components/ui/breadcrumb.tsx +117 -0
- package/lib/__templates__/taro/src/components/ui/button-group.tsx +83 -0
- package/lib/__templates__/taro/src/components/ui/button.tsx +67 -0
- package/lib/__templates__/taro/src/components/ui/calendar.tsx +394 -0
- package/lib/__templates__/taro/src/components/ui/card.tsx +108 -0
- package/lib/__templates__/taro/src/components/ui/carousel.tsx +228 -0
- package/lib/__templates__/taro/src/components/ui/checkbox.tsx +58 -0
- package/lib/__templates__/taro/src/components/ui/code-block.tsx +169 -0
- package/lib/__templates__/taro/src/components/ui/collapsible.tsx +71 -0
- package/lib/__templates__/taro/src/components/ui/command.tsx +385 -0
- package/lib/__templates__/taro/src/components/ui/context-menu.tsx +614 -0
- package/lib/__templates__/taro/src/components/ui/dialog.tsx +256 -0
- package/lib/__templates__/taro/src/components/ui/drawer.tsx +192 -0
- package/lib/__templates__/taro/src/components/ui/dropdown-menu.tsx +561 -0
- package/lib/__templates__/taro/src/components/ui/field.tsx +228 -0
- package/lib/__templates__/taro/src/components/ui/hover-card.tsx +282 -0
- package/lib/__templates__/taro/src/components/ui/input-group.tsx +197 -0
- package/lib/__templates__/taro/src/components/ui/input-otp.tsx +136 -0
- package/lib/__templates__/taro/src/components/ui/input.tsx +56 -0
- package/lib/__templates__/taro/src/components/ui/label.tsx +24 -0
- package/lib/__templates__/taro/src/components/ui/menubar.tsx +595 -0
- package/lib/__templates__/taro/src/components/ui/navigation-menu.tsx +264 -0
- package/lib/__templates__/taro/src/components/ui/pagination.tsx +118 -0
- package/lib/__templates__/taro/src/components/ui/popover.tsx +291 -0
- package/lib/__templates__/taro/src/components/ui/portal.tsx +19 -0
- package/lib/__templates__/taro/src/components/ui/progress.tsx +28 -0
- package/lib/__templates__/taro/src/components/ui/radio-group.tsx +64 -0
- package/lib/__templates__/taro/src/components/ui/resizable.tsx +346 -0
- package/lib/__templates__/taro/src/components/ui/scroll-area.tsx +34 -0
- package/lib/__templates__/taro/src/components/ui/select.tsx +438 -0
- package/lib/__templates__/taro/src/components/ui/separator.tsx +30 -0
- package/lib/__templates__/taro/src/components/ui/sheet.tsx +262 -0
- package/lib/__templates__/taro/src/components/ui/skeleton.tsx +17 -0
- package/lib/__templates__/taro/src/components/ui/slider.tsx +203 -0
- package/lib/__templates__/taro/src/components/ui/sonner.tsx +1 -0
- package/lib/__templates__/taro/src/components/ui/switch.tsx +55 -0
- package/lib/__templates__/taro/src/components/ui/table.tsx +142 -0
- package/lib/__templates__/taro/src/components/ui/tabs.tsx +114 -0
- package/lib/__templates__/taro/src/components/ui/textarea.tsx +54 -0
- package/lib/__templates__/taro/src/components/ui/toast.tsx +517 -0
- package/lib/__templates__/taro/src/components/ui/toggle-group.tsx +120 -0
- package/lib/__templates__/taro/src/components/ui/toggle.tsx +77 -0
- package/lib/__templates__/taro/src/components/ui/tooltip.tsx +455 -0
- package/lib/__templates__/taro/src/index.html +39 -0
- package/lib/__templates__/taro/src/lib/hooks/use-keyboard-offset.ts +37 -0
- package/lib/__templates__/taro/src/lib/measure.ts +115 -0
- package/lib/__templates__/taro/src/lib/platform.ts +12 -0
- package/lib/__templates__/taro/src/lib/utils.ts +6 -0
- package/lib/__templates__/taro/src/network.ts +39 -0
- package/lib/__templates__/taro/src/pages/index/index.config.ts +3 -0
- package/lib/__templates__/taro/src/pages/index/index.css +1 -0
- package/lib/__templates__/taro/src/pages/index/index.tsx +33 -0
- package/lib/__templates__/taro/src/presets/dev-debug.ts +23 -0
- package/lib/__templates__/taro/src/presets/h5-container.tsx +15 -0
- package/lib/__templates__/taro/src/presets/h5-navbar.tsx +238 -0
- package/lib/__templates__/taro/src/presets/h5-styles.ts +220 -0
- package/lib/__templates__/taro/src/presets/index.tsx +18 -0
- package/lib/__templates__/taro/stylelint.config.mjs +4 -0
- package/lib/__templates__/taro/template.config.js +68 -0
- package/lib/__templates__/taro/tsconfig.json +29 -0
- package/lib/__templates__/taro/types/global.d.ts +32 -0
- package/lib/__templates__/templates.json +52 -14
- package/lib/__templates__/vite/AGENTS.md +41 -0
- package/lib/__templates__/vite/README.md +190 -11
- package/lib/__templates__/vite/_gitignore +1 -0
- package/lib/__templates__/vite/eslint.config.mjs +6 -1
- package/lib/__templates__/vite/package.json +18 -2
- package/lib/__templates__/vite/pnpm-lock.yaml +1079 -2330
- package/lib/__templates__/vite/scripts/build.sh +4 -1
- package/lib/__templates__/vite/scripts/dev.sh +9 -2
- package/lib/__templates__/vite/scripts/start.sh +9 -3
- package/lib/__templates__/vite/server/routes/index.ts +31 -0
- package/lib/__templates__/vite/server/server.ts +65 -0
- package/lib/__templates__/vite/server/vite.ts +67 -0
- package/lib/__templates__/vite/src/main.ts +17 -48
- package/lib/__templates__/vite/template.config.js +47 -10
- package/lib/__templates__/vite/tsconfig.json +4 -3
- package/lib/__templates__/vite/vite.config.ts +5 -0
- package/lib/cli.js +909 -187
- package/package.json +7 -3
- package/lib/__templates__/expo/client/app/home.tsx +0 -1
- /package/lib/__templates__/expo/client/screens/{home → demo}/styles.ts +0 -0
package/lib/cli.js
CHANGED
|
@@ -4,12 +4,14 @@
|
|
|
4
4
|
var commander = require('commander');
|
|
5
5
|
var path = require('path');
|
|
6
6
|
var fs = require('fs');
|
|
7
|
+
var node_path = require('node:path');
|
|
8
|
+
var node_fs = require('node:fs');
|
|
7
9
|
var shelljs = require('shelljs');
|
|
8
10
|
var perf_hooks = require('perf_hooks');
|
|
9
11
|
var fs$1 = require('fs/promises');
|
|
10
12
|
var os = require('os');
|
|
11
|
-
var toml = require('@iarna/toml');
|
|
12
13
|
var jsYaml = require('js-yaml');
|
|
14
|
+
var toml = require('@iarna/toml');
|
|
13
15
|
var child_process = require('child_process');
|
|
14
16
|
var addFormats = require('ajv-formats');
|
|
15
17
|
var Ajv = require('ajv');
|
|
@@ -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
|
};
|
|
@@ -727,7 +731,7 @@ const parseConfigContent = (content) => {
|
|
|
727
731
|
return config ;
|
|
728
732
|
} catch (error) {
|
|
729
733
|
// TOML 解析失败,继续尝试其他格式
|
|
730
|
-
|
|
734
|
+
|
|
731
735
|
console.debug('TOML parse failed:', error);
|
|
732
736
|
}
|
|
733
737
|
|
|
@@ -739,7 +743,7 @@ const parseConfigContent = (content) => {
|
|
|
739
743
|
}
|
|
740
744
|
} catch (error) {
|
|
741
745
|
// YAML 解析失败,继续尝试其他格式
|
|
742
|
-
|
|
746
|
+
|
|
743
747
|
console.debug('YAML parse failed:', error);
|
|
744
748
|
}
|
|
745
749
|
|
|
@@ -823,18 +827,242 @@ const getCommandConfig = (
|
|
|
823
827
|
return commandConfig;
|
|
824
828
|
};
|
|
825
829
|
|
|
826
|
-
|
|
830
|
+
// ABOUTME: Fix rule to comment out problematic outputFileTracingRoot config in Next.js projects
|
|
831
|
+
// ABOUTME: This config can cause issues in monorepo environments and should be removed
|
|
832
|
+
|
|
833
|
+
|
|
834
|
+
|
|
835
|
+
|
|
836
|
+
/**
|
|
837
|
+
* 检查是否为 Next.js 项目
|
|
838
|
+
*/
|
|
839
|
+
const isNextProject = (projectFolder) => {
|
|
840
|
+
const packageJsonPath = path.join(projectFolder, 'package.json');
|
|
841
|
+
|
|
842
|
+
if (!fs.existsSync(packageJsonPath)) {
|
|
843
|
+
return false;
|
|
844
|
+
}
|
|
845
|
+
|
|
846
|
+
try {
|
|
847
|
+
// eslint-disable-next-line no-restricted-syntax
|
|
848
|
+
const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf-8'));
|
|
849
|
+
const deps = {
|
|
850
|
+
...packageJson.dependencies,
|
|
851
|
+
...packageJson.devDependencies,
|
|
852
|
+
};
|
|
853
|
+
|
|
854
|
+
return 'next' in deps;
|
|
855
|
+
} catch (e) {
|
|
856
|
+
return false;
|
|
857
|
+
}
|
|
858
|
+
};
|
|
859
|
+
|
|
860
|
+
/**
|
|
861
|
+
* 查找 Next.js 配置文件
|
|
862
|
+
*/
|
|
863
|
+
const findNextConfigFile = (projectFolder) => {
|
|
864
|
+
const possibleConfigs = [
|
|
865
|
+
'next.config.ts',
|
|
866
|
+
'next.config.js',
|
|
867
|
+
'next.config.mjs',
|
|
868
|
+
];
|
|
869
|
+
|
|
870
|
+
for (const configFile of possibleConfigs) {
|
|
871
|
+
const configPath = path.join(projectFolder, configFile);
|
|
872
|
+
if (fs.existsSync(configPath)) {
|
|
873
|
+
return configPath;
|
|
874
|
+
}
|
|
875
|
+
}
|
|
876
|
+
|
|
877
|
+
return null;
|
|
878
|
+
};
|
|
879
|
+
|
|
880
|
+
/**
|
|
881
|
+
* 注释掉 outputFileTracingRoot 配置
|
|
882
|
+
*/
|
|
883
|
+
const commentOutOutputTracingRoot = (
|
|
884
|
+
configPath,
|
|
885
|
+
) => {
|
|
886
|
+
let content = fs.readFileSync(configPath, 'utf-8');
|
|
887
|
+
let modified = false;
|
|
888
|
+
let originalLine = null;
|
|
889
|
+
|
|
890
|
+
// 匹配包含 outputFileTracingRoot 的行(尚未被注释的)
|
|
891
|
+
// 支持 path.resolve(...) 后面有空格,然后可选逗号
|
|
892
|
+
// 只匹配单行配置,不匹配跨多行的配置
|
|
893
|
+
const pattern =
|
|
894
|
+
/^(\s*)(outputFileTracingRoot:\s*path\.resolve\([^\n\r)]+\)\s*,?)\s*$/gm;
|
|
895
|
+
|
|
896
|
+
const matches = content.match(pattern);
|
|
897
|
+
|
|
898
|
+
if (matches && matches.length > 0) {
|
|
899
|
+
originalLine = matches[0].trim();
|
|
900
|
+
|
|
901
|
+
// 在匹配的行前添加 // 注释
|
|
902
|
+
content = content.replace(pattern, '$1// $2');
|
|
903
|
+
modified = true;
|
|
904
|
+
|
|
905
|
+
fs.writeFileSync(configPath, content, 'utf-8');
|
|
906
|
+
}
|
|
907
|
+
|
|
908
|
+
return { modified, originalLine };
|
|
909
|
+
};
|
|
910
|
+
|
|
911
|
+
// start_aigc
|
|
912
|
+
/**
|
|
913
|
+
* Fix 规则:注释掉 Next.js 项目中的 outputFileTracingRoot 配置
|
|
914
|
+
* 这个配置在 monorepo 环境中可能会导致问题
|
|
915
|
+
*/
|
|
916
|
+
const fixNextOutputTracingRoot = context => {
|
|
917
|
+
const ruleName = 'next-output-tracing-root';
|
|
918
|
+
|
|
919
|
+
// 1. 检查是否为 Next.js 项目
|
|
920
|
+
if (!isNextProject(context.projectFolder)) {
|
|
921
|
+
return {
|
|
922
|
+
ruleName,
|
|
923
|
+
applied: false,
|
|
924
|
+
message: 'Not a Next.js project, skipping',
|
|
925
|
+
};
|
|
926
|
+
}
|
|
927
|
+
|
|
928
|
+
// 2. 查找 Next.js 配置文件
|
|
929
|
+
const configPath = findNextConfigFile(context.projectFolder);
|
|
930
|
+
|
|
931
|
+
if (!configPath) {
|
|
932
|
+
return {
|
|
933
|
+
ruleName,
|
|
934
|
+
applied: false,
|
|
935
|
+
message: 'Next.js config file not found, skipping',
|
|
936
|
+
};
|
|
937
|
+
}
|
|
938
|
+
|
|
939
|
+
// 3. 注释掉 outputFileTracingRoot 配置
|
|
940
|
+
const { modified, originalLine } = commentOutOutputTracingRoot(configPath);
|
|
941
|
+
|
|
942
|
+
if (modified && originalLine) {
|
|
943
|
+
logger.success(
|
|
944
|
+
`Commented out outputFileTracingRoot in ${configPath.split('/').pop()}`,
|
|
945
|
+
);
|
|
946
|
+
logger.info(` Original: ${originalLine}`);
|
|
947
|
+
|
|
948
|
+
return {
|
|
949
|
+
ruleName,
|
|
950
|
+
applied: true,
|
|
951
|
+
message: `Successfully commented out: ${originalLine}`,
|
|
952
|
+
};
|
|
953
|
+
}
|
|
954
|
+
|
|
955
|
+
return {
|
|
956
|
+
ruleName,
|
|
957
|
+
applied: false,
|
|
958
|
+
message: 'No outputFileTracingRoot config found, skipping',
|
|
959
|
+
};
|
|
960
|
+
};
|
|
961
|
+
// end_aigc
|
|
962
|
+
|
|
963
|
+
/**
|
|
964
|
+
* 所有修复规则的数组
|
|
965
|
+
* 按顺序执行,新增规则直接添加到数组中
|
|
966
|
+
*/
|
|
967
|
+
const rules = [
|
|
968
|
+
// Next.js related fixes
|
|
969
|
+
fixNextOutputTracingRoot,
|
|
970
|
+
|
|
971
|
+
// Add more rules here
|
|
972
|
+
] ;
|
|
973
|
+
|
|
974
|
+
// ABOUTME: Fix command for resolving legacy issues from previous project versions
|
|
975
|
+
// ABOUTME: Applies a series of fix rules defined in the fix-rules directory
|
|
976
|
+
|
|
977
|
+
|
|
978
|
+
// start_aigc
|
|
979
|
+
/**
|
|
980
|
+
* 执行 fix 命令的内部实现
|
|
981
|
+
*/
|
|
982
|
+
const executeFix = async (options = {}) => {
|
|
983
|
+
try {
|
|
984
|
+
const cwd = process.cwd();
|
|
985
|
+
const projectFolder = options.directory
|
|
986
|
+
? path.resolve(cwd, options.directory)
|
|
987
|
+
: cwd;
|
|
988
|
+
|
|
989
|
+
logger.info(`Running fix command on: ${projectFolder}`);
|
|
990
|
+
logger.info(`Found ${rules.length} fix rule(s) to apply\n`);
|
|
991
|
+
|
|
992
|
+
const context = {
|
|
993
|
+
cwd,
|
|
994
|
+
projectFolder,
|
|
995
|
+
};
|
|
996
|
+
|
|
997
|
+
let appliedCount = 0;
|
|
998
|
+
let skippedCount = 0;
|
|
999
|
+
|
|
1000
|
+
// 依次执行所有修复规则
|
|
1001
|
+
for (const rule of rules) {
|
|
1002
|
+
try {
|
|
1003
|
+
const result = await Promise.resolve(rule(context));
|
|
1004
|
+
|
|
1005
|
+
if (result.applied) {
|
|
1006
|
+
appliedCount++;
|
|
1007
|
+
logger.success(`✓ ${result.ruleName}: ${result.message}`);
|
|
1008
|
+
} else {
|
|
1009
|
+
skippedCount++;
|
|
1010
|
+
logger.info(`○ ${result.ruleName}: ${result.message}`);
|
|
1011
|
+
}
|
|
1012
|
+
} catch (error) {
|
|
1013
|
+
logger.error(
|
|
1014
|
+
`✗ Rule execution failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
1015
|
+
);
|
|
1016
|
+
}
|
|
1017
|
+
}
|
|
1018
|
+
|
|
1019
|
+
// 输出汇总信息
|
|
1020
|
+
logger.info(
|
|
1021
|
+
`\nSummary: ${appliedCount} fixed, ${skippedCount} skipped, ${rules.length} total`,
|
|
1022
|
+
);
|
|
1023
|
+
|
|
1024
|
+
if (appliedCount > 0) {
|
|
1025
|
+
logger.success('\nFixes applied successfully!');
|
|
1026
|
+
} else {
|
|
1027
|
+
logger.info('\nNo fixes needed');
|
|
1028
|
+
}
|
|
1029
|
+
} catch (error) {
|
|
1030
|
+
logger.error('Failed to run fix command:');
|
|
1031
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
1032
|
+
process.exit(1);
|
|
1033
|
+
}
|
|
1034
|
+
};
|
|
1035
|
+
// end_aigc
|
|
1036
|
+
|
|
1037
|
+
/**
|
|
1038
|
+
* 注册 fix 命令到 program
|
|
1039
|
+
*/
|
|
1040
|
+
const registerCommand$3 = program => {
|
|
1041
|
+
program
|
|
1042
|
+
.command('fix')
|
|
1043
|
+
.description(
|
|
1044
|
+
'Fix legacy issues from previous versions (e.g., problematic configs)',
|
|
1045
|
+
)
|
|
1046
|
+
.argument(
|
|
1047
|
+
'[directory]',
|
|
1048
|
+
'Target directory to fix (defaults to current directory)',
|
|
1049
|
+
)
|
|
1050
|
+
.action(async (directory) => {
|
|
1051
|
+
await executeFix({ directory });
|
|
1052
|
+
});
|
|
1053
|
+
};
|
|
827
1054
|
|
|
1055
|
+
function _nullishCoalesce$1(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }
|
|
828
1056
|
/**
|
|
829
1057
|
* 日志文件名常量
|
|
830
1058
|
*/
|
|
831
|
-
const LOG_FILE_NAME = 'dev.log';
|
|
1059
|
+
const LOG_FILE_NAME$1 = 'dev.log';
|
|
832
1060
|
|
|
833
1061
|
/**
|
|
834
1062
|
* 获取日志目录
|
|
835
1063
|
* 优先使用环境变量 COZE_LOG_DIR,否则使用 ~/.coze-logs
|
|
836
1064
|
*/
|
|
837
|
-
const getLogDir = () =>
|
|
1065
|
+
const getLogDir$1 = () =>
|
|
838
1066
|
process.env.COZE_LOG_DIR || path.join(os.homedir(), '.coze-logs');
|
|
839
1067
|
|
|
840
1068
|
/**
|
|
@@ -843,22 +1071,22 @@ const getLogDir = () =>
|
|
|
843
1071
|
* - 如果是相对路径,基于 getLogDir() + 相对路径
|
|
844
1072
|
* - 如果为空,使用 getLogDir() + LOG_FILE_NAME
|
|
845
1073
|
*/
|
|
846
|
-
const resolveLogFilePath = (logFile) => {
|
|
1074
|
+
const resolveLogFilePath$1 = (logFile) => {
|
|
847
1075
|
if (!logFile) {
|
|
848
|
-
return path.join(getLogDir(), LOG_FILE_NAME);
|
|
1076
|
+
return path.join(getLogDir$1(), LOG_FILE_NAME$1);
|
|
849
1077
|
}
|
|
850
1078
|
|
|
851
1079
|
if (path.isAbsolute(logFile)) {
|
|
852
1080
|
return logFile;
|
|
853
1081
|
}
|
|
854
1082
|
|
|
855
|
-
return path.join(getLogDir(), logFile);
|
|
1083
|
+
return path.join(getLogDir$1(), logFile);
|
|
856
1084
|
};
|
|
857
1085
|
|
|
858
1086
|
/**
|
|
859
1087
|
* 创建日志写入流
|
|
860
1088
|
*/
|
|
861
|
-
const createLogStream = (logFilePath) => {
|
|
1089
|
+
const createLogStream$1 = (logFilePath) => {
|
|
862
1090
|
const logDir = path.dirname(logFilePath);
|
|
863
1091
|
|
|
864
1092
|
// 确保日志目录存在
|
|
@@ -880,15 +1108,27 @@ const executeRun = async (
|
|
|
880
1108
|
try {
|
|
881
1109
|
logger.info(`Running ${commandName} command...`);
|
|
882
1110
|
|
|
883
|
-
// 1.
|
|
1111
|
+
// 1. 对于 build 命令,先执行 fix 以确保项目配置正确
|
|
1112
|
+
if (['dev', 'build'].includes(commandName)) {
|
|
1113
|
+
logger.info('\n🔧 Running fix command before build...\n');
|
|
1114
|
+
try {
|
|
1115
|
+
await executeFix();
|
|
1116
|
+
// eslint-disable-next-line @coze-arch/no-empty-catch
|
|
1117
|
+
} catch (e) {
|
|
1118
|
+
// just ignore
|
|
1119
|
+
}
|
|
1120
|
+
logger.info('');
|
|
1121
|
+
}
|
|
1122
|
+
|
|
1123
|
+
// 2. 加载 .coze 配置
|
|
884
1124
|
const config = await loadCozeConfig();
|
|
885
1125
|
const commandArgs = getCommandConfig(config, commandName);
|
|
886
1126
|
|
|
887
|
-
//
|
|
888
|
-
const logFilePath = resolveLogFilePath(options.logFile);
|
|
889
|
-
const logStream = createLogStream(logFilePath);
|
|
1127
|
+
// 3. 准备日志
|
|
1128
|
+
const logFilePath = resolveLogFilePath$1(options.logFile);
|
|
1129
|
+
const logStream = createLogStream$1(logFilePath);
|
|
890
1130
|
|
|
891
|
-
//
|
|
1131
|
+
// 4. 执行命令
|
|
892
1132
|
const commandString = commandArgs.join(' ');
|
|
893
1133
|
|
|
894
1134
|
logger.info(`Executing: ${commandString}`);
|
|
@@ -949,7 +1189,7 @@ const executeRun = async (
|
|
|
949
1189
|
/**
|
|
950
1190
|
* 注册 dev/build/start 命令到 program
|
|
951
1191
|
*/
|
|
952
|
-
const registerCommand$
|
|
1192
|
+
const registerCommand$2 = program => {
|
|
953
1193
|
// dev 命令
|
|
954
1194
|
program
|
|
955
1195
|
.command('dev')
|
|
@@ -1201,6 +1441,11 @@ const shouldIgnoreFile = (filePath) => {
|
|
|
1201
1441
|
return directoryPatterns.some(dir => pathParts.includes(dir));
|
|
1202
1442
|
};
|
|
1203
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
|
|
1204
1449
|
/**
|
|
1205
1450
|
* 递归获取目录中的所有文件
|
|
1206
1451
|
*
|
|
@@ -1270,7 +1515,19 @@ const convertDotfileName = (filePath) => {
|
|
|
1270
1515
|
|
|
1271
1516
|
return filePath;
|
|
1272
1517
|
};
|
|
1518
|
+
// end_aigc
|
|
1519
|
+
|
|
1520
|
+
// ABOUTME: File rendering utilities for template processing
|
|
1521
|
+
// ABOUTME: Handles file content rendering, hook execution, and file writing
|
|
1522
|
+
|
|
1523
|
+
|
|
1524
|
+
|
|
1525
|
+
|
|
1526
|
+
|
|
1273
1527
|
|
|
1528
|
+
|
|
1529
|
+
|
|
1530
|
+
// start_aigc
|
|
1274
1531
|
/**
|
|
1275
1532
|
* 执行文件渲染钩子
|
|
1276
1533
|
*
|
|
@@ -1316,22 +1573,25 @@ const executeFileRenderHook = async (
|
|
|
1316
1573
|
};
|
|
1317
1574
|
|
|
1318
1575
|
/**
|
|
1319
|
-
*
|
|
1576
|
+
* 准备单个文件的渲染信息(不实际写入)
|
|
1577
|
+
* 用于 dry-run 阶段,收集所有将要写入的文件信息
|
|
1578
|
+
*
|
|
1579
|
+
* @param options - 准备选项
|
|
1580
|
+
* @returns 文件渲染信息,或 null 表示该文件被跳过
|
|
1320
1581
|
*/
|
|
1321
|
-
const
|
|
1322
|
-
|
|
1582
|
+
const prepareFileInfo = async (options
|
|
1323
1583
|
|
|
1324
1584
|
|
|
1325
1585
|
|
|
1326
1586
|
|
|
1327
1587
|
) => {
|
|
1328
|
-
const { file, templatePath,
|
|
1588
|
+
const { file, templatePath, context, templateConfig } = options;
|
|
1329
1589
|
|
|
1330
1590
|
const srcPath = path.join(templatePath, file);
|
|
1331
1591
|
const destFile = convertDotfileName(file);
|
|
1332
1592
|
|
|
1333
1593
|
logger.verbose(
|
|
1334
|
-
` -
|
|
1594
|
+
` - Preparing: ${file}${destFile !== file ? ` -> ${destFile}` : ''}`,
|
|
1335
1595
|
);
|
|
1336
1596
|
|
|
1337
1597
|
// 判断是否为二进制文件
|
|
@@ -1365,51 +1625,177 @@ const processSingleFile = async (options
|
|
|
1365
1625
|
context,
|
|
1366
1626
|
);
|
|
1367
1627
|
|
|
1368
|
-
// 如果返回 null
|
|
1628
|
+
// 如果返回 null,表示该文件被 hook 跳过
|
|
1369
1629
|
if (processedFileInfo === null) {
|
|
1370
1630
|
logger.verbose(' ⊘ Skipped by onFileRender hook');
|
|
1371
|
-
return;
|
|
1631
|
+
return null;
|
|
1372
1632
|
}
|
|
1373
1633
|
|
|
1374
|
-
|
|
1375
|
-
|
|
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}`);
|
|
1376
1650
|
|
|
1377
1651
|
// 确保目标目录存在
|
|
1378
|
-
await ensureDir(path.dirname(
|
|
1652
|
+
await ensureDir(path.dirname(destPath));
|
|
1379
1653
|
|
|
1380
1654
|
// 写入文件
|
|
1381
|
-
if (
|
|
1382
|
-
//
|
|
1383
|
-
|
|
1384
|
-
|
|
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);
|
|
1385
1662
|
logger.verbose(' ✓ Copied (binary)');
|
|
1386
1663
|
} else {
|
|
1387
|
-
const
|
|
1388
|
-
await fs$1.writeFile(
|
|
1664
|
+
const modifiedBuffer = Buffer.from(fileInfo.content, 'base64');
|
|
1665
|
+
await fs$1.writeFile(destPath, modifiedBuffer);
|
|
1389
1666
|
logger.verbose(' ✓ Written (binary, modified by hook)');
|
|
1390
1667
|
}
|
|
1391
1668
|
} else {
|
|
1392
1669
|
// 文本文件
|
|
1393
|
-
await fs$1.writeFile(
|
|
1670
|
+
await fs$1.writeFile(destPath, fileInfo.content, 'utf-8');
|
|
1394
1671
|
logger.verbose(' ✓ Rendered and written');
|
|
1395
1672
|
}
|
|
1396
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
|
+
};
|
|
1397
1772
|
|
|
1398
1773
|
/**
|
|
1399
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 - 处理选项
|
|
1400
1785
|
*/
|
|
1401
1786
|
const processTemplateFiles = async (options
|
|
1402
1787
|
|
|
1403
1788
|
|
|
1404
1789
|
|
|
1405
1790
|
|
|
1791
|
+
|
|
1406
1792
|
) => {
|
|
1407
|
-
const { templatePath, outputPath, context, templateConfig
|
|
1793
|
+
const { templatePath, outputPath, context, templateConfig} = options;
|
|
1408
1794
|
logger.verbose('Processing template files:');
|
|
1409
1795
|
logger.verbose(` - Template path: ${templatePath}`);
|
|
1410
1796
|
logger.verbose(` - Output path: ${outputPath}`);
|
|
1411
1797
|
|
|
1412
|
-
// 验证模板目录是否存在
|
|
1798
|
+
// 阶段 0: 验证模板目录是否存在
|
|
1413
1799
|
try {
|
|
1414
1800
|
const stat = await fs$1.stat(templatePath);
|
|
1415
1801
|
logger.verbose(
|
|
@@ -1425,6 +1811,7 @@ const processTemplateFiles = async (options
|
|
|
1425
1811
|
throw error;
|
|
1426
1812
|
}
|
|
1427
1813
|
|
|
1814
|
+
// 阶段 1: 扫描所有模板文件
|
|
1428
1815
|
const files = await getAllFiles(templatePath);
|
|
1429
1816
|
|
|
1430
1817
|
logger.verbose(` - Found ${files.length} files to process`);
|
|
@@ -1434,6 +1821,23 @@ const processTemplateFiles = async (options
|
|
|
1434
1821
|
return;
|
|
1435
1822
|
}
|
|
1436
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...');
|
|
1437
1841
|
await Promise.all(
|
|
1438
1842
|
files.map(file =>
|
|
1439
1843
|
processSingleFile({
|
|
@@ -1448,78 +1852,14 @@ const processTemplateFiles = async (options
|
|
|
1448
1852
|
|
|
1449
1853
|
logger.verbose('✓ All files processed successfully');
|
|
1450
1854
|
|
|
1451
|
-
//
|
|
1452
|
-
|
|
1453
|
-
|
|
1454
|
-
|
|
1455
|
-
|
|
1456
|
-
|
|
1457
|
-
|
|
1458
|
-
|
|
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
|
-
}
|
|
1518
|
-
};
|
|
1519
|
-
|
|
1520
|
-
/**
|
|
1521
|
-
* 模板引擎执行选项
|
|
1522
|
-
*/
|
|
1855
|
+
// node_modules 将由 pnpm install 处理(利用缓存和硬链接机制)
|
|
1856
|
+
};
|
|
1857
|
+
// end_aigc
|
|
1858
|
+
|
|
1859
|
+
/**
|
|
1860
|
+
* 模板引擎执行选项
|
|
1861
|
+
*/
|
|
1862
|
+
|
|
1523
1863
|
|
|
1524
1864
|
|
|
1525
1865
|
|
|
@@ -1595,22 +1935,47 @@ const executeAfterRenderHook = async (
|
|
|
1595
1935
|
}
|
|
1596
1936
|
};
|
|
1597
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
|
+
|
|
1598
1951
|
/**
|
|
1599
1952
|
* 准备输出目录
|
|
1600
1953
|
*/
|
|
1601
|
-
const prepareOutputDirectory =
|
|
1954
|
+
const prepareOutputDirectory = (outputPath) => {
|
|
1602
1955
|
const absolutePath = path.resolve(process.cwd(), outputPath);
|
|
1603
|
-
|
|
1956
|
+
// 不再在这里验证目录是否为空,冲突检测已移至 processTemplateFiles 中
|
|
1604
1957
|
return absolutePath;
|
|
1605
1958
|
};
|
|
1606
1959
|
|
|
1960
|
+
/**
|
|
1961
|
+
* 模板引擎执行结果
|
|
1962
|
+
*/
|
|
1963
|
+
|
|
1964
|
+
|
|
1965
|
+
|
|
1966
|
+
|
|
1967
|
+
|
|
1968
|
+
|
|
1969
|
+
|
|
1970
|
+
|
|
1971
|
+
|
|
1607
1972
|
/**
|
|
1608
1973
|
* 执行完整的模板渲染流程
|
|
1609
1974
|
*/
|
|
1610
1975
|
const execute = async (
|
|
1611
1976
|
options,
|
|
1612
1977
|
) => {
|
|
1613
|
-
const { templateName, outputPath, command
|
|
1978
|
+
const { templateName, outputPath, command} = options;
|
|
1614
1979
|
|
|
1615
1980
|
// 1. 加载模板
|
|
1616
1981
|
const { templatePath } = await loadTemplateMetadata(templateName);
|
|
@@ -1627,20 +1992,23 @@ const execute = async (
|
|
|
1627
1992
|
});
|
|
1628
1993
|
|
|
1629
1994
|
// 5. 准备输出目录
|
|
1630
|
-
const absoluteOutputPath =
|
|
1995
|
+
const absoluteOutputPath = prepareOutputDirectory(outputPath);
|
|
1631
1996
|
|
|
1632
1997
|
// 6. 处理模板文件
|
|
1633
1998
|
await processTemplateFiles({
|
|
1634
1999
|
templatePath,
|
|
1635
2000
|
outputPath: absoluteOutputPath,
|
|
1636
2001
|
context,
|
|
1637
|
-
templateConfig
|
|
1638
|
-
});
|
|
2002
|
+
templateConfig});
|
|
1639
2003
|
|
|
1640
2004
|
// 7. 执行 onAfterRender 钩子
|
|
1641
2005
|
await executeAfterRenderHook(templateConfig, context, absoluteOutputPath);
|
|
1642
2006
|
|
|
1643
|
-
return
|
|
2007
|
+
return {
|
|
2008
|
+
outputPath: absoluteOutputPath,
|
|
2009
|
+
templateConfig,
|
|
2010
|
+
context,
|
|
2011
|
+
};
|
|
1644
2012
|
};
|
|
1645
2013
|
|
|
1646
2014
|
function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } }
|
|
@@ -1682,7 +2050,40 @@ const runPnpmInstall = (projectPath) => {
|
|
|
1682
2050
|
};
|
|
1683
2051
|
|
|
1684
2052
|
/**
|
|
1685
|
-
*
|
|
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 仓库
|
|
1686
2087
|
* 如果目录中已存在 .git,则跳过初始化
|
|
1687
2088
|
*/
|
|
1688
2089
|
const runGitInit = (projectPath) => {
|
|
@@ -1695,46 +2096,9 @@ const runGitInit = (projectPath) => {
|
|
|
1695
2096
|
return;
|
|
1696
2097
|
}
|
|
1697
2098
|
|
|
1698
|
-
const runGitCommand = (command) => {
|
|
1699
|
-
logger.info(`Executing: ${command}`);
|
|
1700
|
-
|
|
1701
|
-
const result = shelljs.exec(command, {
|
|
1702
|
-
cwd: projectPath,
|
|
1703
|
-
silent: true,
|
|
1704
|
-
});
|
|
1705
|
-
|
|
1706
|
-
// 输出命令的结果
|
|
1707
|
-
if (result.stdout) {
|
|
1708
|
-
process.stdout.write(result.stdout);
|
|
1709
|
-
}
|
|
1710
|
-
|
|
1711
|
-
if (result.stderr) {
|
|
1712
|
-
process.stderr.write(result.stderr);
|
|
1713
|
-
}
|
|
1714
|
-
|
|
1715
|
-
if (result.code !== 0) {
|
|
1716
|
-
const errorMessage = [
|
|
1717
|
-
`${command} failed with exit code ${result.code}`,
|
|
1718
|
-
result.stderr ? `\nStderr:\n${result.stderr}` : '',
|
|
1719
|
-
result.stdout ? `\nStdout:\n${result.stdout}` : '',
|
|
1720
|
-
]
|
|
1721
|
-
.filter(Boolean)
|
|
1722
|
-
.join('');
|
|
1723
|
-
|
|
1724
|
-
throw new Error(errorMessage);
|
|
1725
|
-
}
|
|
1726
|
-
};
|
|
1727
|
-
|
|
1728
2099
|
try {
|
|
1729
2100
|
logger.info('\nInitializing git repository...');
|
|
1730
|
-
runGitCommand('git init');
|
|
1731
|
-
|
|
1732
|
-
logger.info('Adding files to git...');
|
|
1733
|
-
runGitCommand('git add .');
|
|
1734
|
-
|
|
1735
|
-
logger.info('Creating initial commit...');
|
|
1736
|
-
runGitCommand('git commit -m "chore: initial commit"');
|
|
1737
|
-
|
|
2101
|
+
runGitCommand('git init', projectPath);
|
|
1738
2102
|
logger.success('Git repository initialized successfully!');
|
|
1739
2103
|
} catch (error) {
|
|
1740
2104
|
// Git 初始化失败不应该导致整个流程失败
|
|
@@ -1745,16 +2109,52 @@ const runGitInit = (projectPath) => {
|
|
|
1745
2109
|
}
|
|
1746
2110
|
};
|
|
1747
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
|
+
|
|
1748
2141
|
/**
|
|
1749
2142
|
* 运行开发服务器(后台模式)
|
|
1750
2143
|
* 启动后台子进程运行开发服务器,父进程可以直接退出
|
|
2144
|
+
* 使用 CLI 自己的 dev 命令(定义在 run.ts)而不是直接运行 npm run dev
|
|
1751
2145
|
*/
|
|
1752
|
-
const
|
|
2146
|
+
const runDev = (projectPath) => {
|
|
1753
2147
|
logger.info('\nStarting development server in background...');
|
|
1754
|
-
|
|
2148
|
+
|
|
2149
|
+
// 获取当前 CLI 的可执行文件路径
|
|
2150
|
+
// process.argv[0] 是 node,process.argv[1] 是 CLI 入口文件
|
|
2151
|
+
const cliPath = process.argv[1];
|
|
2152
|
+
|
|
2153
|
+
logger.info(`Executing: ${cliPath} dev in ${projectPath}`);
|
|
1755
2154
|
|
|
1756
2155
|
// 使用通用的后台执行函数启动开发服务器
|
|
1757
|
-
|
|
2156
|
+
// 调用 CLI 自己的 dev 命令
|
|
2157
|
+
const pid = spawnDetached(process.argv[0], [cliPath, 'dev'], {
|
|
1758
2158
|
cwd: projectPath,
|
|
1759
2159
|
verbose: false, // 不输出额外的进程信息,由 logger 统一处理
|
|
1760
2160
|
});
|
|
@@ -1779,6 +2179,8 @@ const executeInit = async (
|
|
|
1779
2179
|
|
|
1780
2180
|
|
|
1781
2181
|
|
|
2182
|
+
|
|
2183
|
+
|
|
1782
2184
|
,
|
|
1783
2185
|
command,
|
|
1784
2186
|
) => {
|
|
@@ -1790,61 +2192,76 @@ const executeInit = async (
|
|
|
1790
2192
|
output: outputPath,
|
|
1791
2193
|
skipInstall,
|
|
1792
2194
|
skipGit,
|
|
2195
|
+
skipCommit,
|
|
1793
2196
|
skipDev,
|
|
2197
|
+
force,
|
|
1794
2198
|
} = options;
|
|
1795
2199
|
|
|
1796
2200
|
logger.info(`Initializing project with template: ${templateName}`);
|
|
1797
2201
|
timer.logPhase('Initialization');
|
|
1798
2202
|
|
|
1799
|
-
//
|
|
1800
|
-
const
|
|
2203
|
+
// 执行模板引擎,返回结果对象
|
|
2204
|
+
const result = await execute({
|
|
1801
2205
|
templateName,
|
|
1802
2206
|
outputPath,
|
|
1803
2207
|
command,
|
|
2208
|
+
force,
|
|
1804
2209
|
});
|
|
2210
|
+
const { outputPath: absoluteOutputPath, templateConfig, context } = result;
|
|
1805
2211
|
|
|
1806
2212
|
timer.logPhase('Template engine execution');
|
|
1807
2213
|
logger.success('Project created successfully!');
|
|
1808
2214
|
|
|
1809
|
-
//
|
|
1810
|
-
|
|
1811
|
-
|
|
1812
|
-
const hasNodeModules = fs.existsSync(nodeModulesPath);
|
|
2215
|
+
// 检查是否存在 package.json
|
|
2216
|
+
const packageJsonPath = path.join(absoluteOutputPath, 'package.json');
|
|
2217
|
+
const hasPackageJson = fs.existsSync(packageJsonPath);
|
|
1813
2218
|
|
|
1814
|
-
|
|
1815
|
-
|
|
1816
|
-
|
|
1817
|
-
);
|
|
1818
|
-
timer.logPhase('Node modules (pre-warmed)');
|
|
1819
|
-
} else {
|
|
2219
|
+
// 安装依赖(始终使用 pnpm install,利用缓存机制)
|
|
2220
|
+
if (!skipInstall) {
|
|
2221
|
+
if (hasPackageJson) {
|
|
1820
2222
|
runPnpmInstall(absoluteOutputPath);
|
|
1821
2223
|
timer.logPhase('Dependencies installation');
|
|
2224
|
+
} else {
|
|
2225
|
+
logger.info(
|
|
2226
|
+
'\n💡 No package.json found, skipping dependency installation',
|
|
2227
|
+
);
|
|
1822
2228
|
}
|
|
1823
2229
|
}
|
|
1824
2230
|
|
|
2231
|
+
// 执行 onComplete 钩子(在 pnpm install 之后)
|
|
2232
|
+
await executeCompleteHook(templateConfig, context, absoluteOutputPath);
|
|
2233
|
+
timer.logPhase('Complete hook execution');
|
|
2234
|
+
|
|
1825
2235
|
// 如果没有跳过 git,则初始化 git 仓库
|
|
1826
2236
|
if (!skipGit) {
|
|
1827
2237
|
runGitInit(absoluteOutputPath);
|
|
1828
2238
|
timer.logPhase('Git initialization');
|
|
1829
2239
|
}
|
|
1830
2240
|
|
|
2241
|
+
// 如果没有跳过 commit,则提交初始化生成的文件
|
|
2242
|
+
if (!skipCommit) {
|
|
2243
|
+
commitChanges(absoluteOutputPath);
|
|
2244
|
+
timer.logPhase('Git commit');
|
|
2245
|
+
}
|
|
2246
|
+
|
|
1831
2247
|
// 如果没有跳过 dev,则启动开发服务器
|
|
1832
2248
|
if (!skipDev) {
|
|
1833
|
-
|
|
2249
|
+
runDev(absoluteOutputPath);
|
|
1834
2250
|
timer.logPhase('Dev server startup');
|
|
1835
2251
|
} else {
|
|
1836
2252
|
// 只有跳过 dev 时才显示 Next steps
|
|
1837
2253
|
logger.info('\nNext steps:');
|
|
1838
2254
|
logger.info(` cd ${outputPath}`);
|
|
1839
|
-
if (skipInstall) {
|
|
2255
|
+
if (skipInstall && hasPackageJson) {
|
|
1840
2256
|
logger.info(' pnpm install');
|
|
1841
2257
|
}
|
|
1842
2258
|
if (skipGit) {
|
|
1843
|
-
logger.info(
|
|
1844
|
-
|
|
1845
|
-
|
|
2259
|
+
logger.info(' git init');
|
|
2260
|
+
}
|
|
2261
|
+
if (skipCommit) {
|
|
2262
|
+
logger.info(' git add --all && git commit -m "chore: init env"');
|
|
1846
2263
|
}
|
|
1847
|
-
logger.info('
|
|
2264
|
+
logger.info(' coze dev');
|
|
1848
2265
|
}
|
|
1849
2266
|
|
|
1850
2267
|
// 输出总耗时
|
|
@@ -1860,7 +2277,7 @@ const executeInit = async (
|
|
|
1860
2277
|
/**
|
|
1861
2278
|
* 注册 init 命令到 program
|
|
1862
2279
|
*/
|
|
1863
|
-
const registerCommand = program => {
|
|
2280
|
+
const registerCommand$1 = program => {
|
|
1864
2281
|
program
|
|
1865
2282
|
.command('init')
|
|
1866
2283
|
.description('Initialize a new project from a template')
|
|
@@ -1869,23 +2286,328 @@ const registerCommand = program => {
|
|
|
1869
2286
|
.option('-o, --output <path>', 'Output directory', process.cwd())
|
|
1870
2287
|
.option('--skip-install', 'Skip automatic pnpm install', false)
|
|
1871
2288
|
.option('--skip-git', 'Skip automatic git initialization', false)
|
|
2289
|
+
.option(
|
|
2290
|
+
'--skip-commit',
|
|
2291
|
+
'Skip automatic git commit after initialization',
|
|
2292
|
+
false,
|
|
2293
|
+
)
|
|
1872
2294
|
.option('--skip-dev', 'Skip automatic dev server start', false)
|
|
1873
2295
|
.allowUnknownOption() // 允许透传参数
|
|
1874
2296
|
.action(async (directory, options, command) => {
|
|
1875
2297
|
// 位置参数优先级高于 --output 选项
|
|
1876
2298
|
const outputPath = _nullishCoalesce(directory, () => ( options.output));
|
|
1877
|
-
|
|
2299
|
+
// Always use force mode - overwrite existing files without conflict check
|
|
2300
|
+
const force = true;
|
|
2301
|
+
await executeInit({ ...options, output: outputPath, force }, command);
|
|
1878
2302
|
});
|
|
1879
2303
|
};
|
|
1880
2304
|
|
|
1881
|
-
|
|
2305
|
+
// ABOUTME: This file implements the update command for coze CLI
|
|
2306
|
+
// ABOUTME: It wraps pnpm update/install to update package dependencies with logging support
|
|
2307
|
+
|
|
2308
|
+
|
|
2309
|
+
|
|
2310
|
+
|
|
2311
|
+
/**
|
|
2312
|
+
* 日志文件名常量
|
|
2313
|
+
*/
|
|
2314
|
+
const LOG_FILE_NAME = 'update.log';
|
|
2315
|
+
|
|
2316
|
+
/**
|
|
2317
|
+
* 获取日志目录
|
|
2318
|
+
* 优先使用环境变量 COZE_LOG_DIR,否则使用 ~/.coze-logs
|
|
2319
|
+
*/
|
|
2320
|
+
const getLogDir = () =>
|
|
2321
|
+
process.env.COZE_LOG_DIR || path.join(os.homedir(), '.coze-logs');
|
|
2322
|
+
|
|
2323
|
+
/**
|
|
2324
|
+
* 解析日志文件路径
|
|
2325
|
+
* - 如果是绝对路径,直接使用
|
|
2326
|
+
* - 如果是相对路径,基于 getLogDir() + 相对路径
|
|
2327
|
+
* - 如果为空,使用 getLogDir() + LOG_FILE_NAME
|
|
2328
|
+
*/
|
|
2329
|
+
const resolveLogFilePath = (logFile) => {
|
|
2330
|
+
if (!logFile) {
|
|
2331
|
+
return path.join(getLogDir(), LOG_FILE_NAME);
|
|
2332
|
+
}
|
|
2333
|
+
|
|
2334
|
+
if (path.isAbsolute(logFile)) {
|
|
2335
|
+
return logFile;
|
|
2336
|
+
}
|
|
2337
|
+
|
|
2338
|
+
return path.join(getLogDir(), logFile);
|
|
2339
|
+
};
|
|
2340
|
+
|
|
2341
|
+
/**
|
|
2342
|
+
* 创建日志写入流
|
|
2343
|
+
*/
|
|
2344
|
+
const createLogStream = (logFilePath) => {
|
|
2345
|
+
const logDir = path.dirname(logFilePath);
|
|
2346
|
+
|
|
2347
|
+
// 确保日志目录存在
|
|
2348
|
+
if (!fs.existsSync(logDir)) {
|
|
2349
|
+
fs.mkdirSync(logDir, { recursive: true });
|
|
2350
|
+
}
|
|
2351
|
+
|
|
2352
|
+
// 使用 'w' 标志覆盖之前的日志
|
|
2353
|
+
return fs.createWriteStream(logFilePath, { flags: 'w' });
|
|
2354
|
+
};
|
|
2355
|
+
|
|
2356
|
+
/**
|
|
2357
|
+
* 格式化时间戳
|
|
2358
|
+
*/
|
|
2359
|
+
const formatTimestamp = () => {
|
|
2360
|
+
const now = new Date();
|
|
2361
|
+
return now.toISOString();
|
|
2362
|
+
};
|
|
2363
|
+
|
|
2364
|
+
/**
|
|
2365
|
+
* 写入带时间戳的日志
|
|
2366
|
+
*/
|
|
2367
|
+
const writeLogWithTimestamp = (stream, message) => {
|
|
2368
|
+
const timestamp = formatTimestamp();
|
|
2369
|
+
const lines = message.split('\n');
|
|
2370
|
+
lines.forEach(line => {
|
|
2371
|
+
if (line) {
|
|
2372
|
+
stream.write(`[${timestamp}] ${line}\n`);
|
|
2373
|
+
} else {
|
|
2374
|
+
stream.write('\n');
|
|
2375
|
+
}
|
|
2376
|
+
});
|
|
2377
|
+
// 确保数据写入磁盘
|
|
2378
|
+
stream.uncork();
|
|
2379
|
+
};
|
|
2380
|
+
|
|
2381
|
+
/**
|
|
2382
|
+
* 同时输出到控制台和日志文件
|
|
2383
|
+
*/
|
|
2384
|
+
const logWithFile = (
|
|
2385
|
+
stream,
|
|
2386
|
+
level,
|
|
2387
|
+
message,
|
|
2388
|
+
) => {
|
|
2389
|
+
// 输出到控制台
|
|
2390
|
+
switch (level) {
|
|
2391
|
+
case 'info':
|
|
2392
|
+
logger.info(message);
|
|
2393
|
+
break;
|
|
2394
|
+
case 'success':
|
|
2395
|
+
logger.success(message);
|
|
2396
|
+
break;
|
|
2397
|
+
case 'error':
|
|
2398
|
+
logger.error(message);
|
|
2399
|
+
break;
|
|
2400
|
+
default:
|
|
2401
|
+
logger.info(message);
|
|
2402
|
+
break;
|
|
2403
|
+
}
|
|
2404
|
+
|
|
2405
|
+
// 写入日志文件(带时间戳)
|
|
2406
|
+
writeLogWithTimestamp(stream, `[${level.toUpperCase()}] ${message}`);
|
|
2407
|
+
};
|
|
2408
|
+
|
|
2409
|
+
// start_aigc
|
|
2410
|
+
/**
|
|
2411
|
+
* 构建 pnpm add 命令
|
|
2412
|
+
*/
|
|
2413
|
+
const buildPnpmCommand = (
|
|
2414
|
+
packageName,
|
|
2415
|
+
options
|
|
2416
|
+
|
|
2417
|
+
|
|
2418
|
+
|
|
2419
|
+
|
|
2420
|
+
,
|
|
2421
|
+
) => {
|
|
2422
|
+
const { global, version, registry, extraArgs } = options;
|
|
2423
|
+
|
|
2424
|
+
const parts = ['pnpm', 'add'];
|
|
2425
|
+
|
|
2426
|
+
// 添加全局标记
|
|
2427
|
+
if (global) {
|
|
2428
|
+
parts.push('-g');
|
|
2429
|
+
}
|
|
2430
|
+
|
|
2431
|
+
// 添加包名和版本
|
|
2432
|
+
if (version && version !== 'latest') {
|
|
2433
|
+
parts.push(`${packageName}@${version}`);
|
|
2434
|
+
} else {
|
|
2435
|
+
parts.push(`${packageName}@latest`);
|
|
2436
|
+
}
|
|
2437
|
+
|
|
2438
|
+
// 添加 registry
|
|
2439
|
+
if (registry) {
|
|
2440
|
+
parts.push(`--registry=${registry}`);
|
|
2441
|
+
}
|
|
2442
|
+
|
|
2443
|
+
// 添加额外参数
|
|
2444
|
+
if (extraArgs.length > 0) {
|
|
2445
|
+
parts.push(...extraArgs);
|
|
2446
|
+
}
|
|
2447
|
+
|
|
2448
|
+
return parts.join(' ');
|
|
2449
|
+
};
|
|
2450
|
+
// end_aigc
|
|
2451
|
+
|
|
2452
|
+
// start_aigc
|
|
2453
|
+
/**
|
|
2454
|
+
* 执行 update 命令的内部实现
|
|
2455
|
+
*/
|
|
2456
|
+
const executeUpdate = (
|
|
2457
|
+
packageName,
|
|
2458
|
+
options
|
|
2459
|
+
|
|
2460
|
+
|
|
2461
|
+
|
|
2462
|
+
|
|
2463
|
+
|
|
2464
|
+
|
|
2465
|
+
,
|
|
2466
|
+
) => {
|
|
2467
|
+
let logStream = null;
|
|
2468
|
+
|
|
2469
|
+
try {
|
|
2470
|
+
const { global, cwd, version, registry, logFile, extraArgs } = options;
|
|
2471
|
+
|
|
2472
|
+
// 准备日志
|
|
2473
|
+
const logFilePath = resolveLogFilePath(logFile);
|
|
2474
|
+
|
|
2475
|
+
// 调试:确认日志路径
|
|
2476
|
+
logger.info(`Log file path resolved to: ${logFilePath}`);
|
|
2477
|
+
|
|
2478
|
+
logStream = createLogStream(logFilePath);
|
|
2479
|
+
|
|
2480
|
+
// 调试:确认流已创建
|
|
2481
|
+
logger.info('Log stream created successfully');
|
|
2482
|
+
|
|
2483
|
+
logWithFile(logStream, 'info', `Updating package: ${packageName}`);
|
|
2484
|
+
|
|
2485
|
+
// 构建命令
|
|
2486
|
+
const command = buildPnpmCommand(packageName, {
|
|
2487
|
+
global,
|
|
2488
|
+
version,
|
|
2489
|
+
registry,
|
|
2490
|
+
extraArgs,
|
|
2491
|
+
});
|
|
2492
|
+
|
|
2493
|
+
// 确定工作目录
|
|
2494
|
+
const workingDir = cwd
|
|
2495
|
+
? path.isAbsolute(cwd)
|
|
2496
|
+
? cwd
|
|
2497
|
+
: path.join(process.cwd(), cwd)
|
|
2498
|
+
: process.cwd();
|
|
2499
|
+
|
|
2500
|
+
logWithFile(logStream, 'info', `Executing: ${command}`);
|
|
2501
|
+
logWithFile(logStream, 'info', `Working directory: ${workingDir}`);
|
|
2502
|
+
logWithFile(logStream, 'info', `Log file: ${logFilePath}`);
|
|
2503
|
+
|
|
2504
|
+
// 记录命令开始时间
|
|
2505
|
+
writeLogWithTimestamp(logStream, '--- Command execution started ---');
|
|
2506
|
+
|
|
2507
|
+
// 同步执行命令
|
|
2508
|
+
const result = shelljs.exec(command, {
|
|
2509
|
+
cwd: workingDir,
|
|
2510
|
+
silent: true, // 使用 silent 来捕获输出
|
|
2511
|
+
});
|
|
2512
|
+
|
|
2513
|
+
// 将输出写入控制台和日志文件(带时间戳)
|
|
2514
|
+
if (result.stdout) {
|
|
2515
|
+
process.stdout.write(result.stdout);
|
|
2516
|
+
writeLogWithTimestamp(logStream, result.stdout.trim());
|
|
2517
|
+
}
|
|
2518
|
+
|
|
2519
|
+
if (result.stderr) {
|
|
2520
|
+
process.stderr.write(result.stderr);
|
|
2521
|
+
writeLogWithTimestamp(logStream, result.stderr.trim());
|
|
2522
|
+
}
|
|
2523
|
+
|
|
2524
|
+
// 记录命令结束时间
|
|
2525
|
+
writeLogWithTimestamp(logStream, '--- Command execution ended ---');
|
|
2526
|
+
|
|
2527
|
+
// 检查执行结果并记录到日志
|
|
2528
|
+
if (result.code === 0) {
|
|
2529
|
+
logWithFile(logStream, 'success', 'Package updated successfully');
|
|
2530
|
+
logWithFile(logStream, 'info', `Log file: ${logFilePath}`);
|
|
2531
|
+
} else {
|
|
2532
|
+
logWithFile(
|
|
2533
|
+
logStream,
|
|
2534
|
+
'error',
|
|
2535
|
+
`Command exited with code ${result.code}`,
|
|
2536
|
+
);
|
|
2537
|
+
logWithFile(
|
|
2538
|
+
logStream,
|
|
2539
|
+
'error',
|
|
2540
|
+
`Check log file for details: ${logFilePath}`,
|
|
2541
|
+
);
|
|
2542
|
+
}
|
|
2543
|
+
|
|
2544
|
+
// 关闭日志流并等待写入完成
|
|
2545
|
+
logStream.end(() => {
|
|
2546
|
+
// 流关闭后再退出进程
|
|
2547
|
+
if (result.code !== 0) {
|
|
2548
|
+
process.exit(result.code || 1);
|
|
2549
|
+
}
|
|
2550
|
+
});
|
|
2551
|
+
} catch (error) {
|
|
2552
|
+
logger.error('Failed to update package:');
|
|
2553
|
+
logger.error(error instanceof Error ? error.message : String(error));
|
|
2554
|
+
|
|
2555
|
+
// 写入错误到日志文件
|
|
2556
|
+
if (logStream) {
|
|
2557
|
+
writeLogWithTimestamp(
|
|
2558
|
+
logStream,
|
|
2559
|
+
`[ERROR] ${error instanceof Error ? error.message : String(error)}`,
|
|
2560
|
+
);
|
|
2561
|
+
// 等待流关闭后再退出
|
|
2562
|
+
logStream.end(() => {
|
|
2563
|
+
process.exit(1);
|
|
2564
|
+
});
|
|
2565
|
+
} else {
|
|
2566
|
+
process.exit(1);
|
|
2567
|
+
}
|
|
2568
|
+
}
|
|
2569
|
+
};
|
|
2570
|
+
// end_aigc
|
|
2571
|
+
|
|
2572
|
+
/**
|
|
2573
|
+
* 注册 update 命令到 program
|
|
2574
|
+
*/
|
|
2575
|
+
const registerCommand = program => {
|
|
2576
|
+
program
|
|
2577
|
+
.command('update <package>')
|
|
2578
|
+
.description('Update a package dependency')
|
|
2579
|
+
.option('-g, --global', 'Update package globally', false)
|
|
2580
|
+
.option('-c, --cwd <path>', 'Working directory for the update')
|
|
2581
|
+
.option(
|
|
2582
|
+
'--to <version>',
|
|
2583
|
+
'Version to update to (default: latest)',
|
|
2584
|
+
'latest',
|
|
2585
|
+
)
|
|
2586
|
+
.option('--registry <url>', 'Registry URL to use for the update')
|
|
2587
|
+
.option('--log-file <path>', 'Log file path')
|
|
2588
|
+
.allowUnknownOption() // 允许透传参数给 pnpm
|
|
2589
|
+
.action((packageName, options, command) => {
|
|
2590
|
+
// 收集所有未知选项作为额外参数
|
|
2591
|
+
const extraArgs = command.args.slice(1);
|
|
2592
|
+
|
|
2593
|
+
executeUpdate(packageName, {
|
|
2594
|
+
...options,
|
|
2595
|
+
version: options.to, // 将 --to 映射到 version
|
|
2596
|
+
extraArgs,
|
|
2597
|
+
});
|
|
2598
|
+
});
|
|
2599
|
+
};
|
|
2600
|
+
|
|
2601
|
+
var version = "0.0.1-alpha.b287c1";
|
|
1882
2602
|
var packageJson = {
|
|
1883
2603
|
version: version};
|
|
1884
2604
|
|
|
1885
2605
|
const commands = [
|
|
1886
|
-
registerCommand,
|
|
1887
2606
|
registerCommand$1,
|
|
1888
2607
|
registerCommand$2,
|
|
2608
|
+
registerCommand$4,
|
|
2609
|
+
registerCommand$3,
|
|
2610
|
+
registerCommand,
|
|
1889
2611
|
];
|
|
1890
2612
|
|
|
1891
2613
|
const main = () => {
|