@lark-apaas/miaoda-cli 0.1.19-alpha.6dad9cd → 0.1.19-alpha.834fd32

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 CHANGED
@@ -11,12 +11,17 @@ pnpm add -g @lark-apaas/miaoda-cli
11
11
  ## 使用
12
12
 
13
13
  ```bash
14
- # 设置默认应用(需要应用上下文的命令可用 --app-id 覆盖)
14
+ # 设置默认应用(需要应用上下文的命令从 MIAODA_APP_ID 读取)
15
15
  export MIAODA_APP_ID=app_demo_xxx
16
16
 
17
17
  # 文件操作
18
18
  miaoda file ls
19
19
  miaoda file upload ./local/path
20
+
21
+ # 评论操作
22
+ miaoda comment list --json
23
+ miaoda comment list --only-unresolved --json
24
+ miaoda comment resolve 1703677660120110076 --json
20
25
  ```
21
26
 
22
27
  JSON 结构化输出(Agent 推荐):
@@ -41,9 +46,37 @@ miaoda file ls --output json
41
46
  | `miaoda file ...` | 文件操作:上传、下载、元数据、签名下载、批量删除 |
42
47
  | `miaoda db ...` | 数据操作:SQL 执行、表结构查询、数据导入导出 |
43
48
  | `miaoda observability ...` | 线上日志、链路、前端源码堆栈反查、监控指标、运营指标 |
49
+ | `miaoda comment ...` | 应用评论:查看评论列表、解决评论 |
50
+ | `miaoda registry ...` | 组件 registry:列出 / copy-in design-html SDK 文件(本地执行) |
44
51
 
45
52
  完整命令通过 `miaoda --help` 或 `miaoda <domain> --help` 查看。
46
53
 
54
+ ## 组件 registry(design-html)
55
+
56
+ `miaoda registry` 为 design-html 技术栈提供 shadcn 式的「组件 registry」:把 SDK 文件**拷进**当前项目(copy-in),而非作为依赖安装。registry 是独立 npm 包,按运行时分发(design-html → buildless runtime → `@lark-apaas/coding-registry-buildless`)。
57
+
58
+ 前置:当前目录(或 `--dir`)已走过 `miaoda app init`,`.spark/meta.json` 含 `stack=design-html`。
59
+
60
+ ```bash
61
+ # 列出可用 SDK 条目(输出各文件 USAGE 注释块原文,便于 Agent 直接读用法)
62
+ miaoda registry list
63
+ miaoda registry list --json
64
+
65
+ # 把条目 copy-in 到当前项目(自动按 USAGE 块的 Depends 解析依赖闭包一并拷入)
66
+ miaoda registry add deck-slide
67
+ miaoda registry add foo.css theme --dry-run # 只报告将写 / 将跳,不落盘
68
+ miaoda registry add deck-slide --overwrite # 目标已存在时覆盖(默认跳过以保护改动)
69
+ ```
70
+
71
+ 要点:
72
+
73
+ - **条目**:registry 包根 `components/` · `scripts/` · `stylesheets/` 三个顶层目录(任意深度)下的代码文件即条目。USAGE 注释块**可选**(有则提供描述 + Depends,无则空描述、无依赖);包根直放文件、其它目录、非代码文件都不算条目。
74
+ - **寻址**:`add` / `Depends` 的 token 可写三种形式 —— `name`(去扩展名,如 `foo`)/ 完整文件名(如 `foo.css`)/ relPath(如 `stylesheets/foo.css`)。`name` **不要求唯一**,`list --json` 的 `file`(relPath)是永远无歧义的 canonical 地址。
75
+ - **多匹配全拉**:token 命中多个同名条目时全部 copy-in(如 `add foo` 把 `foo.jsx` + `foo.css` 一起拉),想只要其一就用完整文件名或 relPath。
76
+ - **拍平落地**:所有文件**拍平**到项目根(`scripts/deck-slide.js` → 项目 `deck-slide.js`,不保留子目录);闭包内同名 basename 撞车时后者覆盖前者。
77
+ - **skip-by-disk**:项目已有同名(拍平后的 basename)文件默认跳过,`--overwrite` 才覆盖;不写任何安装记账文件,项目文件树即状态。
78
+ - `--version <ver>` 指定 registry 包版本或 dist-tag(缺省 `latest`),`--dir <path>` 指定项目目录(缺省当前目录)。
79
+
47
80
  ## 全局参数
48
81
 
49
82
  - `--json [fields]`:输出结构化 JSON,可选字段级选择(如 `--json id,name`)。
@@ -0,0 +1,24 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getComments = getComments;
4
+ exports.resolveComment = resolveComment;
5
+ const http_1 = require("../../utils/http");
6
+ /**
7
+ * 评论接口走管理端 inner-api(getHttpClient + 小写 status_code 信封),与 app / deploy /
8
+ * observability 等域一致:统一的 verbose 请求/响应日志、错误映射、信封解析,不再域内自实现。
9
+ */
10
+ /** GET /api/v1/studio/innerapi/apps/:appID/comments — 获取应用评论列表 */
11
+ async function getComments(appID, onlyUnresolved = false) {
12
+ let url = `/api/v1/studio/innerapi/apps/${encodeURIComponent(appID)}/comments`;
13
+ if (onlyUnresolved)
14
+ url += '?onlyUnresolved=true';
15
+ const body = await (0, http_1.getInnerApi)(url, {
16
+ errPrefix: 'Failed to list comments',
17
+ });
18
+ return body.comments ?? [];
19
+ }
20
+ /** POST /api/v1/studio/innerapi/apps/:appID/comments/:commentID/resolve — 解决评论 */
21
+ async function resolveComment(appID, commentID) {
22
+ const url = `/api/v1/studio/innerapi/apps/${encodeURIComponent(appID)}/comments/${encodeURIComponent(commentID)}/resolve`;
23
+ await (0, http_1.postInnerApi)(url, undefined, { errPrefix: 'Failed to resolve comment' });
24
+ }
@@ -0,0 +1,6 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.resolveComment = exports.getComments = void 0;
4
+ var api_1 = require("./api");
5
+ Object.defineProperty(exports, "getComments", { enumerable: true, get: function () { return api_1.getComments; } });
6
+ Object.defineProperty(exports, "resolveComment", { enumerable: true, get: function () { return api_1.resolveComment; } });
@@ -0,0 +1,2 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
package/dist/api/index.js CHANGED
@@ -33,16 +33,18 @@ var __importStar = (this && this.__importStar) || (function () {
33
33
  };
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
- exports.deploy = exports.app = exports.observability = exports.db = exports.file = exports.plugin = void 0;
36
+ exports.comment = exports.deploy = exports.app = exports.observability = exports.db = exports.file = exports.plugin = void 0;
37
37
  const _plugin = __importStar(require("../api/plugin/index"));
38
38
  const _file = __importStar(require("../api/file/index"));
39
39
  const _db = __importStar(require("../api/db/index"));
40
40
  const _observability = __importStar(require("../api/observability/index"));
41
41
  const _app = __importStar(require("../api/app/index"));
42
42
  const _deploy = __importStar(require("../api/deploy/index"));
43
+ const _comment = __importStar(require("../api/comment/index"));
43
44
  exports.plugin = { ..._plugin };
44
45
  exports.file = { ..._file };
45
46
  exports.db = { ..._db };
46
47
  exports.observability = { ..._observability };
47
48
  exports.app = { ..._app };
48
49
  exports.deploy = { ..._deploy };
50
+ exports.comment = { ..._comment };
@@ -0,0 +1,55 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.registerCommentCommands = registerCommentCommands;
4
+ const shared_1 = require("../../../cli/commands/shared");
5
+ const index_1 = require("../../../cli/handlers/comment/index");
6
+ function registerCommentCommands(program) {
7
+ const commentCmd = program
8
+ .command('comment')
9
+ .description('应用评论:查看评论列表、解决评论')
10
+ .usage('<command> [flags]');
11
+ commentCmd.action(() => {
12
+ commentCmd.outputHelp();
13
+ });
14
+ commentCmd.addHelpText('after', `
15
+ 作用范围
16
+ 操作当前应用的评论。应用上下文:--app-id <id> 或环境变量 MIAODA_APP_ID。
17
+ `);
18
+ const listCmd = commentCmd
19
+ .command('list')
20
+ .description('查看应用评论列表')
21
+ .addOption((0, shared_1.appIdOption)().hideHelp())
22
+ .option('--only-unresolved', '仅返回未解决的评论', false)
23
+ .addHelpText('after', `
24
+ JSON 输出
25
+ {"data": [{"comment_id": "...", "content": "...", "user_id": "...",
26
+ "finish": 0, "resolved": false, "create_time": 1700000000}]}
27
+
28
+ 示例
29
+ $ miaoda comment list --json
30
+ $ miaoda comment list --only-unresolved --json
31
+ `);
32
+ listCmd.action((0, shared_1.withHelp)(listCmd, async (rawOpts) => {
33
+ (0, shared_1.rejectCliOverride)(listCmd, 'appId');
34
+ await (0, index_1.handleCommentList)({
35
+ appId: rawOpts.appId,
36
+ onlyUnresolved: rawOpts.onlyUnresolved,
37
+ });
38
+ }));
39
+ const resolveCmd = commentCmd
40
+ .command('resolve')
41
+ .description('把一条评论标记为已解决')
42
+ .argument('<commentID>', '待解决的评论 ID')
43
+ .addOption((0, shared_1.appIdOption)().hideHelp())
44
+ .addHelpText('after', `
45
+ JSON 输出
46
+ {"data": {"comment_id": "...", "resolved": true}}
47
+
48
+ 示例
49
+ $ miaoda comment resolve 1703677660120110076 --json
50
+ `);
51
+ resolveCmd.action((0, shared_1.withHelp)(resolveCmd, async (commentId, rawOpts) => {
52
+ (0, shared_1.rejectCliOverride)(resolveCmd, 'appId');
53
+ await (0, index_1.handleCommentResolve)(commentId, { appId: rawOpts.appId });
54
+ }));
55
+ }
@@ -59,15 +59,17 @@ JSON 输出(stdout)
59
59
  }));
60
60
  const patchCmd = deployCmd
61
61
  .command('patch')
62
- .description('design-html 增量发布:按文件 create/update/delete,.html 变化时同步 routes.json')
62
+ .description('design-html 增量发布:按文件 create/update/delete,.html 增删时同步 routes.json')
63
63
  .option('--create <relpath>', '新增文件(可重复)', shared_1.collectRepeatedOption, [])
64
64
  .option('--update <relpath>', '覆盖已有文件(可重复)', shared_1.collectRepeatedOption, [])
65
65
  .option('--delete <relpath>', '删除文件(可重复)', shared_1.collectRepeatedOption, [])
66
66
  .addHelpText('after', `
67
67
  --dir 为项目目录(与 deploy 同名参数,默认当前目录)。
68
68
  仅支持 design-html(design_local_deploy)。路径为工程根相对路径(= 服务端 latest 下 key),
69
- 禁绝对路径与 ".."。--create/--update 的本地文件必须存在。改动集含 .html 时自动重算并随包
70
- 更新 routes.json。
69
+ 禁绝对路径与 ".."。--create/--update 的本地文件必须存在。.html 被新增/删除(路由集合变化)
70
+ 时自动重算并随包更新 routes.json;纯 --update 现有 .html 只改内容,不动 routes.json
71
+ routes.json 每项为 {path, file}:path 为相对路径(不含 base 前缀,由消费侧拼接),file 为
72
+ 对应源文件(如 / → index.html)。
71
73
 
72
74
  JSON 输出
73
75
  {"data": {"upsertCount": <n>, "deleteCount": <n>, "actionsSent": <n>, "routesRegenerated": <bool>}}
@@ -9,7 +9,8 @@ const index_4 = require("../../cli/commands/app/index");
9
9
  const index_5 = require("../../cli/commands/deploy/index");
10
10
  const modern_1 = require("../../cli/commands/deploy/modern");
11
11
  const index_6 = require("../../cli/commands/skills/index");
12
- const index_7 = require("../../cli/commands/registry/index");
12
+ const index_7 = require("../../cli/commands/comment/index");
13
+ const index_8 = require("../../cli/commands/registry/index");
13
14
  // scene 跟 dispatcher(MIAODA_APP_TYPE)同语义层级 —— app 业务类型维度,
14
15
  // 对齐后端 devops app_common.AppType 枚举:
15
16
  // 3 = AppType_APPLICATION(全栈应用,当前仅 nestjs-react-fullstack 一个 stack)
@@ -41,11 +42,13 @@ const SCENE_REGISTRARS = {
41
42
  (0, index_1.registerFileCommands)(p);
42
43
  (0, index_3.registerObservabilityCommands)(p);
43
44
  (0, index_6.registerSkillsCommands)(p);
45
+ (0, index_7.registerCommentCommands)(p);
44
46
  },
45
47
  modern: (p) => {
46
48
  (0, index_4.registerAppCommands)(p, { includeInit: true });
47
49
  (0, modern_1.registerDeployCommandsModern)(p);
48
50
  (0, index_6.registerSkillsCommands)(p);
51
+ (0, index_7.registerCommentCommands)(p);
49
52
  },
50
53
  // application scene(AppType_APPLICATION=3):
51
54
  // 在 default 命令集基础上加 skills(按 meta.json.stack 拉对应 stack 技能)
@@ -57,6 +60,7 @@ const SCENE_REGISTRARS = {
57
60
  (0, index_1.registerFileCommands)(p);
58
61
  (0, index_3.registerObservabilityCommands)(p);
59
62
  (0, index_6.registerSkillsCommands)(p);
63
+ (0, index_7.registerCommentCommands)(p);
60
64
  },
61
65
  // design scene(AppType_DESIGN=4):design-stack 仅 SSR 渲染、无后端业务逻辑、
62
66
  // 无数据库、无 UGC 文件。命令集裁掉 db / file,保留 deploy / observability / app(init) / skills。
@@ -65,6 +69,7 @@ const SCENE_REGISTRARS = {
65
69
  (0, index_5.registerDeployCommands)(p);
66
70
  (0, index_3.registerObservabilityCommands)(p);
67
71
  (0, index_6.registerSkillsCommands)(p);
72
+ (0, index_7.registerCommentCommands)(p);
68
73
  },
69
74
  // design-html scene(AppType_DESIGN_HTML=8):buildless 静态 HTML,
70
75
  // 命令集与 modern 一致(app init/sync + modern 拆分版 deploy + skills),不挂 db/file/observability。
@@ -73,7 +78,8 @@ const SCENE_REGISTRARS = {
73
78
  (0, index_4.registerAppCommands)(p, { includeInit: true });
74
79
  (0, modern_1.registerDeployCommandsModern)(p);
75
80
  (0, index_6.registerSkillsCommands)(p);
76
- (0, index_7.registerRegistryCommands)(p);
81
+ (0, index_8.registerRegistryCommands)(p);
82
+ (0, index_7.registerCommentCommands)(p);
77
83
  },
78
84
  };
79
85
  function readEnv(name) {
@@ -7,9 +7,9 @@ const index_1 = require("../../../cli/handlers/registry/index");
7
7
  * miaoda registry:design-html 技术栈的「组件 registry」(shadcn 式 copy-in)。
8
8
  *
9
9
  * registry 是独立 npm 包,按 runtime 命名分发(buildless runtime → @lark-apaas/coding-registry-buildless)。
10
- * SDK 文件按目录组织**平铺在包根**(scripts/ · components/ · stylesheets/),每个文件以 USAGE 注释块
11
- * 开头(声明描述 / 用法 / Depends)。CLI 抓 tarball、按 .spark/meta.json.stack → runtime → 包名 选包,
12
- * SDK 文件**保留相对包根的路径** copy-in 到用户项目根。
10
+ * SDK 文件放在包根的 components/ · scripts/ · stylesheets/ 三个目录下(这三个顶层目录里的代码文件
11
+ * 才算条目);文件可带 USAGE 注释块(可选,声明描述 / 用法 / Depends)。CLI 抓 tarball、按
12
+ * .spark/meta.json.stack → runtime → 包名 选包,把 SDK 文件**拍平**copy-in 到用户项目根。
13
13
  * 不写安装记账文件——项目文件树即状态(skip-by-disk)。
14
14
  */
15
15
  function registerRegistryCommands(program) {
@@ -22,11 +22,14 @@ function registerRegistryCommands(program) {
22
22
  });
23
23
  registryCmd.addHelpText('after', `
24
24
  概念
25
- - registry 包:独立 npm 包(按 runtime 命名),SDK 文件平铺在包根(如 scripts/deck-stage.js)
26
- - 条目 (entry):name = 文件名去扩展名(全树唯一);每个文件以 USAGE 注释块开头
27
- - USAGE 块:声明描述 / Exports / Usage / Depends(可选,依赖其他 SDK 的 name)
25
+ - registry 包:独立 npm 包(按 runtime 命名)
26
+ - 条目 (entry):包根 components/ · scripts/ · stylesheets/ 顶层目录(任意深度)下的代码文件;
27
+ name = 文件名去扩展名(不要求唯一)。包根直放文件、其它目录、非代码文件都不算条目
28
+ - 寻址:add / Depends 的 token 可写三种形式——name(foo)/ 完整文件名(foo.css)/ relPath(stylesheets/foo.css)
29
+ - 多匹配全拉:token 命中多个同名条目时全部 copy-in(不报歧义);想只要其一就写完整文件名 / relPath
30
+ - USAGE 块:可选元数据,声明描述 / Exports / Usage / Depends;无 USAGE 也是条目(空描述、无依赖)
28
31
  - Depends:从 USAGE 块解析的跨条目依赖,add 时传递解析、一并 copy-in
29
- - add 落地:保留文件相对包根的路径(包根 scripts/x.js → 项目 scripts/x.js
32
+ - add 落地:拍平到项目根(包根 scripts/x.js → 项目 x.js);同名 basename 撞车时后者覆盖前者
30
33
 
31
34
  前置
32
35
  当前目录(或 --dir)已走过 'miaoda app init'(.spark/meta.json 含 stack=design-html)
@@ -62,23 +65,25 @@ JSON 输出
62
65
  function registerRegistryAdd(parent) {
63
66
  const cmd = parent
64
67
  .command('add')
65
- .description('把一个或多个 SDK 文件 copy-in 到当前项目(保留目录结构,含 Depends 闭包)')
66
- .argument('<name...>', '条目名(文件名去扩展名),可传多个;Depends 会自动传递解析')
68
+ .description('把一个或多个 SDK 文件 copy-in 到当前项目(拍平到项目根,含 Depends 闭包)')
69
+ .argument('<token...>', 'name / 完整文件名 / relPath,可传多个;同名多匹配全拉,Depends 自动传递解析')
67
70
  .option('--dir <path>', '项目目录,默认当前目录', '.')
68
71
  .option('--version <ver>', 'registry 包版本或 dist-tag,缺省 latest')
69
72
  .option('--overwrite', '目标文件已存在时覆盖(默认跳过以保护用户改动)', false)
70
73
  .option('--dry-run', '只报告将写 / 将跳哪些文件,不落盘', false)
71
74
  .addHelpText('after', `
72
75
  行为
73
- - 解析 USAGE Depends 传递闭包,把 SDK 文件从包根 <rel> 按相对路径拷到项目根
74
- (包根 scripts/deck-stage.js 项目 scripts/deck-stage.js,1:1 保留路径)
76
+ - token 寻址:name(foo)/ 完整文件名(foo.css)/ relPath(stylesheets/foo.css)任一形式
77
+ - 多匹配全拉:token 命中多个同名条目时全部 copy-in;想只要其一就写完整文件名 / relPath
78
+ - 解析 USAGE 块 Depends 传递闭包,把 SDK 文件**拍平**拷到项目根
79
+ (包根 scripts/deck-stage.js → 项目 deck-stage.js;同名 basename 撞车时后者覆盖前者)
75
80
  - skip-by-disk:目标文件已存在默认跳过;--overwrite 才覆盖
76
81
  - 不写任何安装记账文件(项目文件树即状态)
77
82
 
78
83
  JSON 输出
79
84
  {"data": {"registryVersion": "...", "requested": [...], "resolved": [...],
80
85
  "added": [...], "skipped": [...], "dryRun": false}}
81
- (added / skipped 是项目内相对路径,如 scripts/deck-stage.js)
86
+ resolved 为来源 relPath;added / skipped 为拍平后的落地 basename,如 deck-stage.js)
82
87
 
83
88
  示例
84
89
  $ miaoda registry add deck-slide
@@ -0,0 +1,7 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.handleCommentResolve = exports.handleCommentList = void 0;
4
+ var list_1 = require("./list");
5
+ Object.defineProperty(exports, "handleCommentList", { enumerable: true, get: function () { return list_1.handleCommentList; } });
6
+ var resolve_1 = require("./resolve");
7
+ Object.defineProperty(exports, "handleCommentResolve", { enumerable: true, get: function () { return resolve_1.handleCommentResolve; } });
@@ -0,0 +1,62 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.handleCommentList = handleCommentList;
37
+ const api = __importStar(require("../../../api/index"));
38
+ const shared_1 = require("../../../cli/commands/shared");
39
+ const output_1 = require("../../../utils/output");
40
+ const listSchema = {
41
+ columns: [
42
+ { key: 'comment_id', label: 'id' },
43
+ { key: 'finish' },
44
+ { key: 'resolved' },
45
+ { key: 'user_id', label: 'user' },
46
+ { key: 'create_time', label: 'time', format: output_1.fmt.sec() },
47
+ { key: 'content', format: output_1.fmt.truncate(80) },
48
+ ],
49
+ };
50
+ async function handleCommentList(opts) {
51
+ const appId = (0, shared_1.resolveAppId)({ appId: opts.appId });
52
+ const comments = await api.comment.getComments(appId, opts.onlyUnresolved ?? false);
53
+ const rows = comments.map((c) => ({
54
+ comment_id: c.commentID,
55
+ content: c.content,
56
+ user_id: c.userID,
57
+ finish: c.finish,
58
+ resolved: c.finish === 1,
59
+ create_time: c.createTime,
60
+ }));
61
+ (0, output_1.emit)({ data: rows }, listSchema);
62
+ }
@@ -0,0 +1,44 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.handleCommentResolve = handleCommentResolve;
37
+ const api = __importStar(require("../../../api/index"));
38
+ const shared_1 = require("../../../cli/commands/shared");
39
+ const output_1 = require("../../../utils/output");
40
+ async function handleCommentResolve(commentId, opts) {
41
+ const appId = (0, shared_1.resolveAppId)({ appId: opts.appId });
42
+ await api.comment.resolveComment(appId, commentId);
43
+ (0, output_1.emitOk)({ comment_id: commentId, resolved: true });
44
+ }
@@ -41,7 +41,7 @@ async function handleRegistryAdd(opts) {
41
41
  data: {
42
42
  registryVersion: reg.version,
43
43
  requested: opts.names,
44
- resolved: plan.entries.map((e) => e.name),
44
+ resolved: plan.entries.map((e) => e.relPath),
45
45
  added: result.added,
46
46
  skipped: result.skipped,
47
47
  dryRun: opts.dryRun === true,
@@ -36,7 +36,8 @@ function isHtml(rel) {
36
36
  /**
37
37
  * 把 changeset 组成 TosFileAction[]:
38
38
  * - create/update 读本地文件、自动编码;delete 仅 FilePath。
39
- * - 任一改动是 .html 追加 routes.json(TS 重算)的 UPDATE action
39
+ * - .html create / delete(路由集合变化)时,追加重算 routes.json UPDATE action
40
+ * 纯 update 现有 .html 只改内容、不改路由集合,不重算 routes.json。
40
41
  */
41
42
  function buildTosActions(projectDir, cs) {
42
43
  const actions = [];
@@ -47,8 +48,8 @@ function buildTosActions(projectDir, cs) {
47
48
  for (const rel of cs.deletes) {
48
49
  actions.push({ actionType: index_1.TosActionType.DELETE, filePath: normalizeRel(rel) });
49
50
  }
50
- const touchesHtml = [...cs.creates, ...cs.updates, ...cs.deletes].some(isHtml);
51
- if (touchesHtml) {
51
+ const routeSetChanged = [...cs.creates, ...cs.deletes].some(isHtml);
52
+ if (routeSetChanged) {
52
53
  actions.push({
53
54
  actionType: index_1.TosActionType.UPDATE,
54
55
  filePath: 'routes.json',
@@ -11,8 +11,8 @@ const actions_1 = require("./actions");
11
11
  var actions_2 = require("./actions");
12
12
  Object.defineProperty(exports, "buildTosActions", { enumerable: true, get: function () { return actions_2.buildTosActions; } });
13
13
  /**
14
- * design-html 增量发布:读本地文件组 TosFileAction(含 .html 时带 routes.json),
15
- * 调后端 applyTosDiff 应用到 latest。scope 限 design_local_deploy。
14
+ * design-html 增量发布:读本地文件组 TosFileAction(create/delete .html 时带重算的
15
+ * routes.json),调后端 applyTosDiff 应用到 latest。scope 限 design_local_deploy。
16
16
  */
17
17
  async function patchDesignDeploy(opts) {
18
18
  if (opts.creates.length + opts.updates.length + opts.deletes.length === 0) {
@@ -2,27 +2,36 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.generateRoutes = generateRoutes;
4
4
  const source_scan_1 = require("./source-scan");
5
- /** 与 build.sh 一致的 .html → route path 推导。rel 为 posix 相对路径。 */
6
- function toRoute(rel, base) {
5
+ /**
6
+ * .html → route path 推导(相对路径,不带 base 前缀)。rel 为 posix 相对路径。
7
+ * base 前缀交由消费侧(运行时/网关)自行拼接,design-html 链路不在 routes.json 内带前缀。
8
+ */
9
+ function toRoute(rel) {
7
10
  let r = '/' + rel;
8
11
  r = r.replace(/\.html$/i, '');
9
12
  r = r.replace(/(^|\/)index$/, '$1');
10
13
  if (r.length > 1)
11
14
  r = r.replace(/\/$/, '');
12
- const full = base + r;
13
- return full === '' ? '/' : full;
15
+ return r === '' ? '/' : r;
14
16
  }
15
17
  /**
16
- * 扫描 projectDir 下所有 .html(套 source-scan 的 EXCLUDES),生成与 build.sh 同款 routes.json
17
- * 文本(含尾换行)。CLIENT_BASE_PATH 取自 process.env(平台预置),去尾斜杠后作前缀。
18
+ * 扫描 projectDir 下所有 .html(套 source-scan 的 EXCLUDES),生成 routes.json 文本(含尾换行)。
19
+ * 每项为 { path, file }:path 为相对路径(不带 CLIENT_BASE_PATH 前缀,由消费侧拼接),
20
+ * file 为该路由对应的原始 .html 源文件。按 path 排序、同 path 取字典序靠前的 file 去重。
21
+ * 零 html 时输出空数组——不伪造解析不到文件的根路由。
18
22
  */
19
23
  function generateRoutes(projectDir) {
20
- const base = (process.env.CLIENT_BASE_PATH ?? '').replace(/\/+$/, '');
21
24
  const htmls = (0, source_scan_1.listSourceFiles)(projectDir).filter((f) => f.toLowerCase().endsWith('.html'));
22
- const routes = [...new Set(htmls.map((rel) => toRoute(rel, base)))]
23
- .sort()
24
- .map((p) => ({ path: p }));
25
- if (routes.length === 0)
26
- routes.push({ path: base ? `${base}/` : '/' });
25
+ const byPath = new Map();
26
+ for (const entry of htmls.map((file) => ({ path: toRoute(file), file })).sort(byPathThenFile)) {
27
+ if (!byPath.has(entry.path))
28
+ byPath.set(entry.path, entry);
29
+ }
30
+ const routes = [...byPath.values()];
27
31
  return JSON.stringify(routes, null, 2) + '\n';
28
32
  }
33
+ function byPathThenFile(a, b) {
34
+ return a.path === b.path
35
+ ? (a.file ?? '').localeCompare(b.file ?? '')
36
+ : a.path.localeCompare(b.path);
37
+ }
@@ -8,7 +8,7 @@ exports.resolveRegistryPackage = resolveRegistryPackage;
8
8
  exports.loadRegistry = loadRegistry;
9
9
  exports.extractUsageBlock = extractUsageBlock;
10
10
  exports.parseDependsOn = parseDependsOn;
11
- exports.indexEntries = indexEntries;
11
+ exports.tokenMatches = tokenMatches;
12
12
  exports.resolveDependencyClosure = resolveDependencyClosure;
13
13
  exports.planAdd = planAdd;
14
14
  exports.applyAdd = applyAdd;
@@ -95,12 +95,21 @@ function loadRegistry(opts) {
95
95
  */
96
96
  const SDK_FILE_EXTENSIONS = new Set(['.js', '.jsx', '.mjs', '.cjs', '.ts', '.tsx', '.css']);
97
97
  /**
98
- * 从包根递归扫整棵树,提取含 USAGE 块的 SDK 条目(任意子目录深度)。
99
- * 只看代码文件(SDK_FILE_EXTENSIONS);条目 name 全树唯一,撞名抛错。
98
+ * 条目所在的顶层目录白名单。只有包根下这几个目录(任意深度)里的代码文件才算条目,
99
+ * 包根直放的文件(eslint.config.js 等)、其它顶层目录(lib/ 等)一律排除——
100
+ * 避免 USAGE 可选后把工程配置 / 非 SDK 代码误判为条目。
101
+ */
102
+ const SDK_TOP_DIRS = new Set(['components', 'scripts', 'stylesheets']);
103
+ /**
104
+ * 从包根递归扫整棵树,提取 SDK 条目。条目判据:
105
+ * 1. 在白名单顶层目录下(SDK_TOP_DIRS,任意深度);
106
+ * 2. 是代码文件(SDK_FILE_EXTENSIONS)。
107
+ * USAGE 块**可选**:有则解析描述 + Depends,无则 usage 为空、dependsOn 为空——仍是条目。
108
+ * name(去扩展名 basename)**不要求唯一**:同名的不同文件全部收录为独立条目,由 relPath
109
+ * (树内唯一)区分;寻址 / 拉取由 tokenMatches + resolveDependencyClosure 处理(同名 → 全拉)。
100
110
  */
101
111
  function scanEntries(registryRoot) {
102
112
  const entries = [];
103
- const byName = new Map(); // name → relPath(撞名诊断)
104
113
  const walk = (dir) => {
105
114
  for (const dirent of node_fs_1.default.readdirSync(dir, { withFileTypes: true })) {
106
115
  const absPath = node_path_1.default.join(dir, dirent.name);
@@ -110,23 +119,17 @@ function scanEntries(registryRoot) {
110
119
  }
111
120
  if (!dirent.isFile())
112
121
  continue;
113
- // 只扫代码文件,排除 README/LICENSE/package.json 等(它们正文可能含 USAGE 字面量)
122
+ // 只扫代码文件,排除 README/LICENSE/package.json 等非 SDK 文件
114
123
  if (!SDK_FILE_EXTENSIONS.has(node_path_1.default.extname(dirent.name).toLowerCase()))
115
124
  continue;
116
- const content = node_fs_1.default.readFileSync(absPath, 'utf-8');
117
- const usage = extractUsageBlock(content);
118
- if (usage === null)
119
- continue; // 非 SDK 文件(无 USAGE 块)
120
125
  // 相对包根的路径,统一用 '/' 分隔(落地时再按平台 join)
121
126
  const relPath = node_path_1.default.relative(registryRoot, absPath).split(node_path_1.default.sep).join('/');
122
- const name = stripExt(dirent.name);
123
- const prev = byName.get(name);
124
- if (prev !== undefined) {
125
- throw new error_1.AppError('REGISTRY_DUPLICATE_NAME', `registry 条目名 "${name}" 重复:${prev} 与 ${relPath}`);
126
- }
127
- byName.set(name, relPath);
127
+ // 只认白名单顶层目录下的文件(顶层段 SDK_TOP_DIRS)
128
+ if (!SDK_TOP_DIRS.has(relPath.split('/')[0]))
129
+ continue;
130
+ const usage = extractUsageBlock(node_fs_1.default.readFileSync(absPath, 'utf-8')) ?? '';
128
131
  entries.push({
129
- name,
132
+ name: stripExt(dirent.name),
130
133
  relPath,
131
134
  usage,
132
135
  dependsOn: parseDependsOn(usage),
@@ -135,8 +138,8 @@ function scanEntries(registryRoot) {
135
138
  }
136
139
  };
137
140
  walk(registryRoot);
138
- // 稳定排序,便于 list 输出可预期
139
- entries.sort((a, b) => a.name.localeCompare(b.name));
141
+ // relPath(唯一)稳定排序,便于 list 输出可预期
142
+ entries.sort((a, b) => a.relPath.localeCompare(b.relPath));
140
143
  return entries;
141
144
  }
142
145
  /** 提取 BEGIN/END USAGE 标记之间的原文(含标记行)。无 USAGE 块返回 null。 */
@@ -172,62 +175,81 @@ function stripExt(file) {
172
175
  const ext = node_path_1.default.extname(file);
173
176
  return ext === '' ? file : file.slice(0, -ext.length);
174
177
  }
175
- /** name → entry 索引,便于依赖解析。 */
176
- function indexEntries(entries) {
177
- const map = new Map();
178
- for (const e of entries)
179
- map.set(e.name, e);
180
- return map;
178
+ /**
179
+ * 一个 token 是否命中某条目。token 可写三种寻址形式(按特异度,任一相等即命中):
180
+ * - relPath:相对包根路径,如 components/foo.jsx(含 '/')
181
+ * - 完整文件名:含扩展名的 basename,如 foo.jsx
182
+ * - name:去扩展名的 basename,如 foo
183
+ * 三种形式无歧义重叠(relPath 含 '/',basename 不含;完整文件名含原扩展名,name 不含),
184
+ * 故用一个等值谓词即可。多个条目可同时命中同一 token(同名 → 全拉),由调用方处理。
185
+ */
186
+ function tokenMatches(entry, token) {
187
+ if (token === entry.relPath)
188
+ return true;
189
+ const fileName = entry.relPath.split('/').pop() ?? '';
190
+ if (token === fileName)
191
+ return true;
192
+ return token === entry.name;
181
193
  }
182
194
  /**
183
- * 解析 Depends 的传递闭包:从 requested 出发,递归把所有依赖一并纳入。
195
+ * 解析 Depends 的传递闭包:从 requested token 出发,递归把所有依赖一并纳入。
184
196
  *
185
- * 返回拓扑序(依赖在前、被依赖在后),便于按序拷贝。检测循环依赖并抛错;
186
- * 引用了不存在的条目名也抛错(fail-fast,避免拷半套)。
197
+ * token 经 tokenMatches 解析成**一组**条目(同名 → 全拉),逐个展开。返回拓扑序
198
+ * (依赖在前、被依赖在后),便于按序拷贝。token 零命中抛 REGISTRY_ENTRY_NOT_FOUND;
199
+ * 检测循环依赖并抛 REGISTRY_CYCLE。visited / onStack / 环检测均以 relPath(唯一)为键,
200
+ * 故同名(去扩展名)的不同文件互不干扰、不会误判成环。
187
201
  */
188
202
  function resolveDependencyClosure(entries, requested) {
189
- const index = indexEntries(entries);
190
203
  const ordered = [];
191
- const visited = new Set();
192
- const onStack = new Set();
193
- const visit = (name, trail) => {
194
- if (visited.has(name))
195
- return;
196
- if (onStack.has(name)) {
197
- throw new error_1.AppError('REGISTRY_CYCLE', `registry 条目依赖成环:${[...trail, name].join(' -> ')}`);
198
- }
199
- const entry = index.get(name);
200
- if (entry === undefined) {
201
- // requested 顶层缺失 vs 依赖链里缺失,都报清楚是谁引的
204
+ const visited = new Set(); // by relPath
205
+ const onStack = new Set(); // by relPath
206
+ // token 命中条目集合;零命中 fail-fast(避免拷半套)
207
+ const resolveToken = (token, trail) => {
208
+ const matches = entries.filter((e) => tokenMatches(e, token));
209
+ if (matches.length === 0) {
202
210
  const via = trail.length > 0 ? `(被 "${trail[trail.length - 1]}" 依赖)` : '';
203
- throw new error_1.AppError('REGISTRY_ENTRY_NOT_FOUND', `registry 没有条目 "${name}"${via}`, {
211
+ throw new error_1.AppError('REGISTRY_ENTRY_NOT_FOUND', `registry 没有条目 "${token}"${via}`, {
204
212
  next_actions: ['用 `miaoda registry list` 查看可用条目'],
205
213
  });
206
214
  }
207
- onStack.add(name);
208
- for (const dep of entry.dependsOn) {
209
- visit(dep, [...trail, name]);
215
+ return matches;
216
+ };
217
+ const visit = (entry, trail) => {
218
+ if (visited.has(entry.relPath))
219
+ return;
220
+ if (onStack.has(entry.relPath)) {
221
+ throw new error_1.AppError('REGISTRY_CYCLE', `registry 条目依赖成环:${[...trail, entry.relPath].join(' -> ')}`);
222
+ }
223
+ onStack.add(entry.relPath);
224
+ for (const depToken of entry.dependsOn) {
225
+ for (const depEntry of resolveToken(depToken, [...trail, entry.relPath])) {
226
+ visit(depEntry, [...trail, entry.relPath]);
227
+ }
210
228
  }
211
- onStack.delete(name);
212
- visited.add(name);
229
+ onStack.delete(entry.relPath);
230
+ visited.add(entry.relPath);
213
231
  ordered.push(entry);
214
232
  };
215
- for (const name of requested)
216
- visit(name, []);
233
+ for (const token of requested) {
234
+ for (const entry of resolveToken(token, []))
235
+ visit(entry, []);
236
+ }
217
237
  return ordered;
218
238
  }
219
239
  /**
220
- * 规划 add:解析依赖闭包 → 每个条目按其相对包根的路径落地到项目根 → 标记是否已存在。
221
- * 纯计算,不落盘(dry-run 与真实 add 共用)。
240
+ * 规划 add:解析依赖闭包 → 每个条目**拍平为 basename** 落到项目根 → 标记是否已存在。
241
+ * 纯计算,不落盘(dry-run 与真实 add 共用)。exists 在此快照(落盘前),故闭包内同名
242
+ * basename 撞车时双方 exists 都为 false、applyAdd 里后者覆盖前者。
222
243
  */
223
244
  function planAdd(opts) {
224
245
  const entries = resolveDependencyClosure(opts.entries, opts.names);
225
246
  const files = entries.map((entry) => {
226
- // relPath '/' 存,落地时按平台拆分 join,保证 Windows 也正确
227
- const destAbs = node_path_1.default.join(opts.targetDir, ...entry.relPath.split('/'));
247
+ const dest = entry.relPath.split('/').pop() ?? entry.relPath; // 拍平为 basename
248
+ const destAbs = node_path_1.default.join(opts.targetDir, dest);
228
249
  return {
229
250
  entry: entry.name,
230
- relPath: entry.relPath,
251
+ srcRelPath: entry.relPath,
252
+ dest,
231
253
  destAbs,
232
254
  srcAbs: entry.absPath,
233
255
  exists: node_fs_1.default.existsSync(destAbs),
@@ -236,28 +258,36 @@ function planAdd(opts) {
236
258
  return { entries, files };
237
259
  }
238
260
  /**
239
- * 执行 add 计划:按 skip-by-disk 把 SDK 文件保留相对结构拷到项目根。
240
- * - 目标已存在且未 overwrite → skip(保护用户改动)
261
+ * 执行 add 计划:把 SDK 文件**拍平**拷到项目根。
262
+ * - 目标已存在且未 overwrite → skip(skip-by-disk 保护用户改动)
241
263
  * - 目标已存在且 overwrite → 覆盖写
242
264
  * - 目标不存在 → 写
243
- * dryRun=true 时只分类不落盘。不写任何安装记账文件(项目文件树即状态)。
265
+ * 闭包内同名 basename 撞车(双方 exists 均为 plan 前快照 false)→ 顺序拷贝、后者覆盖前者,
266
+ * added 去重只报一次。dryRun=true 时只分类不落盘。不写任何安装记账文件(项目文件树即状态)。
244
267
  */
245
268
  function applyAdd(files, opts) {
246
269
  const added = [];
247
270
  const skipped = [];
271
+ const seen = new Set();
272
+ const pushUnique = (list, dest) => {
273
+ if (seen.has(dest))
274
+ return;
275
+ seen.add(dest);
276
+ list.push(dest);
277
+ };
248
278
  for (const f of files) {
249
279
  if (f.exists && !opts.overwrite) {
250
- skipped.push(f.relPath);
280
+ pushUnique(skipped, f.dest);
251
281
  continue;
252
282
  }
253
283
  if (!opts.dryRun) {
254
284
  if (!node_fs_1.default.existsSync(f.srcAbs)) {
255
- throw new error_1.AppError('REGISTRY_FILE_MISSING', `registry 包内缺少文件 ${f.relPath}(条目 "${f.entry}")`);
285
+ throw new error_1.AppError('REGISTRY_FILE_MISSING', `registry 包内缺少文件 ${f.srcRelPath}(条目 "${f.entry}")`);
256
286
  }
257
287
  node_fs_1.default.mkdirSync(node_path_1.default.dirname(f.destAbs), { recursive: true });
258
288
  node_fs_1.default.copyFileSync(f.srcAbs, f.destAbs);
259
289
  }
260
- added.push(f.relPath);
290
+ pushUnique(added, f.dest);
261
291
  }
262
292
  return { added, skipped };
263
293
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@lark-apaas/miaoda-cli",
3
- "version": "0.1.19-alpha.6dad9cd",
3
+ "version": "0.1.19-alpha.834fd32",
4
4
  "description": "Miaoda 平台命令行工具,面向 Agent 调用",
5
5
  "type": "commonjs",
6
6
  "bin": {