@lark-apaas/miaoda-cli 0.1.2-alpha.55feccd → 0.1.2-alpha.5fec91b

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.
Files changed (55) hide show
  1. package/dist/api/app/api.js +25 -0
  2. package/dist/api/app/index.js +15 -0
  3. package/dist/api/app/schemas.js +79 -0
  4. package/dist/api/app/types.js +58 -0
  5. package/dist/api/db/api.js +6 -250
  6. package/dist/api/db/client.js +0 -36
  7. package/dist/api/db/index.js +1 -9
  8. package/dist/api/deploy/api.js +60 -0
  9. package/dist/api/deploy/index.js +16 -0
  10. package/dist/api/deploy/schemas.js +103 -0
  11. package/dist/api/deploy/types.js +22 -0
  12. package/dist/api/file/api.js +0 -15
  13. package/dist/api/file/index.js +1 -2
  14. package/dist/api/index.js +7 -1
  15. package/dist/api/observability/api.js +52 -0
  16. package/dist/api/observability/index.js +16 -0
  17. package/dist/api/observability/schemas.js +39 -0
  18. package/dist/api/observability/types.js +27 -0
  19. package/dist/cli/commands/app/index.js +62 -0
  20. package/dist/cli/commands/db/index.js +0 -137
  21. package/dist/cli/commands/deploy/index.js +140 -0
  22. package/dist/cli/commands/file/index.js +0 -7
  23. package/dist/cli/commands/index.js +6 -0
  24. package/dist/cli/commands/observability/index.js +229 -0
  25. package/dist/cli/commands/shared.js +61 -3
  26. package/dist/cli/handlers/app/get.js +48 -0
  27. package/dist/cli/handlers/app/index.js +7 -0
  28. package/dist/cli/handlers/{db/quota.js → app/update.js} +20 -29
  29. package/dist/cli/handlers/db/data.js +1 -1
  30. package/dist/cli/handlers/db/index.js +1 -17
  31. package/dist/cli/handlers/deploy/deploy.js +83 -0
  32. package/dist/cli/handlers/{file/quota.js → deploy/error-log.js} +22 -27
  33. package/dist/cli/handlers/deploy/get.js +70 -0
  34. package/dist/cli/handlers/deploy/helpers.js +41 -0
  35. package/dist/cli/handlers/deploy/history.js +70 -0
  36. package/dist/cli/handlers/deploy/index.js +14 -0
  37. package/dist/cli/handlers/deploy/polling.js +139 -0
  38. package/dist/cli/handlers/file/index.js +1 -3
  39. package/dist/cli/handlers/observability/analytics.js +190 -0
  40. package/dist/cli/handlers/observability/helpers.js +66 -0
  41. package/dist/cli/handlers/observability/index.js +12 -0
  42. package/dist/cli/handlers/observability/log.js +94 -0
  43. package/dist/cli/handlers/observability/metric.js +208 -0
  44. package/dist/cli/handlers/observability/trace.js +102 -0
  45. package/dist/utils/devops-error.js +28 -0
  46. package/dist/utils/git.js +29 -0
  47. package/dist/utils/http.js +118 -0
  48. package/dist/utils/index.js +13 -1
  49. package/dist/utils/output.js +338 -29
  50. package/dist/utils/time.js +132 -0
  51. package/package.json +7 -5
  52. package/dist/cli/handlers/db/audit.js +0 -285
  53. package/dist/cli/handlers/db/changelog.js +0 -117
  54. package/dist/cli/handlers/db/migration.js +0 -145
  55. package/dist/cli/handlers/db/recovery.js +0 -188
@@ -0,0 +1,103 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.errorJobSchema = exports.deployGetSchema = exports.deployHistorySchema = void 0;
4
+ exports.nodeStatusText = nodeStatusText;
5
+ exports.nodeStatusFromText = nodeStatusFromText;
6
+ const output_1 = require("../../utils/output");
7
+ const types_1 = require("./types");
8
+ const NODE_STATUS_TEXT = {
9
+ [types_1.NodeStatus.UNSPECIFIED]: "unknown",
10
+ [types_1.NodeStatus.TODO]: "todo",
11
+ [types_1.NodeStatus.RUNNING]: "running",
12
+ [types_1.NodeStatus.SUCCESS]: "success",
13
+ [types_1.NodeStatus.FAILED]: "failed",
14
+ [types_1.NodeStatus.CANCELED]: "canceled",
15
+ [types_1.NodeStatus.HOLD_ON]: "hold_on",
16
+ };
17
+ function nodeStatusText(v) {
18
+ if (typeof v !== "number")
19
+ return undefined;
20
+ return NODE_STATUS_TEXT[v];
21
+ }
22
+ /** NodeStatus 文本(CLI flag 字符串)→ 枚举数值 */
23
+ function nodeStatusFromText(text) {
24
+ const lower = text.toLowerCase();
25
+ for (const [num, label] of Object.entries(NODE_STATUS_TEXT)) {
26
+ if (label === lower)
27
+ return Number(num);
28
+ }
29
+ return undefined;
30
+ }
31
+ function pipelineDurationMs(row) {
32
+ // SimpleInstance.createdAt / updatedAt:BAM IDL 未注单位,e2e 看是 ms。
33
+ const start = Number(row.createdAt);
34
+ const end = Number(row.updatedAt);
35
+ if (!Number.isFinite(start) || !Number.isFinite(end))
36
+ return undefined;
37
+ return end - start;
38
+ }
39
+ /**
40
+ * deploy history 列表的 pretty 渲染契约。
41
+ *
42
+ * BAM 返回 SimpleInstance(pipeline 实例),状态走 NodeStatus。
43
+ * 默认列:deploy-id(=ID)/ status / creator / created-at / duration(updatedAt-createdAt)。
44
+ */
45
+ exports.deployHistorySchema = {
46
+ columns: [
47
+ { key: "ID", label: "deploy-id" },
48
+ {
49
+ key: "status_text",
50
+ label: "status",
51
+ derive: (row) => nodeStatusText(row.status),
52
+ },
53
+ { key: "creator", label: "creator" },
54
+ { key: "createdAt", label: "created-at", format: output_1.fmt.ms() },
55
+ {
56
+ key: "duration",
57
+ label: "duration",
58
+ format: output_1.fmt.durationMs(),
59
+ derive: pipelineDurationMs,
60
+ },
61
+ ],
62
+ strict: true,
63
+ };
64
+ /**
65
+ * deploy get 单条详情的 key-value 渲染契约(基于 SimpleInstance)。
66
+ *
67
+ * 由于 BAM 没有 by-ID 的查询接口,handler 用 listPipelineInstances 在最近窗口
68
+ * 里筛一条 SimpleInstance 透传过来。
69
+ */
70
+ exports.deployGetSchema = {
71
+ columns: [
72
+ { key: "ID", label: "deploy-id" },
73
+ {
74
+ key: "status_text",
75
+ label: "status",
76
+ derive: (row) => nodeStatusText(row.status),
77
+ },
78
+ { key: "creator", label: "creator" },
79
+ { key: "updater", label: "updater" },
80
+ { key: "createdAt", label: "created-at", format: output_1.fmt.ms() },
81
+ { key: "updatedAt", label: "updated-at", format: output_1.fmt.ms() },
82
+ {
83
+ key: "duration",
84
+ label: "duration",
85
+ format: output_1.fmt.durationMs(),
86
+ derive: pipelineDurationMs,
87
+ },
88
+ { key: "templateID", label: "template" },
89
+ { key: "description" },
90
+ { key: "parameters" },
91
+ { key: "envVariables", label: "env-vars" },
92
+ ],
93
+ strict: true,
94
+ };
95
+ /** deploy error-log 表格渲染契约 */
96
+ exports.errorJobSchema = {
97
+ columns: [
98
+ { key: "jobID", label: "job-id" },
99
+ { key: "componentName", label: "component" },
100
+ { key: "errorMsg", label: "error", format: output_1.fmt.truncate(120) },
101
+ ],
102
+ strict: true,
103
+ };
@@ -0,0 +1,22 @@
1
+ "use strict";
2
+ // 与 BAM lark.apaas.devops v1.0.326 / 1.0.327 + lark.apaas.devops_platform v1.0.247 对齐:
3
+ // - CLICreateForceRelease (4070318) 创建发布 lark.apaas.devops
4
+ // - CliQueryPipelineInstance (4069872) 轮询发布状态 lark.apaas.devops_platform
5
+ // - CliListPipelineInstances (4073969) 分页获取发布历史 lark.apaas.devops_platform
6
+ // - CliGetToolInstanceErrorJobs (4073972) 获取发布错误日志 lark.apaas.devops_platform
7
+ //
8
+ // 发布单的唯一标识是 pipelineTaskID,对外即 deploy-id;error-log 等接口直接用它做
9
+ // URL 路径段(instanceID)。曾经的 Release/releaseID 概念已全面废弃。
10
+ Object.defineProperty(exports, "__esModule", { value: true });
11
+ exports.NodeStatus = void 0;
12
+ /** Pipeline 节点状态(devops_platform) */
13
+ var NodeStatus;
14
+ (function (NodeStatus) {
15
+ NodeStatus[NodeStatus["UNSPECIFIED"] = 0] = "UNSPECIFIED";
16
+ NodeStatus[NodeStatus["TODO"] = 1] = "TODO";
17
+ NodeStatus[NodeStatus["RUNNING"] = 2] = "RUNNING";
18
+ NodeStatus[NodeStatus["SUCCESS"] = 3] = "SUCCESS";
19
+ NodeStatus[NodeStatus["FAILED"] = 4] = "FAILED";
20
+ NodeStatus[NodeStatus["CANCELED"] = 5] = "CANCELED";
21
+ NodeStatus[NodeStatus["HOLD_ON"] = 6] = "HOLD_ON";
22
+ })(NodeStatus || (exports.NodeStatus = NodeStatus = {}));
@@ -8,7 +8,6 @@ exports.uploadFile = uploadFile;
8
8
  exports.signDownload = signDownload;
9
9
  exports.downloadFile = downloadFile;
10
10
  exports.deleteFiles = deleteFiles;
11
- exports.getStorageQuota = getStorageQuota;
12
11
  const error_1 = require("../../utils/error");
13
12
  const logger_1 = require("../../utils/logger");
14
13
  const client_1 = require("./client");
@@ -520,17 +519,3 @@ async function deleteFiles(opts) {
520
519
  }
521
520
  return { deleted, failed };
522
521
  }
523
- // ── storage quota ──
524
- /**
525
- * 后端:GET /v1/storage/app/{appId}/bucket/{bucketId}/quota
526
- * 单 bucket 用量;StorageQuotaBytes 暂未对接,CLI 拿到 0 时按 "—" 渲染。
527
- * bucketId 缺省时走默认 bucket(与 ls / stat / cp 等一致)。
528
- */
529
- async function getStorageQuota(opts) {
530
- const bucketId = opts.bucketId ?? (await (0, client_1.getDefaultBucketId)(opts.appId));
531
- const url = `/v1/storage/app/${encodeURIComponent(opts.appId)}/bucket/${encodeURIComponent(bucketId)}/quota`;
532
- const body = await (0, client_1.doGet)(url, {
533
- errorContext: `fetch storage quota for app '${opts.appId}' bucket '${bucketId}'`,
534
- });
535
- return extractEnvelope(body);
536
- }
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.toAbsolutePath = exports.looksLikePath = exports.resetBucketCache = exports.getDefaultBucketId = exports.getStorageQuota = exports.parseTimeFilterMs = exports.resolveInputs = exports.deleteFiles = exports.downloadFile = exports.signDownload = exports.uploadFile = exports.statFile = exports.listFiles = void 0;
3
+ exports.toAbsolutePath = exports.looksLikePath = exports.resetBucketCache = exports.getDefaultBucketId = exports.parseTimeFilterMs = exports.resolveInputs = exports.deleteFiles = exports.downloadFile = exports.signDownload = exports.uploadFile = exports.statFile = exports.listFiles = void 0;
4
4
  var api_1 = require("./api");
5
5
  Object.defineProperty(exports, "listFiles", { enumerable: true, get: function () { return api_1.listFiles; } });
6
6
  Object.defineProperty(exports, "statFile", { enumerable: true, get: function () { return api_1.statFile; } });
@@ -10,7 +10,6 @@ Object.defineProperty(exports, "downloadFile", { enumerable: true, get: function
10
10
  Object.defineProperty(exports, "deleteFiles", { enumerable: true, get: function () { return api_1.deleteFiles; } });
11
11
  Object.defineProperty(exports, "resolveInputs", { enumerable: true, get: function () { return api_1.resolveInputs; } });
12
12
  Object.defineProperty(exports, "parseTimeFilterMs", { enumerable: true, get: function () { return api_1.parseTimeFilterMs; } });
13
- Object.defineProperty(exports, "getStorageQuota", { enumerable: true, get: function () { return api_1.getStorageQuota; } });
14
13
  var client_1 = require("./client");
15
14
  Object.defineProperty(exports, "getDefaultBucketId", { enumerable: true, get: function () { return client_1.getDefaultBucketId; } });
16
15
  Object.defineProperty(exports, "resetBucketCache", { enumerable: true, get: function () { return client_1.resetBucketCache; } });
package/dist/api/index.js CHANGED
@@ -33,10 +33,16 @@ var __importStar = (this && this.__importStar) || (function () {
33
33
  };
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
- exports.db = exports.file = exports.plugin = void 0;
36
+ 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
+ const _observability = __importStar(require("../api/observability/index"));
41
+ const _app = __importStar(require("../api/app/index"));
42
+ const _deploy = __importStar(require("../api/deploy/index"));
40
43
  exports.plugin = { ..._plugin };
41
44
  exports.file = { ..._file };
42
45
  exports.db = { ..._db };
46
+ exports.observability = { ..._observability };
47
+ exports.app = { ..._app };
48
+ exports.deploy = { ..._deploy };
@@ -0,0 +1,52 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.searchLogs = searchLogs;
4
+ exports.searchTraces = searchTraces;
5
+ exports.getTraces = getTraces;
6
+ exports.queryMetricsData = queryMetricsData;
7
+ exports.queryAnalyticsData = queryAnalyticsData;
8
+ const http_1 = require("../../utils/http");
9
+ const DEFAULT_ERR_CODE = "INTERNAL_OB_ERROR";
10
+ // ── 日志 ──
11
+ async function searchLogs(req) {
12
+ const url = `/v1/observability/app/${encodeURIComponent(req.appID)}/logs/search`;
13
+ return (0, http_1.postInnerApi)(url, req, {
14
+ errPrefix: "Failed to search logs",
15
+ defaultErrCode: DEFAULT_ERR_CODE,
16
+ });
17
+ }
18
+ // ── 链路 ──
19
+ async function searchTraces(req) {
20
+ const url = `/v1/observability/app/${encodeURIComponent(req.appID)}/traces/search`;
21
+ return (0, http_1.postInnerApi)(url, req, {
22
+ errPrefix: "Failed to search traces",
23
+ defaultErrCode: DEFAULT_ERR_CODE,
24
+ });
25
+ }
26
+ async function getTraces(req) {
27
+ const url = `/v1/observability/app/${encodeURIComponent(req.appID)}/traces/${encodeURIComponent(req.traceID)}`;
28
+ return (0, http_1.postInnerApi)(url, req, {
29
+ errPrefix: "Failed to get trace",
30
+ defaultErrCode: DEFAULT_ERR_CODE,
31
+ });
32
+ }
33
+ // ── 监控指标 ──
34
+ async function queryMetricsData(req) {
35
+ const url = `/v1/observability/app/${encodeURIComponent(req.appID)}/metrics/data/query`;
36
+ return (0, http_1.postInnerApi)(url, req, {
37
+ errPrefix: "Failed to query metrics",
38
+ defaultErrCode: DEFAULT_ERR_CODE,
39
+ });
40
+ }
41
+ // ── 运营指标 ──
42
+ //
43
+ // 走 BAM v1.0.122 起的批量端点 CLIBatchQueryAnalyticsData:
44
+ // - 路径 /analytics/data/batch_query(旧版 /analytics/data/query 已废弃)
45
+ // - 支持 metricTypes 数组,单次拉多个指标
46
+ async function queryAnalyticsData(req) {
47
+ const url = `/v1/observability/app/${encodeURIComponent(req.appID)}/analytics/data/batch_query`;
48
+ return (0, http_1.postInnerApi)(url, req, {
49
+ errPrefix: "Failed to query analytics",
50
+ defaultErrCode: DEFAULT_ERR_CODE,
51
+ });
52
+ }
@@ -0,0 +1,16 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.StatusCode = exports.SpanKind = exports.WordOperator = exports.spanSchema = exports.logItemSchema = exports.queryAnalyticsData = exports.queryMetricsData = exports.getTraces = exports.searchTraces = exports.searchLogs = void 0;
4
+ var api_1 = require("./api");
5
+ Object.defineProperty(exports, "searchLogs", { enumerable: true, get: function () { return api_1.searchLogs; } });
6
+ Object.defineProperty(exports, "searchTraces", { enumerable: true, get: function () { return api_1.searchTraces; } });
7
+ Object.defineProperty(exports, "getTraces", { enumerable: true, get: function () { return api_1.getTraces; } });
8
+ Object.defineProperty(exports, "queryMetricsData", { enumerable: true, get: function () { return api_1.queryMetricsData; } });
9
+ Object.defineProperty(exports, "queryAnalyticsData", { enumerable: true, get: function () { return api_1.queryAnalyticsData; } });
10
+ var schemas_1 = require("./schemas");
11
+ Object.defineProperty(exports, "logItemSchema", { enumerable: true, get: function () { return schemas_1.logItemSchema; } });
12
+ Object.defineProperty(exports, "spanSchema", { enumerable: true, get: function () { return schemas_1.spanSchema; } });
13
+ var types_1 = require("./types");
14
+ Object.defineProperty(exports, "WordOperator", { enumerable: true, get: function () { return types_1.WordOperator; } });
15
+ Object.defineProperty(exports, "SpanKind", { enumerable: true, get: function () { return types_1.SpanKind; } });
16
+ Object.defineProperty(exports, "StatusCode", { enumerable: true, get: function () { return types_1.StatusCode; } });
@@ -0,0 +1,39 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.spanSchema = exports.logItemSchema = void 0;
4
+ const output_1 = require("../../utils/output");
5
+ /** LogItem 渲染契约(log search) */
6
+ exports.logItemSchema = {
7
+ columns: [
8
+ { key: "timestampNs", label: "time", format: output_1.fmt.ns("yyyy-MM-dd HH:mm:ss.SSS") },
9
+ { key: "severityText", label: "level" },
10
+ { key: "body" },
11
+ ],
12
+ strict: true,
13
+ };
14
+ /** Span 渲染契约(trace list / trace get 复用)。
15
+ * duration 是 client-derived(BAM 只给 start/end,没有原生 durationNs 字段)。 */
16
+ exports.spanSchema = {
17
+ columns: [
18
+ { key: "startTimeUnixNano", label: "start-time", format: output_1.fmt.ns("yyyy-MM-dd HH:mm:ss.SSS") },
19
+ { key: "name", label: "root-span" },
20
+ {
21
+ key: "user-id",
22
+ derive: (row) => {
23
+ return row.attributes?.user_id;
24
+ },
25
+ },
26
+ {
27
+ key: "duration",
28
+ label: "duration",
29
+ format: output_1.fmt.durationMs(),
30
+ derive: (row) => {
31
+ return row.attributes?.duration_ms;
32
+ },
33
+ },
34
+ { key: "traceID", label: "trace-id" },
35
+ ],
36
+ strict: true,
37
+ };
38
+ // metric / analytics 都改用 BAM v1.0.123 起的合并响应(points: {timestamp, values}),
39
+ // pretty 渲染走 handler 内 pivot 后的临时 schema;不再导出固定 schema。
@@ -0,0 +1,27 @@
1
+ "use strict";
2
+ // ── 通用结构 ──
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.StatusCode = exports.SpanKind = exports.WordOperator = void 0;
5
+ /** 模糊搜索词运算符(对应后端 WordOperator) */
6
+ var WordOperator;
7
+ (function (WordOperator) {
8
+ WordOperator[WordOperator["AND"] = 0] = "AND";
9
+ WordOperator[WordOperator["OR"] = 1] = "OR";
10
+ })(WordOperator || (exports.WordOperator = WordOperator = {}));
11
+ /** OpenTelemetry SpanKind(BAM 透传) */
12
+ var SpanKind;
13
+ (function (SpanKind) {
14
+ SpanKind[SpanKind["UNSPECIFIED"] = 0] = "UNSPECIFIED";
15
+ SpanKind[SpanKind["INTERNAL"] = 1] = "INTERNAL";
16
+ SpanKind[SpanKind["SERVER"] = 2] = "SERVER";
17
+ SpanKind[SpanKind["CLIENT"] = 3] = "CLIENT";
18
+ SpanKind[SpanKind["PRODUCER"] = 4] = "PRODUCER";
19
+ SpanKind[SpanKind["CONSUMER"] = 5] = "CONSUMER";
20
+ })(SpanKind || (exports.SpanKind = SpanKind = {}));
21
+ /** Span 状态码(BAM 透传) */
22
+ var StatusCode;
23
+ (function (StatusCode) {
24
+ StatusCode[StatusCode["UNSET"] = 0] = "UNSET";
25
+ StatusCode[StatusCode["OK"] = 1] = "OK";
26
+ StatusCode[StatusCode["ERROR"] = 2] = "ERROR";
27
+ })(StatusCode || (exports.StatusCode = StatusCode = {}));
@@ -0,0 +1,62 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.registerAppCommands = registerAppCommands;
4
+ const shared_1 = require("../../../cli/commands/shared");
5
+ const index_1 = require("../../../cli/handlers/app/index");
6
+ function registerAppCommands(program) {
7
+ const appCmd = program.command("app").description("应用元数据管理:查看 / 修改");
8
+ appCmd.action(() => {
9
+ appCmd.outputHelp();
10
+ });
11
+ appCmd.addHelpText("after", `
12
+ 作用范围
13
+ 仅供 Agent 在妙搭沙箱内调用。
14
+ 应用上下文:沙箱内的默认应用。
15
+ `);
16
+ registerAppGet(appCmd);
17
+ registerAppUpdate(appCmd);
18
+ }
19
+ function registerAppGet(parent) {
20
+ const cmd = parent
21
+ .command("get")
22
+ .description("查看应用元数据(status / type / mode / branch / 时间戳 等)")
23
+ .addOption((0, shared_1.appIdOption)().hideHelp())
24
+ .addHelpText("after", `
25
+ JSON 输出
26
+ {"data": {"appID": "...", "name": "...", ...}}
27
+
28
+ 示例
29
+ $ miaoda app get
30
+ $ miaoda app get --json
31
+ `);
32
+ cmd.action((0, shared_1.withHelp)(cmd, async (rawOpts) => {
33
+ (0, shared_1.rejectCliOverride)(cmd, "appId");
34
+ await (0, index_1.handleAppGet)({
35
+ appId: rawOpts.appId,
36
+ });
37
+ }));
38
+ }
39
+ function registerAppUpdate(parent) {
40
+ const cmd = parent
41
+ .command("update")
42
+ .description("修改应用名称 / 描述(至少传 --name 或 --description)")
43
+ .addOption((0, shared_1.appIdOption)().hideHelp())
44
+ .option("--name <name>", "新应用名称")
45
+ .option("--description <desc>", "新应用描述")
46
+ .addHelpText("after", `
47
+ JSON 输出
48
+ 与 'app get' 一致:{"data": {...新 meta...}, "next_cursor": null, "has_more": false}
49
+
50
+ 示例
51
+ $ miaoda app update --name my-store
52
+ $ miaoda app update --description "在线商城应用"
53
+ `);
54
+ cmd.action((0, shared_1.withHelp)(cmd, async (rawOpts) => {
55
+ (0, shared_1.rejectCliOverride)(cmd, "appId");
56
+ await (0, index_1.handleAppUpdate)({
57
+ appId: rawOpts.appId,
58
+ name: rawOpts.name,
59
+ description: rawOpts.description,
60
+ });
61
+ }));
62
+ }
@@ -1,15 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.registerDbCommands = registerDbCommands;
4
- const error_1 = require("../../../utils/error");
5
4
  const index_1 = require("../../../cli/handlers/db/index");
6
- function parsePositiveInt(raw) {
7
- const n = Number(raw);
8
- if (!Number.isInteger(n) || n < 1) {
9
- throw new error_1.AppError("ARGS_INVALID", `--limit must be a positive integer (got '${raw}')`);
10
- }
11
- return n;
12
- }
13
5
  function registerDbCommands(program) {
14
6
  const dbCmd = program
15
7
  .command("db")
@@ -214,133 +206,4 @@ Examples:
214
206
  Error: Output file 'users.csv' already exists
215
207
  hint: Use -f to specify a different path, or --force to overwrite.
216
208
  `);
217
- // ── changelog ──
218
- dbCmd
219
- .command("changelog")
220
- .summary("查看 DDL 变更历史")
221
- .description("列出 DDL 变更(CREATE / ALTER / DROP),支持按表、时间窗筛选与游标分页。")
222
- .usage("[flags]")
223
- .option("--table <name>", "只看某张表的 DDL 变更")
224
- .option("--since <time>", "起始时间(含),相对 1h/2d / 日期 YYYY-MM-DD / ISO 8601")
225
- .option("--until <time>", "结束时间(含),同 --since 格式")
226
- .option("--limit <n>", "单次返回上限(默认 20)", parsePositiveInt, 20)
227
- .option("--cursor <token>", "分页游标,从上次响应的 next_cursor 取值")
228
- .option("--all", "自动翻页直到取完所有记录")
229
- .action(async function () {
230
- await (0, index_1.handleDbChangelog)(this.optsWithGlobals());
231
- });
232
- // ── audit ──
233
- const auditCmd = dbCmd
234
- .command("audit")
235
- .summary("表级数据审计开关与查询")
236
- .description("配置表的数据审计 (enable/disable/retention),查询审计状态与日志。")
237
- .usage("<command> [flags]");
238
- auditCmd.action(() => {
239
- auditCmd.outputHelp();
240
- });
241
- auditCmd
242
- .command("status")
243
- .summary("查看审计开关状态(不传 table 返列表)")
244
- .usage("[table] [flags]")
245
- .argument("[table]", "表名;省略时返所有已配置审计的表")
246
- .action(async function (table) {
247
- await (0, index_1.handleDbAuditStatus)(table, this.optsWithGlobals());
248
- });
249
- auditCmd
250
- .command("enable")
251
- .summary("启用表审计")
252
- .usage("<table> [flags]")
253
- .argument("<table>", "目标表名")
254
- .option("--retention <ttl>", "保留时长 7d / 30d / 180d / 360d / forever", "7d")
255
- .action(async function (table) {
256
- await (0, index_1.handleDbAuditEnable)(table, this.optsWithGlobals());
257
- });
258
- auditCmd
259
- .command("disable")
260
- .summary("关闭表审计")
261
- .usage("<table> [flags]")
262
- .argument("<table>", "目标表名")
263
- .action(async function (table) {
264
- await (0, index_1.handleDbAuditDisable)(table, this.optsWithGlobals());
265
- });
266
- auditCmd
267
- .command("list")
268
- .summary("查询审计日志")
269
- .description("按表 + 时间窗查询审计日志(INSERT / UPDATE / DELETE)。")
270
- .usage("<table...> [flags]")
271
- .argument("<tables...>", "一个或多个表名(多表时空表会汇总到 stderr)")
272
- .option("--since <time>", "起始时间")
273
- .option("--until <time>", "结束时间")
274
- .option("--limit <n>", "单次返回上限(默认 50)", parsePositiveInt, 50)
275
- .option("--cursor <ts>", "分页游标,传上一页末条 event_time")
276
- .action(async function (tables) {
277
- await (0, index_1.handleDbAuditList)(tables, this.optsWithGlobals());
278
- });
279
- // ── migration ──
280
- const migrationCmd = dbCmd
281
- .command("migration")
282
- .summary("多环境管理(dev / online,仅专家模式应用)")
283
- .description("init 拆分双环境,diff 预览待发布变更,apply 把 dev 同步到 online。")
284
- .usage("<command> [flags]");
285
- migrationCmd.action(() => {
286
- migrationCmd.outputHelp();
287
- });
288
- migrationCmd
289
- .command("init")
290
- .summary("初始化多环境(不可逆)")
291
- .usage("[flags]")
292
- .option("--sync-data", "同时把现有数据复制到 dev")
293
- .option("--yes", "跳过 TTY 确认")
294
- .action(async function () {
295
- await (0, index_1.handleDbMigrationInit)(this.optsWithGlobals());
296
- });
297
- migrationCmd
298
- .command("diff")
299
- .summary("预览 dev → online 待发布变更")
300
- .usage("[flags]")
301
- .action(async function () {
302
- await (0, index_1.handleDbMigrationDiff)(this.optsWithGlobals());
303
- });
304
- migrationCmd
305
- .command("apply")
306
- .summary("应用 dev 变更到 online(多 DDL 单事务原子)")
307
- .usage("[flags]")
308
- .option("--yes", "跳过 TTY 二次确认")
309
- .action(async function () {
310
- await (0, index_1.handleDbMigrationApply)(this.optsWithGlobals());
311
- });
312
- // ── recovery(PITR)──
313
- const recoveryCmd = dbCmd
314
- .command("recovery")
315
- .summary("基于时间点恢复(PITR)")
316
- .description("把数据库整体恢复到指定时间点,diff 预览影响范围,apply 不可逆覆盖。")
317
- .usage("<command> [flags]");
318
- recoveryCmd.action(() => {
319
- recoveryCmd.outputHelp();
320
- });
321
- recoveryCmd
322
- .command("diff")
323
- .summary("预览恢复到指定时间点的影响范围")
324
- .usage("<timestamp> [flags]")
325
- .argument("<timestamp>", "目标时间,YYYY-MM-DD / YYYY-MM-DD HH:MM[:SS] / ISO 8601")
326
- .action(async function (target) {
327
- await (0, index_1.handleDbRecoveryDiff)(target, this.optsWithGlobals());
328
- });
329
- recoveryCmd
330
- .command("apply")
331
- .summary("触发恢复到指定时间点(不可逆覆盖)")
332
- .usage("<timestamp> [flags]")
333
- .argument("<timestamp>", "目标时间,YYYY-MM-DD / YYYY-MM-DD HH:MM[:SS] / ISO 8601")
334
- .option("--yes", "跳过 TTY 二次确认")
335
- .action(async function (target) {
336
- await (0, index_1.handleDbRecoveryApply)(target, this.optsWithGlobals());
337
- });
338
- // ── quota ──
339
- dbCmd
340
- .command("quota")
341
- .summary("查看数据库用量与限额")
342
- .usage("[flags]")
343
- .action(async function () {
344
- await (0, index_1.handleDbQuota)(this.optsWithGlobals());
345
- });
346
209
  }