@lark-apaas/miaoda-cli 0.1.3 → 0.1.4
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/dist/api/app/api.js +3 -3
- package/dist/api/app/schemas.js +43 -43
- package/dist/api/db/api.js +398 -55
- package/dist/api/db/client.js +155 -28
- package/dist/api/db/index.js +12 -1
- package/dist/api/db/parsers.js +20 -20
- package/dist/api/db/sql-keywords.js +87 -87
- package/dist/api/deploy/api.js +5 -5
- package/dist/api/deploy/schemas.js +32 -32
- package/dist/api/file/api.js +89 -87
- package/dist/api/file/client.js +62 -22
- package/dist/api/file/detect.js +3 -3
- package/dist/api/file/index.js +2 -1
- package/dist/api/file/parsers.js +18 -7
- package/dist/api/observability/api.js +6 -6
- package/dist/api/observability/schemas.js +14 -14
- package/dist/api/plugin/api.js +31 -31
- package/dist/cli/commands/app/index.js +12 -12
- package/dist/cli/commands/db/index.js +602 -54
- package/dist/cli/commands/deploy/index.js +28 -28
- package/dist/cli/commands/file/index.js +85 -58
- package/dist/cli/commands/observability/index.js +69 -69
- package/dist/cli/commands/plugin/index.js +27 -27
- package/dist/cli/commands/shared.js +10 -10
- package/dist/cli/handlers/app/update.js +2 -2
- package/dist/cli/handlers/db/_destructive.js +67 -0
- package/dist/cli/handlers/db/_env.js +26 -0
- package/dist/cli/handlers/db/_operator.js +35 -0
- package/dist/cli/handlers/db/audit.js +383 -0
- package/dist/cli/handlers/db/changelog.js +160 -0
- package/dist/cli/handlers/db/data.js +32 -31
- package/dist/cli/handlers/db/index.js +17 -1
- package/dist/cli/handlers/db/migration.js +234 -0
- package/dist/cli/handlers/db/quota.js +68 -0
- package/dist/cli/handlers/db/recovery.js +413 -0
- package/dist/cli/handlers/db/schema.js +33 -33
- package/dist/cli/handlers/db/sql.js +69 -69
- package/dist/cli/handlers/deploy/deploy.js +4 -4
- package/dist/cli/handlers/deploy/error-log.js +1 -1
- package/dist/cli/handlers/deploy/get.js +3 -3
- package/dist/cli/handlers/deploy/polling.js +11 -11
- package/dist/cli/handlers/file/cp.js +30 -30
- package/dist/cli/handlers/file/index.js +3 -1
- package/dist/cli/handlers/file/ls.js +5 -5
- package/dist/cli/handlers/file/quota.js +66 -0
- package/dist/cli/handlers/file/rm.js +32 -30
- package/dist/cli/handlers/file/sign.js +3 -3
- package/dist/cli/handlers/file/stat.js +10 -9
- package/dist/cli/handlers/observability/analytics.js +47 -47
- package/dist/cli/handlers/observability/helpers.js +2 -2
- package/dist/cli/handlers/observability/log.js +9 -9
- package/dist/cli/handlers/observability/metric.js +26 -26
- package/dist/cli/handlers/observability/trace.js +5 -5
- package/dist/cli/handlers/plugin/plugin-local.js +53 -53
- package/dist/cli/handlers/plugin/plugin.js +15 -15
- package/dist/cli/help.js +16 -16
- package/dist/main.js +12 -12
- package/dist/utils/args.js +1 -1
- package/dist/utils/colors.js +2 -2
- package/dist/utils/config.js +2 -2
- package/dist/utils/devops-error.js +9 -9
- package/dist/utils/error.js +2 -2
- package/dist/utils/git.js +4 -4
- package/dist/utils/http.js +19 -19
- package/dist/utils/index.js +3 -1
- package/dist/utils/output.js +67 -45
- package/dist/utils/poll.js +35 -0
- package/dist/utils/render.js +27 -27
- package/dist/utils/spinner.js +46 -0
- package/dist/utils/time.js +47 -42
- package/package.json +1 -1
|
@@ -0,0 +1,383 @@
|
|
|
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.handleDbAuditStatus = handleDbAuditStatus;
|
|
37
|
+
exports.handleDbAuditEnable = handleDbAuditEnable;
|
|
38
|
+
exports.handleDbAuditDisable = handleDbAuditDisable;
|
|
39
|
+
exports.handleDbAuditList = handleDbAuditList;
|
|
40
|
+
const api = __importStar(require("../../../api/index"));
|
|
41
|
+
const shared_1 = require("../../../cli/commands/shared");
|
|
42
|
+
const colors_1 = require("../../../utils/colors");
|
|
43
|
+
const error_1 = require("../../../utils/error");
|
|
44
|
+
const output_1 = require("../../../utils/output");
|
|
45
|
+
const render_1 = require("../../../utils/render");
|
|
46
|
+
const time_1 = require("../../../utils/time");
|
|
47
|
+
const _operator_1 = require("../../../cli/handlers/db/_operator");
|
|
48
|
+
const index_1 = require("../../../api/db/index");
|
|
49
|
+
const VALID_RETENTION = new Set(['7d', '30d', '180d', '360d', 'forever']);
|
|
50
|
+
async function handleDbAuditStatus(table, opts) {
|
|
51
|
+
const appId = (0, shared_1.resolveAppId)(opts);
|
|
52
|
+
const items = await api.db.getAuditStatus({ appId, table, dbBranch: opts.env });
|
|
53
|
+
const rows = items.map(toAuditStatus);
|
|
54
|
+
// 单表查询但后端没记录 → 占位 enabled=false
|
|
55
|
+
if (table !== undefined && table !== '' && rows.length === 0) {
|
|
56
|
+
rows.push({ table, enabled: false, enabled_at: null, retention: null });
|
|
57
|
+
}
|
|
58
|
+
// PRD JSON:单表返 object,多表返 array
|
|
59
|
+
if ((0, output_1.isJsonMode)()) {
|
|
60
|
+
if (table !== undefined && rows.length === 1) {
|
|
61
|
+
(0, output_1.emitOk)((0, output_1.snakeCaseKeys)(rows[0]));
|
|
62
|
+
}
|
|
63
|
+
else {
|
|
64
|
+
(0, output_1.emitOk)((0, output_1.snakeCaseKeys)(rows));
|
|
65
|
+
}
|
|
66
|
+
return;
|
|
67
|
+
}
|
|
68
|
+
if (rows.length === 0) {
|
|
69
|
+
(0, output_1.emit)('No audit configuration found.');
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
const tty = (0, render_1.isStdoutTty)();
|
|
73
|
+
// 单表 → key:value 形态
|
|
74
|
+
if (table !== undefined && rows.length === 1) {
|
|
75
|
+
const it = rows[0];
|
|
76
|
+
(0, output_1.emit)((0, render_1.renderKeyValue)([
|
|
77
|
+
['Table', it.table],
|
|
78
|
+
['Enabled', boolToYesNo(it.enabled)],
|
|
79
|
+
['Enabled at', it.enabled_at ? (0, render_1.formatTime)(it.enabled_at, tty) : '—'],
|
|
80
|
+
['Retention', it.retention ?? '—'],
|
|
81
|
+
], tty));
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
// 列表 → table 形态
|
|
85
|
+
const headers = ['table', 'enabled', 'enabled_at', 'retention'];
|
|
86
|
+
const out = rows.map((it) => [
|
|
87
|
+
it.table,
|
|
88
|
+
boolToYesNo(it.enabled),
|
|
89
|
+
it.enabled_at ? (0, render_1.formatTime)(it.enabled_at, tty) : '—',
|
|
90
|
+
it.retention ?? '—',
|
|
91
|
+
]);
|
|
92
|
+
(0, output_1.emit)(tty ? (0, render_1.renderAlignedTable)(headers, out) : (0, render_1.renderTsv)(headers, out));
|
|
93
|
+
}
|
|
94
|
+
function toAuditStatus(s) {
|
|
95
|
+
return {
|
|
96
|
+
table: s.table,
|
|
97
|
+
enabled: s.enabled,
|
|
98
|
+
enabled_at: s.enabledAt ?? null,
|
|
99
|
+
retention: s.retention ?? null,
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
async function handleDbAuditEnable(table, opts) {
|
|
103
|
+
if (!table) {
|
|
104
|
+
throw new error_1.AppError('ARGS_INVALID', 'table name is required', {
|
|
105
|
+
next_actions: [
|
|
106
|
+
'Usage: miaoda db audit enable <table> [--retention 7d|30d|180d|360d|forever]',
|
|
107
|
+
],
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
const retention = opts.retention ?? '7d';
|
|
111
|
+
if (!VALID_RETENTION.has(retention)) {
|
|
112
|
+
throw new error_1.AppError('INVALID_RETENTION', `Invalid retention '${retention}'`, {
|
|
113
|
+
next_actions: ['Allowed values: 7d, 30d, 180d, 360d, forever.'],
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
const appId = (0, shared_1.resolveAppId)(opts);
|
|
117
|
+
let status;
|
|
118
|
+
try {
|
|
119
|
+
status = await api.db.setAuditConfig({
|
|
120
|
+
appId,
|
|
121
|
+
table,
|
|
122
|
+
enabled: true,
|
|
123
|
+
retention,
|
|
124
|
+
dbBranch: opts.env,
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
catch (err) {
|
|
128
|
+
// PRD: 重复 enable 报错时附带 hint,引导用户去 status 看 retention 或换值更新
|
|
129
|
+
if (err instanceof error_1.AppError && err.code === 'DB_API_k_dl_1300030') {
|
|
130
|
+
throw new error_1.AppError(err.code, err.message, {
|
|
131
|
+
next_actions: [
|
|
132
|
+
`Use \`miaoda db audit status ${table}\` to check current retention, or run this command with a different --retention to update.`,
|
|
133
|
+
],
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
throw err;
|
|
137
|
+
}
|
|
138
|
+
// PRD JSON:{"data": {"table": "...", "enabled": true, "retention": "..."}}
|
|
139
|
+
if ((0, output_1.isJsonMode)()) {
|
|
140
|
+
(0, output_1.emitOk)((0, output_1.snakeCaseKeys)({
|
|
141
|
+
table: status.table,
|
|
142
|
+
enabled: status.enabled,
|
|
143
|
+
retention: status.retention ?? retention,
|
|
144
|
+
}));
|
|
145
|
+
return;
|
|
146
|
+
}
|
|
147
|
+
const tty = (0, render_1.isStdoutTty)();
|
|
148
|
+
// c.highlight 内部按 TTY/NO_COLOR/FORCE_COLOR 自动决定是否染色,无需再判 tty
|
|
149
|
+
const tableLabel = colors_1.c.highlight(`'${status.table}'`);
|
|
150
|
+
const body = `Audit enabled for table ${tableLabel} (retention: ${status.retention ?? retention})`;
|
|
151
|
+
(0, output_1.emit)(tty ? colors_1.c.success(`✓ ${body}`) : `OK ${body}`);
|
|
152
|
+
}
|
|
153
|
+
async function handleDbAuditDisable(table, opts) {
|
|
154
|
+
if (!table) {
|
|
155
|
+
throw new error_1.AppError('ARGS_INVALID', 'table name is required', {
|
|
156
|
+
next_actions: ['Usage: miaoda db audit disable <table>'],
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
const appId = (0, shared_1.resolveAppId)(opts);
|
|
160
|
+
let status;
|
|
161
|
+
try {
|
|
162
|
+
status = await api.db.setAuditConfig({
|
|
163
|
+
appId,
|
|
164
|
+
table,
|
|
165
|
+
enabled: false,
|
|
166
|
+
dbBranch: opts.env,
|
|
167
|
+
});
|
|
168
|
+
}
|
|
169
|
+
catch (err) {
|
|
170
|
+
// PRD: 重复 disable / 未启用就 disable 报错时附带 hint,引导用户去看哪些表已开启
|
|
171
|
+
if (err instanceof error_1.AppError && err.code === 'DB_API_k_dl_1300031') {
|
|
172
|
+
throw new error_1.AppError(err.code, err.message, {
|
|
173
|
+
next_actions: ['Use `miaoda db audit status` to see which tables have audit enabled.'],
|
|
174
|
+
});
|
|
175
|
+
}
|
|
176
|
+
throw err;
|
|
177
|
+
}
|
|
178
|
+
if ((0, output_1.isJsonMode)()) {
|
|
179
|
+
(0, output_1.emitOk)((0, output_1.snakeCaseKeys)({
|
|
180
|
+
table: status.table,
|
|
181
|
+
enabled: status.enabled,
|
|
182
|
+
}));
|
|
183
|
+
return;
|
|
184
|
+
}
|
|
185
|
+
const tty = (0, render_1.isStdoutTty)();
|
|
186
|
+
const body = `Audit disabled for table ${colors_1.c.highlight(`'${status.table}'`)}`;
|
|
187
|
+
(0, output_1.emit)(tty ? colors_1.c.success(`✓ ${body}`) : `OK ${body}`);
|
|
188
|
+
}
|
|
189
|
+
async function handleDbAuditList(tables, opts) {
|
|
190
|
+
if (tables.length === 0) {
|
|
191
|
+
throw new error_1.AppError('ARGS_INVALID', 'at least one table is required', {
|
|
192
|
+
next_actions: [
|
|
193
|
+
'Usage: miaoda db audit list <table> [<table>...] [--since ...] [--until ...]',
|
|
194
|
+
],
|
|
195
|
+
});
|
|
196
|
+
}
|
|
197
|
+
const appId = (0, shared_1.resolveAppId)(opts);
|
|
198
|
+
const limit = opts.limit ?? 20;
|
|
199
|
+
// 时间字段归一化为 ISO 8601 UTC(服务端按 event_time 字段比较)。cursor 由
|
|
200
|
+
// 后端管理(本质也是 ISO 时间),CLI 不再混用 cursor/since。
|
|
201
|
+
const since = normalizeTime(opts.since, '--since');
|
|
202
|
+
const until = normalizeTime(opts.until, '--until');
|
|
203
|
+
// 多表场景:dataloom 对"任一表不存在"是 fail-fast(typo 应立即可见,符合 PRD 单表语义),
|
|
204
|
+
// 但 PRD 多表语义是"其他表继续返回,底部汇总跳过情况"。CLI 在这里前置校验:用 getSchema
|
|
205
|
+
// 的 tableNames 过滤模式(dataloom 端内存过滤,不存在的表静默忽略,不报错)拿到存在的
|
|
206
|
+
// 表集合,本地 diff 出缺失表,再用过滤后的表去查 listAuditLog。单表场景跳过预校验。
|
|
207
|
+
const isMulti = tables.length > 1;
|
|
208
|
+
let queryTables = tables;
|
|
209
|
+
const localMissing = [];
|
|
210
|
+
if (isMulti) {
|
|
211
|
+
const schemaResp = await api.db.getSchema({
|
|
212
|
+
appId,
|
|
213
|
+
tableNames: tables.join(','),
|
|
214
|
+
dbBranch: opts.env,
|
|
215
|
+
});
|
|
216
|
+
const existing = new Set((0, index_1.flattenSchemaList)(schemaResp.schema).map((t) => t.name));
|
|
217
|
+
for (const t of tables) {
|
|
218
|
+
if (!existing.has(t))
|
|
219
|
+
localMissing.push(t);
|
|
220
|
+
}
|
|
221
|
+
queryTables = tables.filter((t) => existing.has(t));
|
|
222
|
+
if (queryTables.length === 0) {
|
|
223
|
+
throw new error_1.AppError('TABLE_NOT_FOUND', `None of the requested tables exist: ${localMissing.join(', ')}`, { next_actions: ['Run `miaoda db schema list` to see all tables.'] });
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
let result;
|
|
227
|
+
try {
|
|
228
|
+
result = await api.db.listAuditLog({
|
|
229
|
+
appId,
|
|
230
|
+
tables: queryTables,
|
|
231
|
+
since,
|
|
232
|
+
until,
|
|
233
|
+
limit,
|
|
234
|
+
cursor: opts.cursor,
|
|
235
|
+
dbBranch: opts.env,
|
|
236
|
+
});
|
|
237
|
+
}
|
|
238
|
+
catch (err) {
|
|
239
|
+
throw decorateAuditListError(err, tables);
|
|
240
|
+
}
|
|
241
|
+
const visible = result.items;
|
|
242
|
+
// 服务端 skipped(audit 未启用,bare 名)+ CLI 本地探测 missing 合并;
|
|
243
|
+
// localMissing 自带 `(table not found)` 后缀,formatSkippedHint 透传渲染
|
|
244
|
+
const skipped = [...result.skipped, ...localMissing.map((t) => `${t} (table not found)`)];
|
|
245
|
+
// PRD JSON:data 是数组,元素是事件对象(snake_case),分页信封 next_cursor / has_more
|
|
246
|
+
// operator 后端用 JSON 字符串内嵌 {id, name},--json 输出还原成对象供下游消费
|
|
247
|
+
if ((0, output_1.isJsonMode)()) {
|
|
248
|
+
const items = visible.map((it) => ({
|
|
249
|
+
event_id: it.eventId,
|
|
250
|
+
event_time: it.eventTime,
|
|
251
|
+
target_table: it.targetTable,
|
|
252
|
+
type: it.type,
|
|
253
|
+
operator: (0, _operator_1.parseOperator)(it.operator),
|
|
254
|
+
summary: it.summary,
|
|
255
|
+
// before/after 服务端是 JSON 字符串,反序列化回结构化对象供下游消费
|
|
256
|
+
...(it.before !== undefined ? { before: safeParseJson(it.before) } : {}),
|
|
257
|
+
...(it.after !== undefined ? { after: safeParseJson(it.after) } : {}),
|
|
258
|
+
}));
|
|
259
|
+
(0, output_1.emitPaged)(items, result.nextCursor, result.hasMore);
|
|
260
|
+
if (skipped.length > 0) {
|
|
261
|
+
process.stderr.write(formatSkippedHint(skipped, tables.length) + '\n');
|
|
262
|
+
}
|
|
263
|
+
if (visible.length === 0)
|
|
264
|
+
process.exitCode = 1;
|
|
265
|
+
return;
|
|
266
|
+
}
|
|
267
|
+
if (visible.length === 0) {
|
|
268
|
+
(0, output_1.emit)('No audit events found.');
|
|
269
|
+
if (skipped.length > 0) {
|
|
270
|
+
process.stderr.write(formatSkippedHint(skipped, tables.length) + '\n');
|
|
271
|
+
}
|
|
272
|
+
process.exitCode = 1;
|
|
273
|
+
return;
|
|
274
|
+
}
|
|
275
|
+
const tty = (0, render_1.isStdoutTty)();
|
|
276
|
+
// PRD:多表渲染时第一列是 target_table,单表时去掉这一列
|
|
277
|
+
const isMultiTable = tables.length > 1;
|
|
278
|
+
const headers = isMultiTable
|
|
279
|
+
? ['target_table', 'event_time', 'type', 'event_id', 'operator', 'summary']
|
|
280
|
+
: ['event_time', 'type', 'event_id', 'operator', 'summary'];
|
|
281
|
+
const rows = visible.map((it) => {
|
|
282
|
+
const eventTime = (0, render_1.formatTime)(it.eventTime, tty);
|
|
283
|
+
// event_id 完整透传——PRD 截图里的 "..." 只是文档省略写法,不是 CLI 行为
|
|
284
|
+
// operator pretty 只展示 name;--json 上面已经走 parseOperator 输出 {id, name}
|
|
285
|
+
const cells = [
|
|
286
|
+
eventTime,
|
|
287
|
+
it.type,
|
|
288
|
+
it.eventId,
|
|
289
|
+
(0, _operator_1.parseOperator)(it.operator).name || '—',
|
|
290
|
+
it.summary,
|
|
291
|
+
];
|
|
292
|
+
return isMultiTable ? [it.targetTable, ...cells] : cells;
|
|
293
|
+
});
|
|
294
|
+
(0, output_1.emit)(tty ? (0, render_1.renderAlignedTable)(headers, rows) : (0, render_1.renderTsv)(headers, rows));
|
|
295
|
+
if (skipped.length > 0) {
|
|
296
|
+
process.stderr.write(formatSkippedHint(skipped, tables.length) + '\n');
|
|
297
|
+
}
|
|
298
|
+
if (result.hasMore && result.nextCursor) {
|
|
299
|
+
process.stderr.write(`(more results; use --cursor '${result.nextCursor}')\n`);
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
// ── helpers ──
|
|
303
|
+
/**
|
|
304
|
+
* 把用户传入的 since/until 归一成 ISO 8601 UTC 字符串;空串返 undefined。
|
|
305
|
+
* 支持格式见 utils/time.ts::TIMESTAMP_HELP(相对时间 / 日期 / 本地 datetime / ISO 8601)。
|
|
306
|
+
*/
|
|
307
|
+
function normalizeTime(input, flagName) {
|
|
308
|
+
if (input === undefined || input === '')
|
|
309
|
+
return undefined;
|
|
310
|
+
try {
|
|
311
|
+
return new Date((0, time_1.parseTimeToMs)(input)).toISOString();
|
|
312
|
+
}
|
|
313
|
+
catch (err) {
|
|
314
|
+
if (err instanceof error_1.AppError) {
|
|
315
|
+
throw new error_1.AppError(err.code, `${flagName}: ${err.message}`, {
|
|
316
|
+
next_actions: err.next_actions,
|
|
317
|
+
});
|
|
318
|
+
}
|
|
319
|
+
throw err;
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
// PRD 41-48 行格式:`— Skipped 1 of 3 tables: orders (audit not enabled)`
|
|
323
|
+
// 服务端 skipped 数组是 audit 未启用的表(bare 名);上面 peel 循环拼出来的
|
|
324
|
+
// "<name> (table not found)" 已经自带 reason,t.includes('(') 走原样透传,bare 名
|
|
325
|
+
// 默认拼 "(audit not enabled)"。也兼容未来后端直接给带 reason 字符串的可能。
|
|
326
|
+
function formatSkippedHint(skipped, totalRequested) {
|
|
327
|
+
const items = skipped.map((t) => (t.includes('(') ? t : `${t} (audit not enabled)`)).join(', ');
|
|
328
|
+
return `— Skipped ${String(skipped.length)} of ${String(totalRequested)} tables: ${items}`;
|
|
329
|
+
}
|
|
330
|
+
// audit list 后端错误码加 hint:
|
|
331
|
+
// - k_dl_000005 表不存在 → 文案与 hint 对齐 `db schema get` 的统一格式
|
|
332
|
+
// - k_dl_1300040 单表 audit 未启用 → 指引 enable
|
|
333
|
+
// - k_dl_1300041 多表全部未启用 → 指引 status
|
|
334
|
+
function decorateAuditListError(err, tables) {
|
|
335
|
+
if (!(err instanceof error_1.AppError))
|
|
336
|
+
return err;
|
|
337
|
+
if (err.code === 'DB_API_k_dl_000005') {
|
|
338
|
+
// 不复用 dataloom 透传的 message(旧版/新版文案不一致:`table [x] not exist` /
|
|
339
|
+
// `table x not found`),统一改写成跟 schema get 一样的 `Table '<name>' does not exist`,
|
|
340
|
+
// 单表传 tables[0],多表场景兜底用解析到的 raw message 里第一个表名。
|
|
341
|
+
const t = tables.length > 0 ? tables[0] : (extractMissingTable(err.message) ?? '<table>');
|
|
342
|
+
return new error_1.AppError('TABLE_NOT_FOUND', `Table '${t}' does not exist`, {
|
|
343
|
+
next_actions: [`Did you mean another table? Run "miaoda db schema list" to see all tables.`],
|
|
344
|
+
});
|
|
345
|
+
}
|
|
346
|
+
if (err.code === 'DB_API_k_dl_1300040') {
|
|
347
|
+
const t = tables[0] ?? '<table>';
|
|
348
|
+
return new error_1.AppError(err.code, err.message, {
|
|
349
|
+
next_actions: [`Enable with: miaoda db audit enable ${t}`],
|
|
350
|
+
});
|
|
351
|
+
}
|
|
352
|
+
if (err.code === 'DB_API_k_dl_1300041') {
|
|
353
|
+
return new error_1.AppError(err.code, err.message, {
|
|
354
|
+
next_actions: ['Check audit status with: miaoda db audit status'],
|
|
355
|
+
});
|
|
356
|
+
}
|
|
357
|
+
return err;
|
|
358
|
+
}
|
|
359
|
+
// extractMissingTable 从 dataloom 原始文案里抠表名兜底。dataloom 端有两种格式:
|
|
360
|
+
// - 旧版:`Table [foo] doesn't exist` / `table [foo] not exist`
|
|
361
|
+
// - 新版(本次 commits):`table foo not found`
|
|
362
|
+
// 都失配时返 null,调用方走 `<table>` 占位。
|
|
363
|
+
function extractMissingTable(msg) {
|
|
364
|
+
const bracket = /\[([^\]]+)\]/.exec(msg);
|
|
365
|
+
if (bracket)
|
|
366
|
+
return bracket[1];
|
|
367
|
+
const m = /table\s+([\w.]+)\s+not (?:exist|found)/i.exec(msg);
|
|
368
|
+
return m ? m[1] : null;
|
|
369
|
+
}
|
|
370
|
+
// 服务端 before/after 是 JSON 字符串透传,CLI JSON 输出再反序列化回对象,
|
|
371
|
+
// 给下游 jq 等工具消费。失败时透传原始字符串避免数据丢失。
|
|
372
|
+
function safeParseJson(s) {
|
|
373
|
+
try {
|
|
374
|
+
return JSON.parse(s);
|
|
375
|
+
}
|
|
376
|
+
catch {
|
|
377
|
+
return s;
|
|
378
|
+
}
|
|
379
|
+
}
|
|
380
|
+
// PRD 输出 enabled 列用 yes/no 而非 true/false
|
|
381
|
+
function boolToYesNo(b) {
|
|
382
|
+
return b ? 'yes' : 'no';
|
|
383
|
+
}
|
|
@@ -0,0 +1,160 @@
|
|
|
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.handleDbChangelog = handleDbChangelog;
|
|
37
|
+
const api = __importStar(require("../../../api/index"));
|
|
38
|
+
const shared_1 = require("../../../cli/commands/shared");
|
|
39
|
+
const error_1 = require("../../../utils/error");
|
|
40
|
+
const output_1 = require("../../../utils/output");
|
|
41
|
+
const render_1 = require("../../../utils/render");
|
|
42
|
+
const time_1 = require("../../../utils/time");
|
|
43
|
+
const _operator_1 = require("../../../cli/handlers/db/_operator");
|
|
44
|
+
function toRow(it) {
|
|
45
|
+
return {
|
|
46
|
+
change_id: it.changeId,
|
|
47
|
+
changed_at: it.changedAt,
|
|
48
|
+
operator: (0, _operator_1.parseOperator)(it.operator),
|
|
49
|
+
target_table: it.targetTable,
|
|
50
|
+
change_type: it.changeType,
|
|
51
|
+
summary: it.summary,
|
|
52
|
+
statement: it.statement,
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* 把用户传入的 since/until 归一成 ISO 8601 UTC 字符串;空串返 undefined。
|
|
57
|
+
* 支持格式见 utils/time.ts::TIMESTAMP_HELP(相对时间 / 日期 / 本地 datetime / ISO 8601)。
|
|
58
|
+
*/
|
|
59
|
+
function normalizeTime(input, flagName) {
|
|
60
|
+
if (input === undefined || input === '')
|
|
61
|
+
return undefined;
|
|
62
|
+
try {
|
|
63
|
+
return new Date((0, time_1.parseTimeToMs)(input)).toISOString();
|
|
64
|
+
}
|
|
65
|
+
catch (err) {
|
|
66
|
+
if (err instanceof error_1.AppError) {
|
|
67
|
+
throw new error_1.AppError(err.code, `${flagName}: ${err.message}`, {
|
|
68
|
+
next_actions: err.next_actions,
|
|
69
|
+
});
|
|
70
|
+
}
|
|
71
|
+
throw err;
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
async function handleDbChangelog(opts) {
|
|
75
|
+
const appId = (0, shared_1.resolveAppId)(opts);
|
|
76
|
+
const since = normalizeTime(opts.since, '--since');
|
|
77
|
+
const until = normalizeTime(opts.until, '--until');
|
|
78
|
+
// --change-id 是精确单条查询,CLI 透传给后端;后端若暂未支持过滤,CLI 端
|
|
79
|
+
// 兜底翻页找命中项,保证 PRD 「最多返回一条」的语义稳定。
|
|
80
|
+
const trimmed = opts.changeId?.trim();
|
|
81
|
+
const changeId = trimmed !== undefined && trimmed !== '' ? trimmed : undefined;
|
|
82
|
+
const allItems = [];
|
|
83
|
+
let cursor = opts.cursor;
|
|
84
|
+
let lastCursor = null;
|
|
85
|
+
let lastHasMore = false;
|
|
86
|
+
// 指定 --change-id 时强制开 --all,确保即使后端不识别 changeId 过滤参数,
|
|
87
|
+
// CLI 也能翻完所有页找到目标记录(changelog 默认按时间倒序,一般在前几页)。
|
|
88
|
+
const allMode = opts.all === true || changeId !== undefined;
|
|
89
|
+
// --all:循环到 hasMore=false;否则只拉一页
|
|
90
|
+
// 没传 --all 时翻页由调用方自己用 --cursor 串
|
|
91
|
+
for (;;) {
|
|
92
|
+
const page = await api.db.listDDLChangelog({
|
|
93
|
+
appId,
|
|
94
|
+
table: opts.table,
|
|
95
|
+
since,
|
|
96
|
+
until,
|
|
97
|
+
changeId,
|
|
98
|
+
limit: opts.limit,
|
|
99
|
+
cursor,
|
|
100
|
+
dbBranch: opts.env,
|
|
101
|
+
});
|
|
102
|
+
allItems.push(...page.items);
|
|
103
|
+
lastCursor = page.nextCursor;
|
|
104
|
+
lastHasMore = page.hasMore;
|
|
105
|
+
if (!allMode || !page.hasMore || !page.nextCursor)
|
|
106
|
+
break;
|
|
107
|
+
cursor = page.nextCursor;
|
|
108
|
+
// 后端命中过滤返单条时提前退出,免得继续无意义翻页
|
|
109
|
+
if (changeId !== undefined && allItems.some((it) => it.changeId === changeId))
|
|
110
|
+
break;
|
|
111
|
+
}
|
|
112
|
+
// CLI 端兜底过滤:把 changeId 匹配的项筛出来。后端如果已支持,列表本身就只有
|
|
113
|
+
// 0 或 1 条,filter 是恒等运算;后端未支持时通过翻页 + 此过滤拿到精确单条。
|
|
114
|
+
if (changeId !== undefined) {
|
|
115
|
+
const matched = allItems.filter((it) => it.changeId === changeId);
|
|
116
|
+
allItems.length = 0;
|
|
117
|
+
allItems.push(...matched);
|
|
118
|
+
// 精确查询语义:永远当作"已取完",不再透出分页 cursor / has_more
|
|
119
|
+
lastCursor = null;
|
|
120
|
+
lastHasMore = false;
|
|
121
|
+
}
|
|
122
|
+
if ((0, output_1.isJsonMode)()) {
|
|
123
|
+
const rows = allItems.map(toRow);
|
|
124
|
+
// --all 或 --change-id 时已经把所有页合一起返,has_more=false / next_cursor=null。
|
|
125
|
+
// PRD: --change-id 即使没命中(rows 长度 0)也保持 data 为数组。
|
|
126
|
+
if (allMode) {
|
|
127
|
+
(0, output_1.emitPaged)((0, output_1.snakeCaseKeys)(rows), null, false);
|
|
128
|
+
}
|
|
129
|
+
else {
|
|
130
|
+
(0, output_1.emitPaged)((0, output_1.snakeCaseKeys)(rows), lastCursor, lastHasMore);
|
|
131
|
+
}
|
|
132
|
+
return;
|
|
133
|
+
}
|
|
134
|
+
if (allItems.length === 0) {
|
|
135
|
+
// --change-id 没命中时单独报错,便于 agent 区分"无变更"和"ID 不存在"
|
|
136
|
+
if (changeId !== undefined) {
|
|
137
|
+
(0, output_1.emit)(`No DDL change with id=${changeId} found.`);
|
|
138
|
+
}
|
|
139
|
+
else {
|
|
140
|
+
(0, output_1.emit)('No DDL changes found.');
|
|
141
|
+
}
|
|
142
|
+
return;
|
|
143
|
+
}
|
|
144
|
+
const tty = (0, render_1.isStdoutTty)();
|
|
145
|
+
// PRD: change_id / changed_at / operator / target_table / change_type / summary
|
|
146
|
+
const headers = ['change_id', 'changed_at', 'operator', 'target_table', 'change_type', 'summary'];
|
|
147
|
+
// pretty 渲染只展示 operator.name(兼容 PRD 原 string 形态);--json 走 toRow 输出完整对象
|
|
148
|
+
const rows = allItems.map((it) => [
|
|
149
|
+
it.changeId,
|
|
150
|
+
(0, render_1.formatTime)(it.changedAt, tty),
|
|
151
|
+
(0, _operator_1.parseOperator)(it.operator).name || '—',
|
|
152
|
+
it.targetTable || '—',
|
|
153
|
+
it.changeType,
|
|
154
|
+
it.summary || '—',
|
|
155
|
+
]);
|
|
156
|
+
(0, output_1.emit)(tty ? (0, render_1.renderAlignedTable)(headers, rows) : (0, render_1.renderTsv)(headers, rows));
|
|
157
|
+
if (!allMode && lastHasMore && lastCursor) {
|
|
158
|
+
process.stderr.write(`(more results; use --cursor ${lastCursor} or --all)\n`);
|
|
159
|
+
}
|
|
160
|
+
}
|