@lark-apaas/miaoda-cli 0.1.17-alpha.0 → 0.1.17-alpha.ddef3e9

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.
@@ -166,6 +166,7 @@ function registerAppInit(parent) {
166
166
  .option('--template <stack>', `技术栈短名(${index_1.SUPPORTED_STACKS.join(' / ')})`)
167
167
  .option('--conf <json>', 'init 配置 JSON。支持 {"version": "<template 版本>"},默认 latest')
168
168
  .option('--skip-install', '跳过依赖安装', false)
169
+ .option('--async-install', '派发后台进程装依赖并立即返回(与 --skip-install 互斥)', false)
169
170
  .addOption((0, shared_1.appIdOption)())
170
171
  .addHelpText('after', `
171
172
  幂等
@@ -194,17 +195,31 @@ function registerAppInit(parent) {
194
195
  为空都会 fallback npm install。
195
196
  JSON 模式下子进程 stdout 重定向到 stderr,避免污染最终 emit 的 JSON。
196
197
 
198
+ 异步安装(--async-install)
199
+ 跳过同步 npm install,派发一个 detached 后台进程装依赖后立即返回。
200
+ .spark/meta.json 在返回前写出(= 脚手架就绪),install 真实结果由 event marker 表达。
201
+ 仅沙箱(SANDBOX_ID 非空)写 marker(presence 定成败,content 给排障):
202
+ 成功 → /tmp/event/WORKSPACE_READY
203
+ 失败 → /tmp/event/WORKSPACE_FAILED
204
+ 安装日志:/tmp/async_install_dep.std.log
205
+ 失败恢复:靠后续 'miaoda app sync'(dev.sh 入口 / 沙箱 pod 启动会跑)兜底,init 不重试。
206
+ 与 --skip-install 互斥。
207
+
197
208
  JSON 输出
198
209
  已初始化:{"data": {"initialized": false, "reason": "already_initialized", "targetDir": "..."}}
199
210
  新初始化:{"data": {"initialized": true, "template": "...", "templateVersion": "...", "steeringVersion": "...",
200
211
  "appId": "...", "platformStackFound": true, "platformSyncedFiles": [...],
201
212
  "installed": true, "installSource": "cache|npm|skipped", "installHash": "...", ...}}
213
+ async 模式:{"data": {"initialized": true, "asyncInstall": true, "installed": false,
214
+ "installSource": "async", "installPid": 123, "installLogPath": "...",
215
+ "eventReadyPath": "...", "eventFailedPath": "..."}}
202
216
 
203
217
  示例
204
218
  $ miaoda app init --template vite-react
205
219
  $ miaoda app init --template nestjs-react-fullstack --app-id app_xxx
206
220
  $ miaoda app init --template vite-react --conf '{"version": "0.1.0"}'
207
221
  $ miaoda app init --template vite-react --skip-install
222
+ $ miaoda app init --template vite-react --async-install
208
223
  $ MIAODA_DEP_CACHE_DIR=/tmp/dep-cache miaoda app init --template vite-react
209
224
  `);
210
225
  cmd.action((0, shared_1.withHelp)(cmd, async (rawOpts) => {
@@ -214,6 +229,7 @@ JSON 输出
214
229
  conf,
215
230
  skipInstall: rawOpts.skipInstall,
216
231
  appId: rawOpts.appId,
232
+ asyncInstall: rawOpts.asyncInstall,
217
233
  });
218
234
  }));
219
235
  }
@@ -60,6 +60,9 @@ async function handleAppInit(opts) {
60
60
  next_actions: [`可用 stack:${index_1.SUPPORTED_STACKS.join(', ')}`],
61
61
  });
62
62
  }
63
+ if (opts.asyncInstall && opts.skipInstall) {
64
+ throw new error_1.AppError('ARGS_INVALID', '--async-install 与 --skip-install 互斥');
65
+ }
63
66
  const version = opts.conf?.version;
64
67
  const projectName = node_path_1.default.basename(targetDir);
65
68
  const tplResult = (0, index_1.renderTemplate)({ stack, version, targetDir, projectName });
@@ -86,13 +89,17 @@ async function handleAppInit(opts) {
86
89
  (0, logger_1.log)('init', `⚠ skills sync failed (continuing init): ${steeringError}`);
87
90
  steeringResult = { version: 'unknown', syncedSkills: [], techSynced: false };
88
91
  }
89
- // 装模板钉死的依赖。不带 <pkg>@latest 位置参数,因此 npm 不会改写 package.json/lockfile
92
+ // 依赖安装:async 模式留到 meta 落盘后派发后台进程(不等装完),同步模式当场装。
93
+ // 同步:装模板钉死的依赖。不带 <pkg>@latest 位置参数,因此 npm 不会改写 package.json/lockfile
90
94
  // —— "升管控包到 latest" 是 sync 的职责,在 dev.sh 入口跑 `miaoda app sync` 时触发。
91
- const installResult = (0, index_1.installDependencies)({
92
- targetDir,
93
- skip: opts.skipInstall,
94
- quietStdout: (0, output_1.isJsonMode)(),
95
- });
95
+ let installResult;
96
+ if (!opts.asyncInstall) {
97
+ installResult = (0, index_1.installDependencies)({
98
+ targetDir,
99
+ skip: opts.skipInstall,
100
+ quietStdout: (0, output_1.isJsonMode)(),
101
+ });
102
+ }
96
103
  // template 自带 .githooks/pre-commit;如果用户项目已 git init,顺便设上 core.hooksPath。
97
104
  const hookActivation = (0, githooks_1.activateGitHooks)(targetDir);
98
105
  (0, index_1.writeSparkMeta)(targetDir, {
@@ -101,8 +108,18 @@ async function handleAppInit(opts) {
101
108
  archType: tplResult.archType,
102
109
  app_id: opts.appId,
103
110
  });
111
+ // async 模式:meta 已落盘(= 脚手架就绪),再派发后台安装并立即返回(不等装完)。
112
+ let asyncDispatch;
113
+ if (opts.asyncInstall) {
114
+ asyncDispatch = (0, index_1.dispatchAsyncInstall)({ targetDir });
115
+ }
104
116
  if (!(0, output_1.isJsonMode)()) {
105
- process.stdout.write(`✓ Initialized ${stack} (template ${tplResult.version}, steering ${steeringResult.version}, install ${installResult.source}) in ${targetDir}\n`);
117
+ if (opts.asyncInstall && asyncDispatch) {
118
+ process.stdout.write(`✓ Initialized ${stack} (template ${tplResult.version}, steering ${steeringResult.version}, install dispatched in background pid ${String(asyncDispatch.pid)}) in ${targetDir}\n`);
119
+ }
120
+ else if (installResult) {
121
+ process.stdout.write(`✓ Initialized ${stack} (template ${tplResult.version}, steering ${steeringResult.version}, install ${installResult.source}) in ${targetDir}\n`);
122
+ }
106
123
  }
107
124
  (0, output_1.emit)({
108
125
  data: {
@@ -117,11 +134,23 @@ async function handleAppInit(opts) {
117
134
  techSynced: steeringResult.techSynced,
118
135
  steeringError,
119
136
  gitHooks: hookActivation.action,
120
- installed: installResult.installed,
121
- installSource: installResult.source,
122
- installHash: installResult.hash,
123
- cacheZip: installResult.cacheZip,
124
- installError: installResult.error,
137
+ ...(opts.asyncInstall && asyncDispatch
138
+ ? {
139
+ asyncInstall: true,
140
+ installed: false,
141
+ installSource: 'async',
142
+ installPid: asyncDispatch.pid,
143
+ installLogPath: asyncDispatch.logPath,
144
+ eventReadyPath: asyncDispatch.eventReadyPath,
145
+ eventFailedPath: asyncDispatch.eventFailedPath,
146
+ }
147
+ : {
148
+ installed: installResult?.installed,
149
+ installSource: installResult?.source,
150
+ installHash: installResult?.hash,
151
+ cacheZip: installResult?.cacheZip,
152
+ installError: installResult?.error,
153
+ }),
125
154
  },
126
155
  });
127
156
  }
@@ -134,6 +134,27 @@ async function handleAppMigrate(opts) {
134
134
  installError = err instanceof Error ? err.message : String(err);
135
135
  (0, logger_1.log)('migrate', `⚠ npm install failed (continuing): ${installError}`);
136
136
  }
137
+ // 沙箱环境下重启 dev process —— scripts/dev.js 被 fullstack 版本覆盖了,但当前正在
138
+ // 跑的 node 进程仍按老逻辑工作(只起 vite,没起 nest),整体没切到 fullstack 模式。
139
+ // 通过 pkill 杀掉 dev.js 主进程,依赖沙箱平台 supervisor 自动重新 exec dev.sh → dev.js,
140
+ // 新进程用新 deps + 新 scripts,正常进入 fullstack 模式。
141
+ // - 只在 SANDBOX_ID 非空时做(本地环境用户进程混杂,主动 pkill 风险大)
142
+ // - install 挂了不做 —— 新依赖不全, 杀旧 dev 后新 dev 起不来反而更糟,
143
+ // 留着旧进程让用户先处理 installError
144
+ // - 软失败:pkill 无匹配进程退出 1,catch 吞掉
145
+ let devRestarted = false;
146
+ if (installError === undefined &&
147
+ process.env.SANDBOX_ID !== undefined &&
148
+ process.env.SANDBOX_ID !== '') {
149
+ (0, logger_1.log)('migrate', '沙箱环境,重启 dev process(平台 supervisor 会自动拉起)...');
150
+ try {
151
+ (0, node_child_process_1.execFileSync)('pkill', ['-f', 'node.*scripts/dev\\.js'], { stdio: 'ignore' });
152
+ devRestarted = true;
153
+ }
154
+ catch {
155
+ // pkill 无匹配进程时退出 1,正常情况(可能 dev 没在跑)
156
+ }
157
+ }
137
158
  (0, output_1.emit)({
138
159
  data: {
139
160
  from,
@@ -148,10 +169,17 @@ async function handleAppMigrate(opts) {
148
169
  ...summarizeResults(results),
149
170
  followLatestPackages: followLatest,
150
171
  installError,
151
- nextActions: [
152
- 'git status / git diff 评估改动并 commit',
153
- 'miaoda skills sync 同步到新 stack 的 agent skills',
154
- ],
172
+ devRestarted,
173
+ nextActions: process.env.SANDBOX_ID !== undefined && process.env.SANDBOX_ID !== ''
174
+ ? [
175
+ 'git status / git diff 评估改动并 commit',
176
+ 'miaoda skills sync 同步到新 stack 的 agent skills',
177
+ ]
178
+ : [
179
+ 'git status / git diff 评估改动并 commit',
180
+ '重启 dev 进程让新 scripts/dev.js + 新依赖生效:重跑 npm run dev(killOrphansByPort 会自动踢掉旧进程)',
181
+ 'miaoda skills sync 同步到新 stack 的 agent skills',
182
+ ],
155
183
  },
156
184
  });
157
185
  }
@@ -83,6 +83,11 @@ exports.MIGRATE_CONFIG = {
83
83
  { type: 'directory', from: 'shared', to: 'shared', overwrite: false },
84
84
  // .env fallback(user 没有时用 template 默认)
85
85
  { type: 'file', from: '.env', to: '.env', overwrite: false },
86
+ // ===== 4.5. plugin 实例从 shared/capabilities 搬到 server/capabilities =====
87
+ // vite-react (jsPage) 时期 plugin 实例 json 放在 shared/capabilities/<id>.json,
88
+ // fullstack 形态走 server 侧装载,路径约定改成 server/capabilities/<id>.json。
89
+ // 顺序约束:必须在 server/ 覆盖之后 —— server/ 是 overwrite:true,放前面会被擦掉。
90
+ { type: 'move-directory', from: 'shared/capabilities', to: 'server/capabilities' },
86
91
  // ===== 5. 加 fullstack 形态必需的 scripts =====
87
92
  // 这些 scripts 用 user 现有的可能性低(vite-react 没有 nest 相关 script),直接覆盖
88
93
  {
@@ -0,0 +1,116 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.ASYNC_INSTALL_LOG = void 0;
7
+ exports.isSandboxEnv = isSandboxEnv;
8
+ exports.runAsyncInstallWorker = runAsyncInstallWorker;
9
+ exports.dispatchAsyncInstall = dispatchAsyncInstall;
10
+ const node_child_process_1 = require("node:child_process");
11
+ const node_fs_1 = __importDefault(require("node:fs"));
12
+ const node_path_1 = __importDefault(require("node:path"));
13
+ const install_1 = require("./install");
14
+ /** event marker 目录,写死;写前 mkdir -p。沙箱里 code server 轮询此目录。 */
15
+ const EVENT_DIR = '/tmp/event';
16
+ /** 安装成功 marker 名 */
17
+ const MARKER_READY = 'WORKSPACE_READY';
18
+ /** 安装失败 marker 名 */
19
+ const MARKER_FAILED = 'WORKSPACE_FAILED';
20
+ /** 后台 worker stdout/stderr 落盘路径 */
21
+ exports.ASYNC_INSTALL_LOG = '/tmp/async_install_dep.std.log';
22
+ /** 沙箱判定:SANDBOX_ID 非空。与 init handler 的 outputLayout 分流口径一致。 */
23
+ function isSandboxEnv() {
24
+ return process.env.SANDBOX_ID !== undefined && process.env.SANDBOX_ID !== '';
25
+ }
26
+ function nowIso() {
27
+ return new Date().toISOString();
28
+ }
29
+ function clearStaleMarkers(eventDir) {
30
+ for (const name of [MARKER_READY, MARKER_FAILED]) {
31
+ const p = node_path_1.default.join(eventDir, name);
32
+ if (node_fs_1.default.existsSync(p))
33
+ node_fs_1.default.rmSync(p, { force: true });
34
+ }
35
+ }
36
+ function writeMarker(eventDir, name, content) {
37
+ node_fs_1.default.mkdirSync(eventDir, { recursive: true });
38
+ node_fs_1.default.writeFileSync(node_path_1.default.join(eventDir, name), JSON.stringify(content) + '\n', 'utf-8');
39
+ }
40
+ /**
41
+ * 后台 install worker 体:跑 installDependencies,按结果写 marker(仅沙箱)。
42
+ * 由 dispatchAsyncInstall 通过 `node -e` 在 detached 子进程里调用(见下)。
43
+ * presence 定成败:成功写 WORKSPACE_READY,失败写 WORKSPACE_FAILED;content 给排障。
44
+ */
45
+ function runAsyncInstallWorker(opts) {
46
+ const eventDir = opts.eventDir ?? EVENT_DIR;
47
+ const sandbox = isSandboxEnv();
48
+ if (sandbox)
49
+ clearStaleMarkers(eventDir);
50
+ const start = Date.now();
51
+ let result;
52
+ try {
53
+ result = (0, install_1.installDependencies)({ targetDir: opts.targetDir, quietStdout: false });
54
+ }
55
+ catch (err) {
56
+ const msg = err instanceof Error ? err.message : String(err);
57
+ if (sandbox) {
58
+ writeMarker(eventDir, MARKER_FAILED, {
59
+ ts: nowIso(),
60
+ error: msg,
61
+ logPath: exports.ASYNC_INSTALL_LOG,
62
+ });
63
+ }
64
+ return;
65
+ }
66
+ if (!sandbox)
67
+ return;
68
+ if (result.source === 'failed') {
69
+ writeMarker(eventDir, MARKER_FAILED, {
70
+ ts: nowIso(),
71
+ error: result.error ?? 'npm install failed',
72
+ logPath: exports.ASYNC_INSTALL_LOG,
73
+ });
74
+ }
75
+ else {
76
+ writeMarker(eventDir, MARKER_READY, {
77
+ ts: nowIso(),
78
+ durationMs: Date.now() - start,
79
+ source: result.source,
80
+ });
81
+ }
82
+ }
83
+ /**
84
+ * 派发 detached 后台进程跑依赖安装并立即返回。
85
+ * 用 `node -e` 直接 require 本模块(编译产物)调 runAsyncInstallWorker,
86
+ * 不挂任何 CLI 子命令、对外不暴露;worker 内复用 installDependencies(cache / registry / fallback)。
87
+ * 沙箱下派发前清旧 marker;stdout/stderr 重定向到 logPath。
88
+ */
89
+ function dispatchAsyncInstall(opts) {
90
+ const eventDir = opts.eventDir ?? EVENT_DIR;
91
+ const logPath = opts.logPath ?? exports.ASYNC_INSTALL_LOG;
92
+ const sandbox = isSandboxEnv();
93
+ if (sandbox)
94
+ clearStaleMarkers(eventDir);
95
+ // targetDir 走 argv 传,不拼进代码串(免转义);只把本模块路径 JSON 内联进去。
96
+ const workerCode = `require(${JSON.stringify(__filename)}).runAsyncInstallWorker({ targetDir: process.argv[1] })`;
97
+ node_fs_1.default.mkdirSync(node_path_1.default.dirname(logPath), { recursive: true });
98
+ const fd = node_fs_1.default.openSync(logPath, 'w');
99
+ try {
100
+ const child = (0, node_child_process_1.spawn)(process.execPath, ['-e', workerCode, opts.targetDir], {
101
+ detached: true,
102
+ stdio: ['ignore', fd, fd],
103
+ });
104
+ child.unref();
105
+ return {
106
+ pid: child.pid,
107
+ logPath,
108
+ eventReadyPath: sandbox ? node_path_1.default.join(eventDir, MARKER_READY) : undefined,
109
+ eventFailedPath: sandbox ? node_path_1.default.join(eventDir, MARKER_FAILED) : undefined,
110
+ };
111
+ }
112
+ finally {
113
+ // 父进程关掉自己那份 fd;子进程已通过 stdio dup 持有独立副本。
114
+ node_fs_1.default.closeSync(fd);
115
+ }
116
+ }
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.resolveNpmInstallRegistry = exports.installDependencies = exports.writeSparkMeta = exports.readSparkMeta = exports.TEMPLATE_PACKAGE_BY_STACK = exports.SUPPORTED_STACKS = exports.renderTemplate = void 0;
3
+ exports.ASYNC_INSTALL_LOG = exports.isSandboxEnv = exports.runAsyncInstallWorker = exports.dispatchAsyncInstall = exports.resolveNpmInstallRegistry = exports.installDependencies = exports.writeSparkMeta = exports.readSparkMeta = exports.TEMPLATE_PACKAGE_BY_STACK = exports.SUPPORTED_STACKS = exports.renderTemplate = void 0;
4
4
  var template_1 = require("./template");
5
5
  Object.defineProperty(exports, "renderTemplate", { enumerable: true, get: function () { return template_1.renderTemplate; } });
6
6
  Object.defineProperty(exports, "SUPPORTED_STACKS", { enumerable: true, get: function () { return template_1.SUPPORTED_STACKS; } });
@@ -11,3 +11,8 @@ Object.defineProperty(exports, "writeSparkMeta", { enumerable: true, get: functi
11
11
  var install_1 = require("./install");
12
12
  Object.defineProperty(exports, "installDependencies", { enumerable: true, get: function () { return install_1.installDependencies; } });
13
13
  Object.defineProperty(exports, "resolveNpmInstallRegistry", { enumerable: true, get: function () { return install_1.resolveNpmInstallRegistry; } });
14
+ var async_install_1 = require("./async-install");
15
+ Object.defineProperty(exports, "dispatchAsyncInstall", { enumerable: true, get: function () { return async_install_1.dispatchAsyncInstall; } });
16
+ Object.defineProperty(exports, "runAsyncInstallWorker", { enumerable: true, get: function () { return async_install_1.runAsyncInstallWorker; } });
17
+ Object.defineProperty(exports, "isSandboxEnv", { enumerable: true, get: function () { return async_install_1.isSandboxEnv; } });
18
+ Object.defineProperty(exports, "ASYNC_INSTALL_LOG", { enumerable: true, get: function () { return async_install_1.ASYNC_INSTALL_LOG; } });
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@lark-apaas/miaoda-cli",
3
- "version": "0.1.17-alpha.0",
3
+ "version": "0.1.17-alpha.ddef3e9",
4
4
  "description": "Miaoda 平台命令行工具,面向 Agent 调用",
5
5
  "type": "commonjs",
6
6
  "bin": {
@@ -17,20 +17,6 @@
17
17
  "dist",
18
18
  "upgrade"
19
19
  ],
20
- "scripts": {
21
- "build": "bash scripts/build.sh",
22
- "typecheck": "tsc --noEmit -p tsconfig.json",
23
- "lint": "eslint src/ --max-warnings 0",
24
- "format": "prettier --write src/",
25
- "format:check": "prettier --check src/",
26
- "test": "vitest run --project=unit",
27
- "test:watch": "vitest --project=unit",
28
- "test:integration": "vitest run --project=integration",
29
- "dev": "node --import tsx src/main.ts",
30
- "cli": "node --env-file-if-exists=integration/.env --import tsx src/main.ts",
31
- "prepare": "husky",
32
- "prepublishOnly": "pnpm format:check && pnpm lint && pnpm build && pnpm test"
33
- },
34
20
  "keywords": [
35
21
  "miaoda",
36
22
  "cli",
@@ -65,5 +51,16 @@
65
51
  "vitest": "^4.1.4",
66
52
  "xml2js": "^0.6.2"
67
53
  },
68
- "packageManager": "pnpm@10.16.1+sha512.0e155aa2629db8672b49e8475da6226aa4bdea85fdcdfdc15350874946d4f3c91faaf64cbdc4a5d1ab8002f473d5c3fcedcd197989cf0390f9badd3c04678706"
69
- }
54
+ "scripts": {
55
+ "build": "bash scripts/build.sh",
56
+ "typecheck": "tsc --noEmit -p tsconfig.json",
57
+ "lint": "eslint src/ --max-warnings 0",
58
+ "format": "prettier --write src/",
59
+ "format:check": "prettier --check src/",
60
+ "test": "vitest run --project=unit",
61
+ "test:watch": "vitest --project=unit",
62
+ "test:integration": "vitest run --project=integration",
63
+ "dev": "node --import tsx src/main.ts",
64
+ "cli": "node --env-file-if-exists=integration/.env --import tsx src/main.ts"
65
+ }
66
+ }
@@ -5,9 +5,10 @@
5
5
  //
6
6
  // 流程:
7
7
  // 1. env pull —— 拉沙箱身份/凭证到 .env.local
8
- // 2. skills sync —— 同步当前 stack agent skills
9
- // 3. dotenv 加载 .env / .env.local process.env(含 SUDA_WEBUSER 适配)
10
- // 4. 起单进程 dev server(design-stack 单进程,无 server/client 拆分)
8
+ // 2. action-plugin init —— user app package.json.actionPlugins 里声明的插件
9
+ // 3. skills sync —— 同步当前 stack agent skills
10
+ // 4. dotenv 加载 .env / .env.local 到 process.env(含 SUDA_WEBUSER 适配)
11
+ // 5. 起单进程 dev server(design-stack 单进程,无 server/client 拆分)
11
12
  //
12
13
  // 设计同 nestjs-react-fullstack/dev-local.js,SDK 包不需要自己 require('dotenv'),
13
14
  // env 加载收敛在启动脚本单点。SUDA_WEBUSER 适配同 nrf 那份。
@@ -26,7 +27,7 @@ function warn(msg) {
26
27
  if (!process.env.MIAODA_APP_TYPE) process.env.MIAODA_APP_TYPE = '4';
27
28
  process.env.MIAODA_LOCAL_DEV = '1';
28
29
 
29
- console.log('[dev-local] (1/4) env pull...');
30
+ console.log('[dev-local] (1/5) env pull...');
30
31
  const hasLarkCli = spawnSync('command', ['-v', 'lark-cli'], { shell: true, stdio: 'ignore' }).status === 0;
31
32
  if (hasLarkCli) {
32
33
  let appId = '';
@@ -47,15 +48,23 @@ if (hasLarkCli) {
47
48
  warn('lark-cli 未安装,跳过 env pull;请确保 .env.local 已就绪');
48
49
  }
49
50
 
51
+ // action-plugin init —— 装 user app 在 package.json.actionPlugins 里声明的插件。
52
+ console.log('[dev-local] (2/5) action-plugin init...');
53
+ try {
54
+ execSync('npx -y @lark-apaas/fullstack-cli@latest action-plugin init', { stdio: 'inherit' });
55
+ } catch {
56
+ warn('action-plugin init 失败,继续启动');
57
+ }
58
+
50
59
  // skills sync —— --local 切 flat layout;不传 --version,handler 默认 coding-steering@latest。
51
- console.log('[dev-local] (2/4) miaoda skills sync...');
60
+ console.log('[dev-local] (3/5) miaoda skills sync...');
52
61
  try {
53
62
  execSync('npx -y @lark-apaas/miaoda-cli@latest skills sync --local', { stdio: 'inherit' });
54
63
  } catch {
55
64
  console.log(' (skills sync 失败,继续启动)');
56
65
  }
57
66
 
58
- console.log('[dev-local] (3/4) loading .env / .env.local...');
67
+ console.log('[dev-local] (4/5) loading .env / .env.local...');
59
68
  const dotenv = require('dotenv');
60
69
  dotenv.config({ path: '.env.local' });
61
70
  dotenv.config({ path: '.env' });
@@ -75,7 +84,7 @@ if (process.env.SUDA_WEBUSER) {
75
84
  }
76
85
  }
77
86
 
78
- console.log('[dev-local] (4/4) npm run dev');
87
+ console.log('[dev-local] (5/5) npm run dev');
79
88
  const child = spawn('npm', ['run', 'dev'], { stdio: 'inherit', env: process.env });
80
89
  child.on('exit', (code) => process.exit(code ?? 0));
81
90
  child.on('error', (err) => {
@@ -5,9 +5,10 @@
5
5
  //
6
6
  // 流程:
7
7
  // 1. env pull —— 拉沙箱身份/凭证到 .env.local
8
- // 2. skills sync —— 同步当前 stack agent skills
9
- // 3. dotenv 加载 .env / .env.local process.env(含 SUDA_WEBUSER 适配)
10
- // 4. 并发起 dev:server + dev:client(子进程继承 process.env
8
+ // 2. action-plugin init —— user app package.json.actionPlugins 里声明的插件
9
+ // 3. skills sync —— 同步当前 stack agent skills
10
+ // 4. dotenv 加载 .env / .env.local 到 process.env(含 SUDA_WEBUSER 适配)
11
+ // 5. 并发起 dev:server + dev:client(子进程继承 process.env)
11
12
  //
12
13
  // 关键设计:本脚本在 spawn 子进程之前先把 .env / .env.local 加载到 process.env,
13
14
  // 然后 spawn 的 server / client 进程通过 env 继承直接拿到——SDK(fullstack-nestjs-core
@@ -34,7 +35,7 @@ if (!process.env.MIAODA_APP_TYPE) process.env.MIAODA_APP_TYPE = '3';
34
35
  process.env.MIAODA_LOCAL_DEV = '1';
35
36
 
36
37
  // 1. env pull
37
- console.log('[dev-local] (1/4) env pull...');
38
+ console.log('[dev-local] (1/5) env pull...');
38
39
  const hasLarkCli = spawnSync('command', ['-v', 'lark-cli'], { shell: true, stdio: 'ignore' }).status === 0;
39
40
  if (hasLarkCli) {
40
41
  let appId = '';
@@ -55,20 +56,28 @@ if (hasLarkCli) {
55
56
  warn('lark-cli 未安装,跳过 env pull;请确保 .env.local 已就绪');
56
57
  }
57
58
 
58
- // 2. skills sync —— --local 切到 flat layout (.agents/skills + .claude/skills 软链),
59
+ // 2. action-plugin init —— user app package.json.actionPlugins 里声明的插件。
60
+ console.log('[dev-local] (2/5) action-plugin init...');
61
+ try {
62
+ execSync('npx -y @lark-apaas/fullstack-cli@latest action-plugin init', { stdio: 'inherit' });
63
+ } catch {
64
+ warn('action-plugin init 失败,继续启动');
65
+ }
66
+
67
+ // 3. skills sync —— --local 切到 flat layout (.agents/skills + .claude/skills 软链),
59
68
  // 跟沙箱 nested layout 区分。不传 --version,handler 默认拉 coding-steering@latest,
60
69
  // 保证每次本地 npm run dev 都把 skills 升到最新。
61
- console.log('[dev-local] (2/4) miaoda skills sync...');
70
+ console.log('[dev-local] (3/5) miaoda skills sync...');
62
71
  try {
63
72
  execSync('npx -y @lark-apaas/miaoda-cli@latest skills sync --local', { stdio: 'inherit' });
64
73
  } catch {
65
74
  console.log(' (skills sync 失败,继续启动)');
66
75
  }
67
76
 
68
- // 3. 加载 .env / .env.local 到 process.env
77
+ // 4. 加载 .env / .env.local 到 process.env
69
78
  // dotenv 默认 override:false,先到先得 → 先 .env.local 让它优先于 .env;
70
79
  // shell env 已在 process.env,两次 config 都不会覆盖。
71
- console.log('[dev-local] (3/4) loading .env / .env.local...');
80
+ console.log('[dev-local] (4/5) loading .env / .env.local...');
72
81
  const dotenv = require('dotenv');
73
82
  dotenv.config({ path: '.env.local' });
74
83
  dotenv.config({ path: '.env' });
@@ -89,8 +98,8 @@ if (process.env.SUDA_WEBUSER) {
89
98
  }
90
99
  }
91
100
 
92
- // 4. 并发起前后端 dev server
93
- console.log('[dev-local] (4/4) 并发起 dev:server + dev:client');
101
+ // 5. 并发起前后端 dev server
102
+ console.log('[dev-local] (5/5) 并发起 dev:server + dev:client');
94
103
  const child = spawn(
95
104
  'npx',
96
105
  [