rl-rockcli 0.0.1 → 0.0.2
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/commands/log/core/constants.js +237 -0
- package/commands/log/core/display.js +370 -0
- package/commands/log/core/search.js +330 -0
- package/commands/log/core/tail.js +216 -0
- package/commands/log/core/utils.js +424 -0
- package/commands/log.js +298 -0
- package/commands/sandbox/core/log-bridge.js +119 -0
- package/commands/sandbox/core/replay/analyzer.js +311 -0
- package/commands/sandbox/core/replay/batch-orchestrator.js +536 -0
- package/commands/sandbox/core/replay/batch-task.js +369 -0
- package/commands/sandbox/core/replay/concurrent-display.js +70 -0
- package/commands/sandbox/core/replay/concurrent-orchestrator.js +170 -0
- package/commands/sandbox/core/replay/data-source.js +86 -0
- package/commands/sandbox/core/replay/display.js +231 -0
- package/commands/sandbox/core/replay/executor.js +634 -0
- package/commands/sandbox/core/replay/history-fetcher.js +124 -0
- package/commands/sandbox/core/replay/index.js +338 -0
- package/commands/sandbox/core/replay/loghouse-data-source.js +177 -0
- package/commands/sandbox/core/replay/pid-mapping.js +26 -0
- package/commands/sandbox/core/replay/request.js +109 -0
- package/commands/sandbox/core/replay/worker.js +166 -0
- package/commands/sandbox/core/session.js +346 -0
- package/commands/sandbox/log-bridge.js +2 -0
- package/commands/sandbox/ray.js +2 -0
- package/commands/sandbox/replay/analyzer.js +311 -0
- package/commands/sandbox/replay/batch-orchestrator.js +536 -0
- package/commands/sandbox/replay/batch-task.js +369 -0
- package/commands/sandbox/replay/concurrent-display.js +70 -0
- package/commands/sandbox/replay/concurrent-orchestrator.js +170 -0
- package/commands/sandbox/replay/display.js +231 -0
- package/commands/sandbox/replay/executor.js +634 -0
- package/commands/sandbox/replay/history-fetcher.js +118 -0
- package/commands/sandbox/replay/index.js +338 -0
- package/commands/sandbox/replay/pid-mapping.js +26 -0
- package/commands/sandbox/replay/request.js +109 -0
- package/commands/sandbox/replay/worker.js +166 -0
- package/commands/sandbox/replay.js +2 -0
- package/commands/sandbox/session.js +2 -0
- package/commands/sandbox-original.js +1393 -0
- package/commands/sandbox.js +499 -0
- package/help/help.json +1071 -0
- package/help/middleware.js +71 -0
- package/help/renderer.js +800 -0
- package/index.js +38 -0
- package/lib/plugin-context.js +40 -0
- package/package.json +29 -43
- package/sdks/sandbox/core/client.js +845 -0
- package/sdks/sandbox/core/config.js +70 -0
- package/sdks/sandbox/core/types.js +74 -0
- package/sdks/sandbox/httpLogger.js +251 -0
- package/sdks/sandbox/index.js +9 -0
- package/utils/asciiArt.js +138 -0
- package/utils/bun-compat.js +59 -0
- package/utils/ciPipelines.js +138 -0
- package/utils/cli.js +17 -0
- package/utils/command-router.js +79 -0
- package/utils/configManager.js +503 -0
- package/utils/dependency-resolver.js +135 -0
- package/utils/eagleeye_traceid.js +151 -0
- package/utils/envDetector.js +78 -0
- package/utils/execution_logger.js +415 -0
- package/utils/featureManager.js +68 -0
- package/utils/firstTimeTip.js +44 -0
- package/utils/hook-manager.js +125 -0
- package/utils/http-logger.js +264 -0
- package/utils/i18n.js +139 -0
- package/utils/image-progress.js +159 -0
- package/utils/logger.js +154 -0
- package/utils/plugin-loader.js +124 -0
- package/utils/plugin-manager.js +348 -0
- package/utils/ray_cli_wrapper.js +746 -0
- package/utils/sandbox-client.js +419 -0
- package/utils/terminal.js +32 -0
- package/utils/tips.js +106 -0
- package/LICENSE +0 -21
- package/bin/rockcli.js +0 -40
- package/src/core/commands/attach/index.js +0 -242
- package/src/core/commands/log/constants.js +0 -20
- package/src/core/commands/log/index.js +0 -94
- package/src/core/commands/log/search.js +0 -106
- package/src/core/commands/sandbox/index.js +0 -428
- package/src/core/config/index.js +0 -77
- package/src/core/display/constants.js +0 -59
- package/src/core/display/format.js +0 -178
- package/src/core/display/highlight.js +0 -34
- package/src/core/index.js +0 -55
- package/src/core/providers/index.js +0 -9
- package/src/core/providers/log-provider.js +0 -79
- package/src/core/sdks/sandbox/client.js +0 -472
- package/src/core/sdks/sandbox/config.js +0 -57
- package/src/core/sdks/sandbox/index.js +0 -13
- package/src/core/sdks/sandbox/types.js +0 -5
- package/src/core/utils/index.js +0 -9
- package/src/core/utils/logger.js +0 -106
- package/src/core/utils/time.js +0 -52
- package/src/plugins/oss-file-log/file-client.js +0 -186
- package/src/plugins/oss-file-log/index.js +0 -18
|
@@ -0,0 +1,369 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 批量回放任务状态常量
|
|
3
|
+
*/
|
|
4
|
+
const TaskStatus = {
|
|
5
|
+
PENDING: 'pending', // 待处理
|
|
6
|
+
FETCHING: 'fetching', // 正在获取历史
|
|
7
|
+
READY: 'ready', // 历史已就绪,待执行
|
|
8
|
+
RUNNING: 'running', // 正在执行回放
|
|
9
|
+
COMPLETED: 'completed', // 执行完成
|
|
10
|
+
FAILED: 'failed', // 执行失败
|
|
11
|
+
SKIPPED: 'skipped' // 跳过(历史获取失败等)
|
|
12
|
+
};
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* 错误发生阶段
|
|
16
|
+
*/
|
|
17
|
+
const ErrorPhase = {
|
|
18
|
+
PARSE: 'parse', // 解析输入
|
|
19
|
+
HISTORY_FETCH: 'history_fetch', // 获取历史
|
|
20
|
+
REPLAY: 'replay' // 执行回放
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* 单个沙箱任务
|
|
25
|
+
* 管理单个沙箱的回放状态和结果
|
|
26
|
+
*/
|
|
27
|
+
class SandboxTask {
|
|
28
|
+
/**
|
|
29
|
+
* @param {number} index - 任务索引(0-based)
|
|
30
|
+
* @param {Object} sandboxConfig - 沙箱配置
|
|
31
|
+
* @param {Object} defaults - 默认配置(从输入文件的 defaults 字段)
|
|
32
|
+
*/
|
|
33
|
+
constructor(index, sandboxConfig, defaults = {}) {
|
|
34
|
+
this.index = index;
|
|
35
|
+
this.sandboxId = sandboxConfig.sandboxId;
|
|
36
|
+
this.name = sandboxConfig.name || sandboxConfig.sandboxId.substring(0, 8);
|
|
37
|
+
|
|
38
|
+
// 合并配置(沙箱配置覆盖默认配置)
|
|
39
|
+
this.cluster = sandboxConfig.cluster || defaults.cluster || null;
|
|
40
|
+
this.userId = sandboxConfig.userId || defaults.userId || null;
|
|
41
|
+
this.experimentId = sandboxConfig.experimentId || defaults.experimentId || null;
|
|
42
|
+
this.mode = sandboxConfig.mode || defaults.mode || 'smart';
|
|
43
|
+
this.timing = sandboxConfig.timing || defaults.timing || 'none';
|
|
44
|
+
|
|
45
|
+
// 元数据
|
|
46
|
+
this.tags = sandboxConfig.tags || {};
|
|
47
|
+
this.uris = sandboxConfig.uris || [];
|
|
48
|
+
|
|
49
|
+
// 合并过滤器(默认 + 沙箱特定)
|
|
50
|
+
this.filters = [
|
|
51
|
+
...(defaults.filters || []),
|
|
52
|
+
...(sandboxConfig.filters || [])
|
|
53
|
+
];
|
|
54
|
+
|
|
55
|
+
// 状态
|
|
56
|
+
this.status = TaskStatus.PENDING;
|
|
57
|
+
|
|
58
|
+
// 历史数据
|
|
59
|
+
this.requests = sandboxConfig.requests || null; // 预加载的请求列表
|
|
60
|
+
this.historyData = null; // 完整的历史数据(包含 startParams, targetCluster 等)
|
|
61
|
+
|
|
62
|
+
// 回放结果
|
|
63
|
+
this.newSandboxId = null; // 新创建的沙箱 ID
|
|
64
|
+
this.replayResult = null; // ReplayExecutor 返回的结果
|
|
65
|
+
this.pidMappings = []; // PID 映射表
|
|
66
|
+
|
|
67
|
+
// 错误信息
|
|
68
|
+
this.error = null;
|
|
69
|
+
this.errorPhase = null;
|
|
70
|
+
|
|
71
|
+
// 时间记录
|
|
72
|
+
this.historyFetchStartTime = null;
|
|
73
|
+
this.historyFetchEndTime = null;
|
|
74
|
+
this.replayStartTime = null;
|
|
75
|
+
this.replayEndTime = null;
|
|
76
|
+
|
|
77
|
+
// 日志文件
|
|
78
|
+
this.logFile = null;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* 是否有预加载的请求列表
|
|
83
|
+
* @returns {boolean}
|
|
84
|
+
*/
|
|
85
|
+
hasRequests() {
|
|
86
|
+
return !!(this.requests && Array.isArray(this.requests) && this.requests.length > 0);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* 设置历史获取开始
|
|
91
|
+
*/
|
|
92
|
+
startHistoryFetch() {
|
|
93
|
+
this.status = TaskStatus.FETCHING;
|
|
94
|
+
this.historyFetchStartTime = Date.now();
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
/**
|
|
98
|
+
* 设置历史获取完成
|
|
99
|
+
* @param {Object} historyData - 历史数据
|
|
100
|
+
*/
|
|
101
|
+
completeHistoryFetch(historyData) {
|
|
102
|
+
this.historyData = historyData;
|
|
103
|
+
this.requests = historyData.requests || [];
|
|
104
|
+
this.status = TaskStatus.READY;
|
|
105
|
+
this.historyFetchEndTime = Date.now();
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
/**
|
|
109
|
+
* 设置历史获取失败
|
|
110
|
+
* @param {Error} error - 错误对象
|
|
111
|
+
*/
|
|
112
|
+
failHistoryFetch(error) {
|
|
113
|
+
this.status = TaskStatus.FAILED;
|
|
114
|
+
this.error = error.message || String(error);
|
|
115
|
+
this.errorPhase = ErrorPhase.HISTORY_FETCH;
|
|
116
|
+
this.historyFetchEndTime = Date.now();
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* 设置回放开始
|
|
121
|
+
*/
|
|
122
|
+
startReplay() {
|
|
123
|
+
this.status = TaskStatus.RUNNING;
|
|
124
|
+
this.replayStartTime = Date.now();
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* 设置回放完成
|
|
129
|
+
* @param {Object} result - 回放结果
|
|
130
|
+
* @param {string} newSandboxId - 新沙箱 ID
|
|
131
|
+
* @param {Array} pidMappings - PID 映射
|
|
132
|
+
*/
|
|
133
|
+
completeReplay(result, newSandboxId, pidMappings = []) {
|
|
134
|
+
// 如果有失败的请求,标记为 FAILED 而不是 COMPLETED
|
|
135
|
+
if (result && result.failed > 0) {
|
|
136
|
+
this.status = TaskStatus.FAILED;
|
|
137
|
+
this.error = `${result.failed} request(s) failed during replay`;
|
|
138
|
+
this.errorPhase = ErrorPhase.REPLAY;
|
|
139
|
+
} else {
|
|
140
|
+
this.status = TaskStatus.COMPLETED;
|
|
141
|
+
}
|
|
142
|
+
this.replayResult = result;
|
|
143
|
+
this.newSandboxId = newSandboxId;
|
|
144
|
+
this.pidMappings = pidMappings;
|
|
145
|
+
this.replayEndTime = Date.now();
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
/**
|
|
149
|
+
* 设置回放失败
|
|
150
|
+
* @param {Error} error - 错误对象
|
|
151
|
+
*/
|
|
152
|
+
failReplay(error) {
|
|
153
|
+
this.status = TaskStatus.FAILED;
|
|
154
|
+
this.error = error.message || String(error);
|
|
155
|
+
this.errorPhase = ErrorPhase.REPLAY;
|
|
156
|
+
this.replayEndTime = Date.now();
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
/**
|
|
160
|
+
* 获取历史获取耗时(毫秒)
|
|
161
|
+
* @returns {number|null}
|
|
162
|
+
*/
|
|
163
|
+
getHistoryFetchDuration() {
|
|
164
|
+
if (this.historyFetchStartTime && this.historyFetchEndTime) {
|
|
165
|
+
return this.historyFetchEndTime - this.historyFetchStartTime;
|
|
166
|
+
}
|
|
167
|
+
return null;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
/**
|
|
171
|
+
* 获取回放耗时(毫秒)
|
|
172
|
+
* @returns {number|null}
|
|
173
|
+
*/
|
|
174
|
+
getReplayDuration() {
|
|
175
|
+
if (this.replayStartTime && this.replayEndTime) {
|
|
176
|
+
return this.replayEndTime - this.replayStartTime;
|
|
177
|
+
}
|
|
178
|
+
return null;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
/**
|
|
182
|
+
* 转换为结果对象
|
|
183
|
+
* @returns {SandboxResult}
|
|
184
|
+
*/
|
|
185
|
+
toResult() {
|
|
186
|
+
return new SandboxResult(this);
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
/**
|
|
191
|
+
* 单个沙箱的执行结果
|
|
192
|
+
*/
|
|
193
|
+
class SandboxResult {
|
|
194
|
+
/**
|
|
195
|
+
* @param {SandboxTask} task - 任务对象
|
|
196
|
+
*/
|
|
197
|
+
constructor(task) {
|
|
198
|
+
this.index = task.index;
|
|
199
|
+
this.sandboxId = task.sandboxId;
|
|
200
|
+
this.name = task.name;
|
|
201
|
+
this.tags = task.tags;
|
|
202
|
+
this.status = task.status;
|
|
203
|
+
this.newSandboxId = task.newSandboxId;
|
|
204
|
+
|
|
205
|
+
// 历史获取结果
|
|
206
|
+
this.historyFetch = {
|
|
207
|
+
success: task.status !== TaskStatus.FAILED || task.errorPhase !== ErrorPhase.HISTORY_FETCH,
|
|
208
|
+
requestCount: task.requests ? task.requests.length : 0,
|
|
209
|
+
duration: task.getHistoryFetchDuration(),
|
|
210
|
+
error: task.errorPhase === ErrorPhase.HISTORY_FETCH ? task.error : null
|
|
211
|
+
};
|
|
212
|
+
|
|
213
|
+
// 回放结果
|
|
214
|
+
if (task.replayResult) {
|
|
215
|
+
this.replay = {
|
|
216
|
+
originalRequests: task.replayResult.total || 0,
|
|
217
|
+
executedRequests: (task.replayResult.success || 0) + (task.replayResult.failed || 0),
|
|
218
|
+
success: task.replayResult.success || 0,
|
|
219
|
+
failed: task.replayResult.failed || 0,
|
|
220
|
+
skipped: task.replayResult.skipped || 0,
|
|
221
|
+
duration: task.getReplayDuration()
|
|
222
|
+
};
|
|
223
|
+
} else {
|
|
224
|
+
this.replay = null;
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
// PID 映射
|
|
228
|
+
this.pidMappings = task.pidMappings;
|
|
229
|
+
|
|
230
|
+
// 错误信息
|
|
231
|
+
this.errors = [];
|
|
232
|
+
if (task.error) {
|
|
233
|
+
this.errors.push({
|
|
234
|
+
phase: task.errorPhase,
|
|
235
|
+
message: task.error
|
|
236
|
+
});
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
// 日志文件
|
|
240
|
+
this.logFile = task.logFile;
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
/**
|
|
245
|
+
* 批量回放结果汇总
|
|
246
|
+
*/
|
|
247
|
+
class BatchReplayResult {
|
|
248
|
+
/**
|
|
249
|
+
* @param {Object} options - 配置选项
|
|
250
|
+
*/
|
|
251
|
+
constructor(options = {}) {
|
|
252
|
+
this.summary = {
|
|
253
|
+
totalSandboxes: 0,
|
|
254
|
+
successful: 0,
|
|
255
|
+
failed: 0,
|
|
256
|
+
skipped: 0,
|
|
257
|
+
startTime: null,
|
|
258
|
+
endTime: null,
|
|
259
|
+
totalDuration: null,
|
|
260
|
+
concurrency: options.concurrency || 1
|
|
261
|
+
};
|
|
262
|
+
|
|
263
|
+
this.sandboxResults = [];
|
|
264
|
+
this.failedSandboxes = [];
|
|
265
|
+
|
|
266
|
+
this.metadata = {
|
|
267
|
+
inputFile: options.inputFile || null,
|
|
268
|
+
inputMetadata: options.inputMetadata || null,
|
|
269
|
+
cliVersion: options.cliVersion || null,
|
|
270
|
+
mode: options.mode || 'smart',
|
|
271
|
+
timing: options.timing || 'none'
|
|
272
|
+
};
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
/**
|
|
276
|
+
* 设置开始时间
|
|
277
|
+
*/
|
|
278
|
+
start() {
|
|
279
|
+
this.summary.startTime = new Date().toISOString();
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
/**
|
|
283
|
+
* 设置结束时间并计算总耗时
|
|
284
|
+
*/
|
|
285
|
+
end() {
|
|
286
|
+
this.summary.endTime = new Date().toISOString();
|
|
287
|
+
|
|
288
|
+
if (this.summary.startTime && this.summary.endTime) {
|
|
289
|
+
const startMs = new Date(this.summary.startTime).getTime();
|
|
290
|
+
const endMs = new Date(this.summary.endTime).getTime();
|
|
291
|
+
const durationMs = endMs - startMs;
|
|
292
|
+
this.summary.totalDuration = formatDuration(durationMs);
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
/**
|
|
297
|
+
* 添加沙箱结果
|
|
298
|
+
* @param {SandboxResult} result - 沙箱结果
|
|
299
|
+
*/
|
|
300
|
+
addResult(result) {
|
|
301
|
+
this.sandboxResults.push(result);
|
|
302
|
+
this.summary.totalSandboxes++;
|
|
303
|
+
|
|
304
|
+
if (result.status === TaskStatus.COMPLETED) {
|
|
305
|
+
this.summary.successful++;
|
|
306
|
+
} else if (result.status === TaskStatus.FAILED) {
|
|
307
|
+
this.summary.failed++;
|
|
308
|
+
this.failedSandboxes.push({
|
|
309
|
+
sandboxId: result.sandboxId,
|
|
310
|
+
name: result.name,
|
|
311
|
+
error: result.errors.length > 0 ? result.errors[0].message : 'Unknown error',
|
|
312
|
+
phase: result.errors.length > 0 ? result.errors[0].phase : 'unknown'
|
|
313
|
+
});
|
|
314
|
+
} else if (result.status === TaskStatus.SKIPPED) {
|
|
315
|
+
this.summary.skipped++;
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
/**
|
|
320
|
+
* 从任务列表生成结果
|
|
321
|
+
* @param {SandboxTask[]} tasks - 任务列表
|
|
322
|
+
*/
|
|
323
|
+
generateFromTasks(tasks) {
|
|
324
|
+
for (const task of tasks) {
|
|
325
|
+
this.addResult(task.toResult());
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
/**
|
|
330
|
+
* 转换为 JSON 对象
|
|
331
|
+
* @returns {Object}
|
|
332
|
+
*/
|
|
333
|
+
toJSON() {
|
|
334
|
+
return {
|
|
335
|
+
summary: this.summary,
|
|
336
|
+
sandboxResults: this.sandboxResults,
|
|
337
|
+
failedSandboxes: this.failedSandboxes,
|
|
338
|
+
metadata: this.metadata
|
|
339
|
+
};
|
|
340
|
+
}
|
|
341
|
+
}
|
|
342
|
+
|
|
343
|
+
/**
|
|
344
|
+
* 格式化持续时间
|
|
345
|
+
* @param {number} ms - 毫秒
|
|
346
|
+
* @returns {string}
|
|
347
|
+
*/
|
|
348
|
+
function formatDuration(ms) {
|
|
349
|
+
if (ms === null || ms === undefined) return '-';
|
|
350
|
+
|
|
351
|
+
if (ms < 1000) {
|
|
352
|
+
return `${ms}ms`;
|
|
353
|
+
} else if (ms < 60000) {
|
|
354
|
+
return `${(ms / 1000).toFixed(1)}s`;
|
|
355
|
+
} else {
|
|
356
|
+
const minutes = Math.floor(ms / 60000);
|
|
357
|
+
const seconds = Math.floor((ms % 60000) / 1000);
|
|
358
|
+
return `${minutes}m ${seconds}s`;
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
module.exports = {
|
|
363
|
+
TaskStatus,
|
|
364
|
+
ErrorPhase,
|
|
365
|
+
SandboxTask,
|
|
366
|
+
SandboxResult,
|
|
367
|
+
BatchReplayResult,
|
|
368
|
+
formatDuration
|
|
369
|
+
};
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 格式化持续时间
|
|
3
|
+
* @param {number} ms - 毫秒
|
|
4
|
+
* @returns {string}
|
|
5
|
+
*/
|
|
6
|
+
function formatDuration(ms) {
|
|
7
|
+
if (ms == null || isNaN(ms)) return 'N/A';
|
|
8
|
+
if (ms < 1000) return `${ms}ms`;
|
|
9
|
+
if (ms < 60000) return `${(ms / 1000).toFixed(1)}s`;
|
|
10
|
+
const minutes = Math.floor(ms / 60000);
|
|
11
|
+
const seconds = Math.round((ms % 60000) / 1000);
|
|
12
|
+
return `${minutes}m${seconds}s`;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* 显示并发回放结果摘要
|
|
17
|
+
* @param {ConcurrentReplayResult} result
|
|
18
|
+
* @param {Object} options
|
|
19
|
+
* @param {boolean} options.quiet - 静默模式
|
|
20
|
+
*/
|
|
21
|
+
function displayConcurrentResultsSummary(result, options = {}) {
|
|
22
|
+
if (options.quiet) return;
|
|
23
|
+
|
|
24
|
+
console.log('\n═══════════════════════════════════════════════════════════════');
|
|
25
|
+
console.log('📊 Concurrent Replay Summary');
|
|
26
|
+
console.log('═══════════════════════════════════════════════════════════════');
|
|
27
|
+
|
|
28
|
+
console.log(`\nConcurrency: ${result.concurrency} sandboxes`);
|
|
29
|
+
console.log(`Status: ${result.allSucceeded ? '✅ All succeeded' : '⚠️ Some failed'}`);
|
|
30
|
+
|
|
31
|
+
// 显示每个 worker 的结果
|
|
32
|
+
console.log('\nPer-Sandbox Results:');
|
|
33
|
+
console.log('┌─────────┬────────────────────────────────────────┬─────────┬─────────┬─────────┐');
|
|
34
|
+
console.log('│ Worker │ Sandbox ID │ Success │ Failed │ Duration│');
|
|
35
|
+
console.log('├─────────┼────────────────────────────────────────┼─────────┼─────────┼─────────┤');
|
|
36
|
+
|
|
37
|
+
result.workers.forEach(w => {
|
|
38
|
+
const status = w.status === 'completed' ? '✅' : '❌';
|
|
39
|
+
const sandboxId = (w.sandboxId || 'N/A').padEnd(38).substring(0, 38);
|
|
40
|
+
const success = String(w.results?.success || 0).padStart(7);
|
|
41
|
+
const failed = String(w.results?.failed || 0).padStart(7);
|
|
42
|
+
const duration = formatDuration(w.duration).padStart(7);
|
|
43
|
+
|
|
44
|
+
console.log(`│ ${status} ${String(w.workerId).padEnd(5)} │ ${sandboxId} │ ${success} │ ${failed} │ ${duration} │`);
|
|
45
|
+
|
|
46
|
+
if (w.error) {
|
|
47
|
+
const errorMsg = w.error.length > 60 ? w.error.substring(0, 57) + '...' : w.error;
|
|
48
|
+
console.log(`│ │ Error: ${errorMsg.padEnd(52)} │`);
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
console.log('└─────────┴────────────────────────────────────────┴─────────┴─────────┴─────────┘');
|
|
53
|
+
|
|
54
|
+
// 显示汇总
|
|
55
|
+
console.log('\nAggregated Results:');
|
|
56
|
+
console.log(`├── ✅ Total Successful: ${result.aggregated.success}`);
|
|
57
|
+
console.log(`├── ❌ Total Failed: ${result.aggregated.failed}`);
|
|
58
|
+
console.log(`└── ⏭️ Total Skipped: ${result.aggregated.skipped}`);
|
|
59
|
+
|
|
60
|
+
// 显示日志文件
|
|
61
|
+
console.log('\nLog Files:');
|
|
62
|
+
result.workers.forEach((w, index) => {
|
|
63
|
+
const prefix = index === result.workers.length - 1 ? '└──' : '├──';
|
|
64
|
+
console.log(`${prefix} Worker ${w.workerId}: ${w.logFile}`);
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
console.log('═══════════════════════════════════════════════════════════════\n');
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
module.exports = { displayConcurrentResultsSummary, formatDuration };
|
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
const { ReplayWorker } = require('./worker');
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* 并发回放协调器
|
|
5
|
+
* 管理多个 ReplayWorker 的并发执行
|
|
6
|
+
*/
|
|
7
|
+
class ConcurrentReplayOrchestrator {
|
|
8
|
+
/**
|
|
9
|
+
* @param {Object} options
|
|
10
|
+
* @param {number} options.concurrency - 并发数
|
|
11
|
+
* @param {Object} options.config - 沙箱配置
|
|
12
|
+
* @param {boolean} options.quiet - 静默模式
|
|
13
|
+
* @param {string} options.logFile - 日志文件基础路径
|
|
14
|
+
* @param {string} options.failureStrategy - 失败策略: 'continue' | 'stop-all'
|
|
15
|
+
*/
|
|
16
|
+
constructor(options) {
|
|
17
|
+
this.concurrency = options.concurrency || 1;
|
|
18
|
+
this.config = options.config;
|
|
19
|
+
this.quiet = options.quiet || false;
|
|
20
|
+
this.logFile = options.logFile || 'replay.log';
|
|
21
|
+
this.failureStrategy = options.failureStrategy || 'continue';
|
|
22
|
+
|
|
23
|
+
this.workers = [];
|
|
24
|
+
this.progressIntervalId = null;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* 执行并发回放
|
|
29
|
+
* @param {ReplayPlan} plan - 回放计划
|
|
30
|
+
* @param {Object} executorOptions - 执行器选项
|
|
31
|
+
* @returns {Promise<ConcurrentReplayResult>}
|
|
32
|
+
*/
|
|
33
|
+
async execute(plan, executorOptions) {
|
|
34
|
+
// 1. 显示并发回放摘要
|
|
35
|
+
this.displayConcurrentSummary(plan);
|
|
36
|
+
|
|
37
|
+
// 2. 创建所有 workers
|
|
38
|
+
for (let i = 1; i <= this.concurrency; i++) {
|
|
39
|
+
const worker = new ReplayWorker(i, this.config, plan, {
|
|
40
|
+
...executorOptions,
|
|
41
|
+
logFile: this.logFile
|
|
42
|
+
});
|
|
43
|
+
this.workers.push(worker);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// 3. 启动进度监控(如果非静默模式)
|
|
47
|
+
if (!this.quiet) {
|
|
48
|
+
this.startProgressMonitor();
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// 4. 并发执行所有 workers
|
|
52
|
+
let workerResults;
|
|
53
|
+
|
|
54
|
+
if (this.failureStrategy === 'stop-all') {
|
|
55
|
+
// 使用 Promise.all,任一失败则全部停止
|
|
56
|
+
try {
|
|
57
|
+
workerResults = await Promise.all(
|
|
58
|
+
this.workers.map(w => w.execute())
|
|
59
|
+
);
|
|
60
|
+
} catch (error) {
|
|
61
|
+
// 如果 Promise.all 抛出异常,收集已有结果
|
|
62
|
+
workerResults = this.workers.map(w => w.getWorkerResult());
|
|
63
|
+
}
|
|
64
|
+
} else {
|
|
65
|
+
// 使用 Promise.allSettled,允许部分失败
|
|
66
|
+
const settled = await Promise.allSettled(
|
|
67
|
+
this.workers.map(w => w.execute())
|
|
68
|
+
);
|
|
69
|
+
workerResults = settled.map((result, index) => {
|
|
70
|
+
if (result.status === 'fulfilled') {
|
|
71
|
+
return result.value;
|
|
72
|
+
} else {
|
|
73
|
+
return this.workers[index].getWorkerResult();
|
|
74
|
+
}
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// 5. 停止进度监控
|
|
79
|
+
this.stopProgressMonitor();
|
|
80
|
+
|
|
81
|
+
// 6. 汇总结果
|
|
82
|
+
return this.aggregateResults(workerResults);
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
/**
|
|
86
|
+
* 显示并发回放摘要
|
|
87
|
+
* @param {ReplayPlan} plan
|
|
88
|
+
*/
|
|
89
|
+
displayConcurrentSummary(plan) {
|
|
90
|
+
if (this.quiet) return;
|
|
91
|
+
|
|
92
|
+
console.log(`\n🔀 Concurrent Replay Mode`);
|
|
93
|
+
console.log(` Concurrency: ${this.concurrency} sandboxes`);
|
|
94
|
+
console.log(` Requests per sandbox: ${plan.executeCount}`);
|
|
95
|
+
console.log(` Total requests: ${plan.executeCount * this.concurrency}`);
|
|
96
|
+
console.log(`\n⏳ Starting ${this.concurrency} sandboxes...\n`);
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* 启动进度监控
|
|
101
|
+
*/
|
|
102
|
+
startProgressMonitor() {
|
|
103
|
+
this.progressIntervalId = setInterval(() => {
|
|
104
|
+
this.displayProgress();
|
|
105
|
+
}, 10000); // 每 10 秒更新一次进度
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
/**
|
|
109
|
+
* 停止进度监控
|
|
110
|
+
*/
|
|
111
|
+
stopProgressMonitor() {
|
|
112
|
+
if (this.progressIntervalId) {
|
|
113
|
+
clearInterval(this.progressIntervalId);
|
|
114
|
+
this.progressIntervalId = null;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
/**
|
|
119
|
+
* 显示当前进度
|
|
120
|
+
*/
|
|
121
|
+
displayProgress() {
|
|
122
|
+
const statuses = this.workers.map(w => {
|
|
123
|
+
const progress = w.getProgress();
|
|
124
|
+
const icon = progress.status === 'completed' ? '✅' :
|
|
125
|
+
progress.status === 'failed' ? '❌' :
|
|
126
|
+
progress.status === 'running' ? '⏳' : '⏸️';
|
|
127
|
+
const sandboxId = progress.sandboxId || '--------';
|
|
128
|
+
return ` Worker ${progress.workerId}: ${icon} sandbox=${sandboxId}... ${progress.success}/${progress.total}`;
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
console.log('\n--- Progress Update ---');
|
|
132
|
+
statuses.forEach(s => console.log(s));
|
|
133
|
+
console.log('-----------------------\n');
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
/**
|
|
137
|
+
* 汇总执行结果
|
|
138
|
+
* @param {WorkerResult[]} workerResults
|
|
139
|
+
* @returns {ConcurrentReplayResult}
|
|
140
|
+
*/
|
|
141
|
+
aggregateResults(workerResults) {
|
|
142
|
+
const totalResults = {
|
|
143
|
+
total: 0,
|
|
144
|
+
success: 0,
|
|
145
|
+
failed: 0,
|
|
146
|
+
skipped: 0
|
|
147
|
+
};
|
|
148
|
+
|
|
149
|
+
workerResults.forEach(r => {
|
|
150
|
+
if (r.results) {
|
|
151
|
+
totalResults.total += r.results.total;
|
|
152
|
+
totalResults.success += r.results.success;
|
|
153
|
+
totalResults.failed += r.results.failed;
|
|
154
|
+
totalResults.skipped += r.results.skipped;
|
|
155
|
+
}
|
|
156
|
+
});
|
|
157
|
+
|
|
158
|
+
return {
|
|
159
|
+
concurrency: this.concurrency,
|
|
160
|
+
workers: workerResults,
|
|
161
|
+
aggregated: totalResults,
|
|
162
|
+
allSucceeded: workerResults.every(r => r.status === 'completed'),
|
|
163
|
+
sandboxIds: workerResults
|
|
164
|
+
.filter(r => r.sandboxId)
|
|
165
|
+
.map(r => ({ workerId: r.workerId, sandboxId: r.sandboxId }))
|
|
166
|
+
};
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
module.exports = { ConcurrentReplayOrchestrator };
|