chainlesschain 0.132.0 → 0.145.0
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/package.json +1 -1
- package/src/commands/a2a.js +230 -0
- package/src/commands/activitypub.js +191 -0
- package/src/commands/agent.js +601 -0
- package/src/commands/audit.js +206 -0
- package/src/commands/bi.js +186 -0
- package/src/commands/bm25.js +162 -0
- package/src/commands/browse.js +225 -0
- package/src/commands/ccron.js +178 -0
- package/src/commands/chat.js +207 -0
- package/src/commands/compliance.js +420 -0
- package/src/commands/compt.js +176 -0
- package/src/commands/consol.js +237 -0
- package/src/commands/cowork.js +588 -0
- package/src/commands/crosschain.js +216 -0
- package/src/commands/dao.js +216 -0
- package/src/commands/dlp.js +206 -0
- package/src/commands/economy.js +211 -0
- package/src/commands/evolution.js +209 -0
- package/src/commands/evomap.js +216 -0
- package/src/commands/fflag.js +230 -0
- package/src/commands/git.js +185 -0
- package/src/commands/hardening.js +209 -0
- package/src/commands/hmemory.js +210 -0
- package/src/commands/incentive.js +209 -0
- package/src/commands/inference.js +178 -0
- package/src/commands/itbudget.js +161 -0
- package/src/commands/kg.js +206 -0
- package/src/commands/lowcode.js +201 -0
- package/src/commands/marketplace.js +206 -0
- package/src/commands/matrix.js +214 -0
- package/src/commands/mcpscaf.js +153 -0
- package/src/commands/meminj.js +153 -0
- package/src/commands/nostr.js +213 -0
- package/src/commands/orchestrate.js +217 -0
- package/src/commands/orchgov.js +156 -0
- package/src/commands/pdfp.js +160 -0
- package/src/commands/perf.js +176 -0
- package/src/commands/perm.js +156 -0
- package/src/commands/pipeline.js +211 -0
- package/src/commands/planmode.js +154 -0
- package/src/commands/privacy.js +203 -0
- package/src/commands/promcomp.js +166 -0
- package/src/commands/recommend.js +185 -0
- package/src/commands/reputation.js +208 -0
- package/src/commands/sandbox.js +206 -0
- package/src/commands/seshhook.js +153 -0
- package/src/commands/seshsearch.js +149 -0
- package/src/commands/seshtail.js +152 -0
- package/src/commands/seshu.js +160 -0
- package/src/commands/sganal.js +172 -0
- package/src/commands/siem.js +207 -0
- package/src/commands/sla.js +212 -0
- package/src/commands/slotfill.js +154 -0
- package/src/commands/social.js +159 -0
- package/src/commands/stress.js +206 -0
- package/src/commands/svccont.js +157 -0
- package/src/commands/terraform.js +206 -0
- package/src/commands/tms.js +183 -0
- package/src/commands/topiccls.js +158 -0
- package/src/commands/uprof.js +154 -0
- package/src/commands/vcheck.js +172 -0
- package/src/commands/webfetch.js +150 -0
- package/src/commands/zkp.js +218 -0
- package/src/harness/prompt-compressor.js +331 -0
- package/src/index.js +101 -1
- package/src/lib/a2a-protocol.js +373 -0
- package/src/lib/activitypub-bridge.js +343 -0
- package/src/lib/agent-economy.js +358 -0
- package/src/lib/app-builder.js +338 -0
- package/src/lib/audit-logger.js +321 -0
- package/src/lib/autonomous-agent.js +341 -0
- package/src/lib/bi-engine.js +339 -0
- package/src/lib/bm25-search.js +333 -0
- package/src/lib/browser-automation.js +352 -0
- package/src/lib/chat-core.js +336 -0
- package/src/lib/claude-code-bridge.js +341 -0
- package/src/lib/compliance-framework-reporter.js +359 -0
- package/src/lib/compliance-manager.js +330 -0
- package/src/lib/compression-telemetry.js +333 -0
- package/src/lib/content-recommender.js +370 -0
- package/src/lib/cowork-cron.js +330 -0
- package/src/lib/cowork-learning.js +333 -0
- package/src/lib/cowork-task-runner.js +362 -0
- package/src/lib/cowork-workflow.js +327 -0
- package/src/lib/cross-chain.js +365 -0
- package/src/lib/dao-governance.js +339 -0
- package/src/lib/dlp-engine.js +343 -0
- package/src/lib/evolution-system.js +336 -0
- package/src/lib/evomap-manager.js +339 -0
- package/src/lib/execution-backend.js +351 -0
- package/src/lib/feature-flags.js +330 -0
- package/src/lib/git-integration.js +343 -0
- package/src/lib/hardening-manager.js +341 -0
- package/src/lib/hierarchical-memory.js +341 -0
- package/src/lib/inference-network.js +362 -0
- package/src/lib/iteration-budget.js +357 -0
- package/src/lib/knowledge-graph.js +333 -0
- package/src/lib/matrix-bridge.js +339 -0
- package/src/lib/mcp-scaffold.js +345 -0
- package/src/lib/memory-injection.js +320 -0
- package/src/lib/nostr-bridge.js +342 -0
- package/src/lib/orchestrator.js +350 -0
- package/src/lib/pdf-parser.js +330 -0
- package/src/lib/perf-tuning.js +364 -0
- package/src/lib/permission-engine.js +319 -0
- package/src/lib/pipeline-orchestrator.js +345 -0
- package/src/lib/plan-mode.js +328 -0
- package/src/lib/privacy-computing.js +335 -0
- package/src/lib/prompt-compressor.js +1 -10
- package/src/lib/reputation-optimizer.js +340 -0
- package/src/lib/sandbox-v2.js +327 -0
- package/src/lib/service-container.js +342 -0
- package/src/lib/session-consolidator.js +352 -0
- package/src/lib/session-hooks.js +340 -0
- package/src/lib/session-search.js +334 -0
- package/src/lib/session-tail.js +320 -0
- package/src/lib/session-usage.js +329 -0
- package/src/lib/siem-exporter.js +352 -0
- package/src/lib/skill-marketplace.js +345 -0
- package/src/lib/sla-manager.js +341 -0
- package/src/lib/slot-filler.js +333 -0
- package/src/lib/social-graph-analytics.js +327 -0
- package/src/lib/social-graph.js +304 -0
- package/src/lib/stress-tester.js +342 -0
- package/src/lib/sub-agent-registry.js +359 -0
- package/src/lib/task-model-selector.js +333 -0
- package/src/lib/terraform-manager.js +333 -0
- package/src/lib/todo-manager.js +339 -0
- package/src/lib/token-incentive.js +341 -0
- package/src/lib/topic-classifier.js +353 -0
- package/src/lib/user-profile.js +325 -0
- package/src/lib/version-checker.js +335 -0
- package/src/lib/web-fetch.js +322 -0
- package/src/lib/zkp-engine.js +342 -0
|
@@ -225,3 +225,342 @@ export class EvoMapManager {
|
|
|
225
225
|
};
|
|
226
226
|
}
|
|
227
227
|
}
|
|
228
|
+
|
|
229
|
+
// ===== V2 Surface: EvoMap Manager governance overlay (CLI v0.135.0) =====
|
|
230
|
+
export const EVOMAP_MAP_MATURITY_V2 = Object.freeze({
|
|
231
|
+
PENDING: "pending",
|
|
232
|
+
ACTIVE: "active",
|
|
233
|
+
STALE: "stale",
|
|
234
|
+
ARCHIVED: "archived",
|
|
235
|
+
});
|
|
236
|
+
export const EVOMAP_EVOLUTION_LIFECYCLE_V2 = Object.freeze({
|
|
237
|
+
QUEUED: "queued",
|
|
238
|
+
RUNNING: "running",
|
|
239
|
+
COMPLETED: "completed",
|
|
240
|
+
FAILED: "failed",
|
|
241
|
+
CANCELLED: "cancelled",
|
|
242
|
+
});
|
|
243
|
+
|
|
244
|
+
const _emMapTrans = new Map([
|
|
245
|
+
[
|
|
246
|
+
EVOMAP_MAP_MATURITY_V2.PENDING,
|
|
247
|
+
new Set([EVOMAP_MAP_MATURITY_V2.ACTIVE, EVOMAP_MAP_MATURITY_V2.ARCHIVED]),
|
|
248
|
+
],
|
|
249
|
+
[
|
|
250
|
+
EVOMAP_MAP_MATURITY_V2.ACTIVE,
|
|
251
|
+
new Set([EVOMAP_MAP_MATURITY_V2.STALE, EVOMAP_MAP_MATURITY_V2.ARCHIVED]),
|
|
252
|
+
],
|
|
253
|
+
[
|
|
254
|
+
EVOMAP_MAP_MATURITY_V2.STALE,
|
|
255
|
+
new Set([EVOMAP_MAP_MATURITY_V2.ACTIVE, EVOMAP_MAP_MATURITY_V2.ARCHIVED]),
|
|
256
|
+
],
|
|
257
|
+
[EVOMAP_MAP_MATURITY_V2.ARCHIVED, new Set()],
|
|
258
|
+
]);
|
|
259
|
+
const _emMapTerminal = new Set([EVOMAP_MAP_MATURITY_V2.ARCHIVED]);
|
|
260
|
+
const _emEvoTrans = new Map([
|
|
261
|
+
[
|
|
262
|
+
EVOMAP_EVOLUTION_LIFECYCLE_V2.QUEUED,
|
|
263
|
+
new Set([
|
|
264
|
+
EVOMAP_EVOLUTION_LIFECYCLE_V2.RUNNING,
|
|
265
|
+
EVOMAP_EVOLUTION_LIFECYCLE_V2.CANCELLED,
|
|
266
|
+
]),
|
|
267
|
+
],
|
|
268
|
+
[
|
|
269
|
+
EVOMAP_EVOLUTION_LIFECYCLE_V2.RUNNING,
|
|
270
|
+
new Set([
|
|
271
|
+
EVOMAP_EVOLUTION_LIFECYCLE_V2.COMPLETED,
|
|
272
|
+
EVOMAP_EVOLUTION_LIFECYCLE_V2.FAILED,
|
|
273
|
+
EVOMAP_EVOLUTION_LIFECYCLE_V2.CANCELLED,
|
|
274
|
+
]),
|
|
275
|
+
],
|
|
276
|
+
[EVOMAP_EVOLUTION_LIFECYCLE_V2.COMPLETED, new Set()],
|
|
277
|
+
[EVOMAP_EVOLUTION_LIFECYCLE_V2.FAILED, new Set()],
|
|
278
|
+
[EVOMAP_EVOLUTION_LIFECYCLE_V2.CANCELLED, new Set()],
|
|
279
|
+
]);
|
|
280
|
+
|
|
281
|
+
const _emMaps = new Map();
|
|
282
|
+
const _emEvos = new Map();
|
|
283
|
+
let _emMaxActivePerOwner = 10;
|
|
284
|
+
let _emMaxPendingPerMap = 15;
|
|
285
|
+
let _emMapIdleMs = 7 * 24 * 60 * 60 * 1000;
|
|
286
|
+
let _emEvoStuckMs = 5 * 60 * 1000;
|
|
287
|
+
|
|
288
|
+
function _emPos(n, lbl) {
|
|
289
|
+
const v = Math.floor(Number(n));
|
|
290
|
+
if (!Number.isFinite(v) || v <= 0)
|
|
291
|
+
throw new Error(`${lbl} must be positive integer`);
|
|
292
|
+
return v;
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
export function setMaxActiveEvoMapsPerOwnerV2(n) {
|
|
296
|
+
_emMaxActivePerOwner = _emPos(n, "maxActiveEvoMapsPerOwner");
|
|
297
|
+
}
|
|
298
|
+
export function getMaxActiveEvoMapsPerOwnerV2() {
|
|
299
|
+
return _emMaxActivePerOwner;
|
|
300
|
+
}
|
|
301
|
+
export function setMaxPendingEvoEvolutionsPerMapV2(n) {
|
|
302
|
+
_emMaxPendingPerMap = _emPos(n, "maxPendingEvoEvolutionsPerMap");
|
|
303
|
+
}
|
|
304
|
+
export function getMaxPendingEvoEvolutionsPerMapV2() {
|
|
305
|
+
return _emMaxPendingPerMap;
|
|
306
|
+
}
|
|
307
|
+
export function setEvoMapIdleMsV2(n) {
|
|
308
|
+
_emMapIdleMs = _emPos(n, "evoMapIdleMs");
|
|
309
|
+
}
|
|
310
|
+
export function getEvoMapIdleMsV2() {
|
|
311
|
+
return _emMapIdleMs;
|
|
312
|
+
}
|
|
313
|
+
export function setEvoEvolutionStuckMsV2(n) {
|
|
314
|
+
_emEvoStuckMs = _emPos(n, "evoEvolutionStuckMs");
|
|
315
|
+
}
|
|
316
|
+
export function getEvoEvolutionStuckMsV2() {
|
|
317
|
+
return _emEvoStuckMs;
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
export function _resetStateEvoMapManagerV2() {
|
|
321
|
+
_emMaps.clear();
|
|
322
|
+
_emEvos.clear();
|
|
323
|
+
_emMaxActivePerOwner = 10;
|
|
324
|
+
_emMaxPendingPerMap = 15;
|
|
325
|
+
_emMapIdleMs = 7 * 24 * 60 * 60 * 1000;
|
|
326
|
+
_emEvoStuckMs = 5 * 60 * 1000;
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
export function registerEvoMapV2({ id, owner, name, metadata } = {}) {
|
|
330
|
+
if (!id || typeof id !== "string") throw new Error("id is required");
|
|
331
|
+
if (!owner || typeof owner !== "string") throw new Error("owner is required");
|
|
332
|
+
if (_emMaps.has(id)) throw new Error(`evomap ${id} already registered`);
|
|
333
|
+
const now = Date.now();
|
|
334
|
+
const m = {
|
|
335
|
+
id,
|
|
336
|
+
owner,
|
|
337
|
+
name: name || id,
|
|
338
|
+
status: EVOMAP_MAP_MATURITY_V2.PENDING,
|
|
339
|
+
createdAt: now,
|
|
340
|
+
updatedAt: now,
|
|
341
|
+
activatedAt: null,
|
|
342
|
+
archivedAt: null,
|
|
343
|
+
lastTouchedAt: now,
|
|
344
|
+
metadata: { ...(metadata || {}) },
|
|
345
|
+
};
|
|
346
|
+
_emMaps.set(id, m);
|
|
347
|
+
return { ...m, metadata: { ...m.metadata } };
|
|
348
|
+
}
|
|
349
|
+
function _emCheckM(from, to) {
|
|
350
|
+
const a = _emMapTrans.get(from);
|
|
351
|
+
if (!a || !a.has(to))
|
|
352
|
+
throw new Error(`invalid evomap transition ${from} → ${to}`);
|
|
353
|
+
}
|
|
354
|
+
function _emCountActive(owner) {
|
|
355
|
+
let n = 0;
|
|
356
|
+
for (const m of _emMaps.values())
|
|
357
|
+
if (m.owner === owner && m.status === EVOMAP_MAP_MATURITY_V2.ACTIVE) n++;
|
|
358
|
+
return n;
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
export function activateEvoMapV2(id) {
|
|
362
|
+
const m = _emMaps.get(id);
|
|
363
|
+
if (!m) throw new Error(`evomap ${id} not found`);
|
|
364
|
+
_emCheckM(m.status, EVOMAP_MAP_MATURITY_V2.ACTIVE);
|
|
365
|
+
const recovery = m.status === EVOMAP_MAP_MATURITY_V2.STALE;
|
|
366
|
+
if (!recovery) {
|
|
367
|
+
const a = _emCountActive(m.owner);
|
|
368
|
+
if (a >= _emMaxActivePerOwner)
|
|
369
|
+
throw new Error(
|
|
370
|
+
`max active evomaps per owner (${_emMaxActivePerOwner}) reached for ${m.owner}`,
|
|
371
|
+
);
|
|
372
|
+
}
|
|
373
|
+
const now = Date.now();
|
|
374
|
+
m.status = EVOMAP_MAP_MATURITY_V2.ACTIVE;
|
|
375
|
+
m.updatedAt = now;
|
|
376
|
+
m.lastTouchedAt = now;
|
|
377
|
+
if (!m.activatedAt) m.activatedAt = now;
|
|
378
|
+
return { ...m, metadata: { ...m.metadata } };
|
|
379
|
+
}
|
|
380
|
+
export function staleEvoMapV2(id) {
|
|
381
|
+
const m = _emMaps.get(id);
|
|
382
|
+
if (!m) throw new Error(`evomap ${id} not found`);
|
|
383
|
+
_emCheckM(m.status, EVOMAP_MAP_MATURITY_V2.STALE);
|
|
384
|
+
m.status = EVOMAP_MAP_MATURITY_V2.STALE;
|
|
385
|
+
m.updatedAt = Date.now();
|
|
386
|
+
return { ...m, metadata: { ...m.metadata } };
|
|
387
|
+
}
|
|
388
|
+
export function archiveEvoMapV2(id) {
|
|
389
|
+
const m = _emMaps.get(id);
|
|
390
|
+
if (!m) throw new Error(`evomap ${id} not found`);
|
|
391
|
+
_emCheckM(m.status, EVOMAP_MAP_MATURITY_V2.ARCHIVED);
|
|
392
|
+
const now = Date.now();
|
|
393
|
+
m.status = EVOMAP_MAP_MATURITY_V2.ARCHIVED;
|
|
394
|
+
m.updatedAt = now;
|
|
395
|
+
if (!m.archivedAt) m.archivedAt = now;
|
|
396
|
+
return { ...m, metadata: { ...m.metadata } };
|
|
397
|
+
}
|
|
398
|
+
export function touchEvoMapV2(id) {
|
|
399
|
+
const m = _emMaps.get(id);
|
|
400
|
+
if (!m) throw new Error(`evomap ${id} not found`);
|
|
401
|
+
if (_emMapTerminal.has(m.status))
|
|
402
|
+
throw new Error(`cannot touch terminal evomap ${id}`);
|
|
403
|
+
const now = Date.now();
|
|
404
|
+
m.lastTouchedAt = now;
|
|
405
|
+
m.updatedAt = now;
|
|
406
|
+
return { ...m, metadata: { ...m.metadata } };
|
|
407
|
+
}
|
|
408
|
+
export function getEvoMapV2(id) {
|
|
409
|
+
const m = _emMaps.get(id);
|
|
410
|
+
if (!m) return null;
|
|
411
|
+
return { ...m, metadata: { ...m.metadata } };
|
|
412
|
+
}
|
|
413
|
+
export function listEvoMapsV2() {
|
|
414
|
+
return [..._emMaps.values()].map((m) => ({
|
|
415
|
+
...m,
|
|
416
|
+
metadata: { ...m.metadata },
|
|
417
|
+
}));
|
|
418
|
+
}
|
|
419
|
+
|
|
420
|
+
function _emCountPending(mid) {
|
|
421
|
+
let n = 0;
|
|
422
|
+
for (const e of _emEvos.values())
|
|
423
|
+
if (
|
|
424
|
+
e.mapId === mid &&
|
|
425
|
+
(e.status === EVOMAP_EVOLUTION_LIFECYCLE_V2.QUEUED ||
|
|
426
|
+
e.status === EVOMAP_EVOLUTION_LIFECYCLE_V2.RUNNING)
|
|
427
|
+
)
|
|
428
|
+
n++;
|
|
429
|
+
return n;
|
|
430
|
+
}
|
|
431
|
+
|
|
432
|
+
export function createEvoEvolutionV2({ id, mapId, strategy, metadata } = {}) {
|
|
433
|
+
if (!id || typeof id !== "string") throw new Error("id is required");
|
|
434
|
+
if (!mapId || typeof mapId !== "string") throw new Error("mapId is required");
|
|
435
|
+
if (_emEvos.has(id)) throw new Error(`evo evolution ${id} already exists`);
|
|
436
|
+
if (!_emMaps.has(mapId)) throw new Error(`evomap ${mapId} not found`);
|
|
437
|
+
const pending = _emCountPending(mapId);
|
|
438
|
+
if (pending >= _emMaxPendingPerMap)
|
|
439
|
+
throw new Error(
|
|
440
|
+
`max pending evo evolutions per map (${_emMaxPendingPerMap}) reached for ${mapId}`,
|
|
441
|
+
);
|
|
442
|
+
const now = Date.now();
|
|
443
|
+
const e = {
|
|
444
|
+
id,
|
|
445
|
+
mapId,
|
|
446
|
+
strategy: strategy || "default",
|
|
447
|
+
status: EVOMAP_EVOLUTION_LIFECYCLE_V2.QUEUED,
|
|
448
|
+
createdAt: now,
|
|
449
|
+
updatedAt: now,
|
|
450
|
+
startedAt: null,
|
|
451
|
+
settledAt: null,
|
|
452
|
+
metadata: { ...(metadata || {}) },
|
|
453
|
+
};
|
|
454
|
+
_emEvos.set(id, e);
|
|
455
|
+
return { ...e, metadata: { ...e.metadata } };
|
|
456
|
+
}
|
|
457
|
+
function _emCheckE(from, to) {
|
|
458
|
+
const a = _emEvoTrans.get(from);
|
|
459
|
+
if (!a || !a.has(to))
|
|
460
|
+
throw new Error(`invalid evo evolution transition ${from} → ${to}`);
|
|
461
|
+
}
|
|
462
|
+
export function startEvoEvolutionV2(id) {
|
|
463
|
+
const e = _emEvos.get(id);
|
|
464
|
+
if (!e) throw new Error(`evo evolution ${id} not found`);
|
|
465
|
+
_emCheckE(e.status, EVOMAP_EVOLUTION_LIFECYCLE_V2.RUNNING);
|
|
466
|
+
const now = Date.now();
|
|
467
|
+
e.status = EVOMAP_EVOLUTION_LIFECYCLE_V2.RUNNING;
|
|
468
|
+
e.updatedAt = now;
|
|
469
|
+
if (!e.startedAt) e.startedAt = now;
|
|
470
|
+
return { ...e, metadata: { ...e.metadata } };
|
|
471
|
+
}
|
|
472
|
+
export function completeEvoEvolutionV2(id) {
|
|
473
|
+
const e = _emEvos.get(id);
|
|
474
|
+
if (!e) throw new Error(`evo evolution ${id} not found`);
|
|
475
|
+
_emCheckE(e.status, EVOMAP_EVOLUTION_LIFECYCLE_V2.COMPLETED);
|
|
476
|
+
const now = Date.now();
|
|
477
|
+
e.status = EVOMAP_EVOLUTION_LIFECYCLE_V2.COMPLETED;
|
|
478
|
+
e.updatedAt = now;
|
|
479
|
+
if (!e.settledAt) e.settledAt = now;
|
|
480
|
+
return { ...e, metadata: { ...e.metadata } };
|
|
481
|
+
}
|
|
482
|
+
export function failEvoEvolutionV2(id, reason) {
|
|
483
|
+
const e = _emEvos.get(id);
|
|
484
|
+
if (!e) throw new Error(`evo evolution ${id} not found`);
|
|
485
|
+
_emCheckE(e.status, EVOMAP_EVOLUTION_LIFECYCLE_V2.FAILED);
|
|
486
|
+
const now = Date.now();
|
|
487
|
+
e.status = EVOMAP_EVOLUTION_LIFECYCLE_V2.FAILED;
|
|
488
|
+
e.updatedAt = now;
|
|
489
|
+
if (!e.settledAt) e.settledAt = now;
|
|
490
|
+
if (reason) e.metadata.failReason = String(reason);
|
|
491
|
+
return { ...e, metadata: { ...e.metadata } };
|
|
492
|
+
}
|
|
493
|
+
export function cancelEvoEvolutionV2(id, reason) {
|
|
494
|
+
const e = _emEvos.get(id);
|
|
495
|
+
if (!e) throw new Error(`evo evolution ${id} not found`);
|
|
496
|
+
_emCheckE(e.status, EVOMAP_EVOLUTION_LIFECYCLE_V2.CANCELLED);
|
|
497
|
+
const now = Date.now();
|
|
498
|
+
e.status = EVOMAP_EVOLUTION_LIFECYCLE_V2.CANCELLED;
|
|
499
|
+
e.updatedAt = now;
|
|
500
|
+
if (!e.settledAt) e.settledAt = now;
|
|
501
|
+
if (reason) e.metadata.cancelReason = String(reason);
|
|
502
|
+
return { ...e, metadata: { ...e.metadata } };
|
|
503
|
+
}
|
|
504
|
+
export function getEvoEvolutionV2(id) {
|
|
505
|
+
const e = _emEvos.get(id);
|
|
506
|
+
if (!e) return null;
|
|
507
|
+
return { ...e, metadata: { ...e.metadata } };
|
|
508
|
+
}
|
|
509
|
+
export function listEvoEvolutionsV2() {
|
|
510
|
+
return [..._emEvos.values()].map((e) => ({
|
|
511
|
+
...e,
|
|
512
|
+
metadata: { ...e.metadata },
|
|
513
|
+
}));
|
|
514
|
+
}
|
|
515
|
+
|
|
516
|
+
export function autoStaleIdleEvoMapsV2({ now } = {}) {
|
|
517
|
+
const t = now ?? Date.now();
|
|
518
|
+
const flipped = [];
|
|
519
|
+
for (const m of _emMaps.values())
|
|
520
|
+
if (
|
|
521
|
+
m.status === EVOMAP_MAP_MATURITY_V2.ACTIVE &&
|
|
522
|
+
t - m.lastTouchedAt >= _emMapIdleMs
|
|
523
|
+
) {
|
|
524
|
+
m.status = EVOMAP_MAP_MATURITY_V2.STALE;
|
|
525
|
+
m.updatedAt = t;
|
|
526
|
+
flipped.push(m.id);
|
|
527
|
+
}
|
|
528
|
+
return { flipped, count: flipped.length };
|
|
529
|
+
}
|
|
530
|
+
export function autoFailStuckEvoEvolutionsV2({ now } = {}) {
|
|
531
|
+
const t = now ?? Date.now();
|
|
532
|
+
const flipped = [];
|
|
533
|
+
for (const e of _emEvos.values())
|
|
534
|
+
if (
|
|
535
|
+
e.status === EVOMAP_EVOLUTION_LIFECYCLE_V2.RUNNING &&
|
|
536
|
+
e.startedAt != null &&
|
|
537
|
+
t - e.startedAt >= _emEvoStuckMs
|
|
538
|
+
) {
|
|
539
|
+
e.status = EVOMAP_EVOLUTION_LIFECYCLE_V2.FAILED;
|
|
540
|
+
e.updatedAt = t;
|
|
541
|
+
if (!e.settledAt) e.settledAt = t;
|
|
542
|
+
e.metadata.failReason = "auto-fail-stuck";
|
|
543
|
+
flipped.push(e.id);
|
|
544
|
+
}
|
|
545
|
+
return { flipped, count: flipped.length };
|
|
546
|
+
}
|
|
547
|
+
|
|
548
|
+
export function getEvoMapManagerStatsV2() {
|
|
549
|
+
const mapsByStatus = {};
|
|
550
|
+
for (const s of Object.values(EVOMAP_MAP_MATURITY_V2)) mapsByStatus[s] = 0;
|
|
551
|
+
for (const m of _emMaps.values()) mapsByStatus[m.status]++;
|
|
552
|
+
const evosByStatus = {};
|
|
553
|
+
for (const s of Object.values(EVOMAP_EVOLUTION_LIFECYCLE_V2))
|
|
554
|
+
evosByStatus[s] = 0;
|
|
555
|
+
for (const e of _emEvos.values()) evosByStatus[e.status]++;
|
|
556
|
+
return {
|
|
557
|
+
totalMapsV2: _emMaps.size,
|
|
558
|
+
totalEvolutionsV2: _emEvos.size,
|
|
559
|
+
maxActiveEvoMapsPerOwner: _emMaxActivePerOwner,
|
|
560
|
+
maxPendingEvoEvolutionsPerMap: _emMaxPendingPerMap,
|
|
561
|
+
evoMapIdleMs: _emMapIdleMs,
|
|
562
|
+
evoEvolutionStuckMs: _emEvoStuckMs,
|
|
563
|
+
mapsByStatus,
|
|
564
|
+
evosByStatus,
|
|
565
|
+
};
|
|
566
|
+
}
|
|
@@ -237,3 +237,354 @@ export function createBackend(config = {}) {
|
|
|
237
237
|
return new LocalBackend(config.options || {});
|
|
238
238
|
}
|
|
239
239
|
}
|
|
240
|
+
|
|
241
|
+
// ===== V2 Surface: Execution Backend governance overlay (CLI v0.133.0) =====
|
|
242
|
+
export const EXECBE_BACKEND_MATURITY_V2 = Object.freeze({
|
|
243
|
+
PENDING: "pending",
|
|
244
|
+
ACTIVE: "active",
|
|
245
|
+
DEGRADED: "degraded",
|
|
246
|
+
RETIRED: "retired",
|
|
247
|
+
});
|
|
248
|
+
export const EXECBE_JOB_LIFECYCLE_V2 = Object.freeze({
|
|
249
|
+
QUEUED: "queued",
|
|
250
|
+
RUNNING: "running",
|
|
251
|
+
SUCCEEDED: "succeeded",
|
|
252
|
+
FAILED: "failed",
|
|
253
|
+
CANCELLED: "cancelled",
|
|
254
|
+
});
|
|
255
|
+
|
|
256
|
+
const _ebBackendTrans = new Map([
|
|
257
|
+
[
|
|
258
|
+
EXECBE_BACKEND_MATURITY_V2.PENDING,
|
|
259
|
+
new Set([
|
|
260
|
+
EXECBE_BACKEND_MATURITY_V2.ACTIVE,
|
|
261
|
+
EXECBE_BACKEND_MATURITY_V2.RETIRED,
|
|
262
|
+
]),
|
|
263
|
+
],
|
|
264
|
+
[
|
|
265
|
+
EXECBE_BACKEND_MATURITY_V2.ACTIVE,
|
|
266
|
+
new Set([
|
|
267
|
+
EXECBE_BACKEND_MATURITY_V2.DEGRADED,
|
|
268
|
+
EXECBE_BACKEND_MATURITY_V2.RETIRED,
|
|
269
|
+
]),
|
|
270
|
+
],
|
|
271
|
+
[
|
|
272
|
+
EXECBE_BACKEND_MATURITY_V2.DEGRADED,
|
|
273
|
+
new Set([
|
|
274
|
+
EXECBE_BACKEND_MATURITY_V2.ACTIVE,
|
|
275
|
+
EXECBE_BACKEND_MATURITY_V2.RETIRED,
|
|
276
|
+
]),
|
|
277
|
+
],
|
|
278
|
+
[EXECBE_BACKEND_MATURITY_V2.RETIRED, new Set()],
|
|
279
|
+
]);
|
|
280
|
+
const _ebBackendTerminal = new Set([EXECBE_BACKEND_MATURITY_V2.RETIRED]);
|
|
281
|
+
const _ebJobTrans = new Map([
|
|
282
|
+
[
|
|
283
|
+
EXECBE_JOB_LIFECYCLE_V2.QUEUED,
|
|
284
|
+
new Set([
|
|
285
|
+
EXECBE_JOB_LIFECYCLE_V2.RUNNING,
|
|
286
|
+
EXECBE_JOB_LIFECYCLE_V2.CANCELLED,
|
|
287
|
+
]),
|
|
288
|
+
],
|
|
289
|
+
[
|
|
290
|
+
EXECBE_JOB_LIFECYCLE_V2.RUNNING,
|
|
291
|
+
new Set([
|
|
292
|
+
EXECBE_JOB_LIFECYCLE_V2.SUCCEEDED,
|
|
293
|
+
EXECBE_JOB_LIFECYCLE_V2.FAILED,
|
|
294
|
+
EXECBE_JOB_LIFECYCLE_V2.CANCELLED,
|
|
295
|
+
]),
|
|
296
|
+
],
|
|
297
|
+
[EXECBE_JOB_LIFECYCLE_V2.SUCCEEDED, new Set()],
|
|
298
|
+
[EXECBE_JOB_LIFECYCLE_V2.FAILED, new Set()],
|
|
299
|
+
[EXECBE_JOB_LIFECYCLE_V2.CANCELLED, new Set()],
|
|
300
|
+
]);
|
|
301
|
+
|
|
302
|
+
const _ebBackends = new Map();
|
|
303
|
+
const _ebJobs = new Map();
|
|
304
|
+
let _ebMaxActivePerOwner = 6;
|
|
305
|
+
let _ebMaxPendingPerBackend = 20;
|
|
306
|
+
let _ebBackendIdleMs = 12 * 60 * 60 * 1000;
|
|
307
|
+
let _ebJobStuckMs = 10 * 60 * 1000;
|
|
308
|
+
|
|
309
|
+
function _ebPos(n, lbl) {
|
|
310
|
+
const v = Math.floor(Number(n));
|
|
311
|
+
if (!Number.isFinite(v) || v <= 0)
|
|
312
|
+
throw new Error(`${lbl} must be positive integer`);
|
|
313
|
+
return v;
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
export function setMaxActiveBackendsPerOwnerV2(n) {
|
|
317
|
+
_ebMaxActivePerOwner = _ebPos(n, "maxActiveBackendsPerOwner");
|
|
318
|
+
}
|
|
319
|
+
export function getMaxActiveBackendsPerOwnerV2() {
|
|
320
|
+
return _ebMaxActivePerOwner;
|
|
321
|
+
}
|
|
322
|
+
export function setMaxPendingJobsPerBackendV2(n) {
|
|
323
|
+
_ebMaxPendingPerBackend = _ebPos(n, "maxPendingJobsPerBackend");
|
|
324
|
+
}
|
|
325
|
+
export function getMaxPendingJobsPerBackendV2() {
|
|
326
|
+
return _ebMaxPendingPerBackend;
|
|
327
|
+
}
|
|
328
|
+
export function setBackendIdleMsV2(n) {
|
|
329
|
+
_ebBackendIdleMs = _ebPos(n, "backendIdleMs");
|
|
330
|
+
}
|
|
331
|
+
export function getBackendIdleMsV2() {
|
|
332
|
+
return _ebBackendIdleMs;
|
|
333
|
+
}
|
|
334
|
+
export function setExecJobStuckMsV2(n) {
|
|
335
|
+
_ebJobStuckMs = _ebPos(n, "execJobStuckMs");
|
|
336
|
+
}
|
|
337
|
+
export function getExecJobStuckMsV2() {
|
|
338
|
+
return _ebJobStuckMs;
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
export function _resetStateExecutionBackendV2() {
|
|
342
|
+
_ebBackends.clear();
|
|
343
|
+
_ebJobs.clear();
|
|
344
|
+
_ebMaxActivePerOwner = 6;
|
|
345
|
+
_ebMaxPendingPerBackend = 20;
|
|
346
|
+
_ebBackendIdleMs = 12 * 60 * 60 * 1000;
|
|
347
|
+
_ebJobStuckMs = 10 * 60 * 1000;
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
export function registerBackendV2({ id, owner, kind, metadata } = {}) {
|
|
351
|
+
if (!id || typeof id !== "string") throw new Error("id is required");
|
|
352
|
+
if (!owner || typeof owner !== "string") throw new Error("owner is required");
|
|
353
|
+
if (_ebBackends.has(id)) throw new Error(`backend ${id} already registered`);
|
|
354
|
+
const now = Date.now();
|
|
355
|
+
const b = {
|
|
356
|
+
id,
|
|
357
|
+
owner,
|
|
358
|
+
kind: kind || "local",
|
|
359
|
+
status: EXECBE_BACKEND_MATURITY_V2.PENDING,
|
|
360
|
+
createdAt: now,
|
|
361
|
+
updatedAt: now,
|
|
362
|
+
activatedAt: null,
|
|
363
|
+
retiredAt: null,
|
|
364
|
+
lastTouchedAt: now,
|
|
365
|
+
metadata: { ...(metadata || {}) },
|
|
366
|
+
};
|
|
367
|
+
_ebBackends.set(id, b);
|
|
368
|
+
return { ...b, metadata: { ...b.metadata } };
|
|
369
|
+
}
|
|
370
|
+
function _ebCheckB(from, to) {
|
|
371
|
+
const allowed = _ebBackendTrans.get(from);
|
|
372
|
+
if (!allowed || !allowed.has(to))
|
|
373
|
+
throw new Error(`invalid backend transition ${from} → ${to}`);
|
|
374
|
+
}
|
|
375
|
+
function _ebCountActive(owner) {
|
|
376
|
+
let n = 0;
|
|
377
|
+
for (const b of _ebBackends.values())
|
|
378
|
+
if (b.owner === owner && b.status === EXECBE_BACKEND_MATURITY_V2.ACTIVE)
|
|
379
|
+
n++;
|
|
380
|
+
return n;
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
export function activateBackendV2(id) {
|
|
384
|
+
const b = _ebBackends.get(id);
|
|
385
|
+
if (!b) throw new Error(`backend ${id} not found`);
|
|
386
|
+
_ebCheckB(b.status, EXECBE_BACKEND_MATURITY_V2.ACTIVE);
|
|
387
|
+
const recovery = b.status === EXECBE_BACKEND_MATURITY_V2.DEGRADED;
|
|
388
|
+
if (!recovery) {
|
|
389
|
+
const a = _ebCountActive(b.owner);
|
|
390
|
+
if (a >= _ebMaxActivePerOwner)
|
|
391
|
+
throw new Error(
|
|
392
|
+
`max active backends per owner (${_ebMaxActivePerOwner}) reached for ${b.owner}`,
|
|
393
|
+
);
|
|
394
|
+
}
|
|
395
|
+
const now = Date.now();
|
|
396
|
+
b.status = EXECBE_BACKEND_MATURITY_V2.ACTIVE;
|
|
397
|
+
b.updatedAt = now;
|
|
398
|
+
b.lastTouchedAt = now;
|
|
399
|
+
if (!b.activatedAt) b.activatedAt = now;
|
|
400
|
+
return { ...b, metadata: { ...b.metadata } };
|
|
401
|
+
}
|
|
402
|
+
export function degradeBackendV2(id) {
|
|
403
|
+
const b = _ebBackends.get(id);
|
|
404
|
+
if (!b) throw new Error(`backend ${id} not found`);
|
|
405
|
+
_ebCheckB(b.status, EXECBE_BACKEND_MATURITY_V2.DEGRADED);
|
|
406
|
+
b.status = EXECBE_BACKEND_MATURITY_V2.DEGRADED;
|
|
407
|
+
b.updatedAt = Date.now();
|
|
408
|
+
return { ...b, metadata: { ...b.metadata } };
|
|
409
|
+
}
|
|
410
|
+
export function retireBackendV2(id) {
|
|
411
|
+
const b = _ebBackends.get(id);
|
|
412
|
+
if (!b) throw new Error(`backend ${id} not found`);
|
|
413
|
+
_ebCheckB(b.status, EXECBE_BACKEND_MATURITY_V2.RETIRED);
|
|
414
|
+
const now = Date.now();
|
|
415
|
+
b.status = EXECBE_BACKEND_MATURITY_V2.RETIRED;
|
|
416
|
+
b.updatedAt = now;
|
|
417
|
+
if (!b.retiredAt) b.retiredAt = now;
|
|
418
|
+
return { ...b, metadata: { ...b.metadata } };
|
|
419
|
+
}
|
|
420
|
+
export function touchBackendV2(id) {
|
|
421
|
+
const b = _ebBackends.get(id);
|
|
422
|
+
if (!b) throw new Error(`backend ${id} not found`);
|
|
423
|
+
if (_ebBackendTerminal.has(b.status))
|
|
424
|
+
throw new Error(`cannot touch terminal backend ${id}`);
|
|
425
|
+
const now = Date.now();
|
|
426
|
+
b.lastTouchedAt = now;
|
|
427
|
+
b.updatedAt = now;
|
|
428
|
+
return { ...b, metadata: { ...b.metadata } };
|
|
429
|
+
}
|
|
430
|
+
export function getBackendV2(id) {
|
|
431
|
+
const b = _ebBackends.get(id);
|
|
432
|
+
if (!b) return null;
|
|
433
|
+
return { ...b, metadata: { ...b.metadata } };
|
|
434
|
+
}
|
|
435
|
+
export function listBackendsV2() {
|
|
436
|
+
return [..._ebBackends.values()].map((b) => ({
|
|
437
|
+
...b,
|
|
438
|
+
metadata: { ...b.metadata },
|
|
439
|
+
}));
|
|
440
|
+
}
|
|
441
|
+
|
|
442
|
+
function _ebCountPending(bid) {
|
|
443
|
+
let n = 0;
|
|
444
|
+
for (const j of _ebJobs.values())
|
|
445
|
+
if (
|
|
446
|
+
j.backendId === bid &&
|
|
447
|
+
(j.status === EXECBE_JOB_LIFECYCLE_V2.QUEUED ||
|
|
448
|
+
j.status === EXECBE_JOB_LIFECYCLE_V2.RUNNING)
|
|
449
|
+
)
|
|
450
|
+
n++;
|
|
451
|
+
return n;
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
export function createExecJobV2({ id, backendId, command, metadata } = {}) {
|
|
455
|
+
if (!id || typeof id !== "string") throw new Error("id is required");
|
|
456
|
+
if (!backendId || typeof backendId !== "string")
|
|
457
|
+
throw new Error("backendId is required");
|
|
458
|
+
if (_ebJobs.has(id)) throw new Error(`exec job ${id} already exists`);
|
|
459
|
+
if (!_ebBackends.has(backendId))
|
|
460
|
+
throw new Error(`backend ${backendId} not found`);
|
|
461
|
+
const pending = _ebCountPending(backendId);
|
|
462
|
+
if (pending >= _ebMaxPendingPerBackend)
|
|
463
|
+
throw new Error(
|
|
464
|
+
`max pending jobs per backend (${_ebMaxPendingPerBackend}) reached for ${backendId}`,
|
|
465
|
+
);
|
|
466
|
+
const now = Date.now();
|
|
467
|
+
const j = {
|
|
468
|
+
id,
|
|
469
|
+
backendId,
|
|
470
|
+
command: command || "",
|
|
471
|
+
status: EXECBE_JOB_LIFECYCLE_V2.QUEUED,
|
|
472
|
+
createdAt: now,
|
|
473
|
+
updatedAt: now,
|
|
474
|
+
startedAt: null,
|
|
475
|
+
settledAt: null,
|
|
476
|
+
metadata: { ...(metadata || {}) },
|
|
477
|
+
};
|
|
478
|
+
_ebJobs.set(id, j);
|
|
479
|
+
return { ...j, metadata: { ...j.metadata } };
|
|
480
|
+
}
|
|
481
|
+
function _ebCheckJ(from, to) {
|
|
482
|
+
const allowed = _ebJobTrans.get(from);
|
|
483
|
+
if (!allowed || !allowed.has(to))
|
|
484
|
+
throw new Error(`invalid exec job transition ${from} → ${to}`);
|
|
485
|
+
}
|
|
486
|
+
export function startExecJobV2(id) {
|
|
487
|
+
const j = _ebJobs.get(id);
|
|
488
|
+
if (!j) throw new Error(`exec job ${id} not found`);
|
|
489
|
+
_ebCheckJ(j.status, EXECBE_JOB_LIFECYCLE_V2.RUNNING);
|
|
490
|
+
const now = Date.now();
|
|
491
|
+
j.status = EXECBE_JOB_LIFECYCLE_V2.RUNNING;
|
|
492
|
+
j.updatedAt = now;
|
|
493
|
+
if (!j.startedAt) j.startedAt = now;
|
|
494
|
+
return { ...j, metadata: { ...j.metadata } };
|
|
495
|
+
}
|
|
496
|
+
export function succeedExecJobV2(id) {
|
|
497
|
+
const j = _ebJobs.get(id);
|
|
498
|
+
if (!j) throw new Error(`exec job ${id} not found`);
|
|
499
|
+
_ebCheckJ(j.status, EXECBE_JOB_LIFECYCLE_V2.SUCCEEDED);
|
|
500
|
+
const now = Date.now();
|
|
501
|
+
j.status = EXECBE_JOB_LIFECYCLE_V2.SUCCEEDED;
|
|
502
|
+
j.updatedAt = now;
|
|
503
|
+
if (!j.settledAt) j.settledAt = now;
|
|
504
|
+
return { ...j, metadata: { ...j.metadata } };
|
|
505
|
+
}
|
|
506
|
+
export function failExecJobV2(id, reason) {
|
|
507
|
+
const j = _ebJobs.get(id);
|
|
508
|
+
if (!j) throw new Error(`exec job ${id} not found`);
|
|
509
|
+
_ebCheckJ(j.status, EXECBE_JOB_LIFECYCLE_V2.FAILED);
|
|
510
|
+
const now = Date.now();
|
|
511
|
+
j.status = EXECBE_JOB_LIFECYCLE_V2.FAILED;
|
|
512
|
+
j.updatedAt = now;
|
|
513
|
+
if (!j.settledAt) j.settledAt = now;
|
|
514
|
+
if (reason) j.metadata.failReason = String(reason);
|
|
515
|
+
return { ...j, metadata: { ...j.metadata } };
|
|
516
|
+
}
|
|
517
|
+
export function cancelExecJobV2(id, reason) {
|
|
518
|
+
const j = _ebJobs.get(id);
|
|
519
|
+
if (!j) throw new Error(`exec job ${id} not found`);
|
|
520
|
+
_ebCheckJ(j.status, EXECBE_JOB_LIFECYCLE_V2.CANCELLED);
|
|
521
|
+
const now = Date.now();
|
|
522
|
+
j.status = EXECBE_JOB_LIFECYCLE_V2.CANCELLED;
|
|
523
|
+
j.updatedAt = now;
|
|
524
|
+
if (!j.settledAt) j.settledAt = now;
|
|
525
|
+
if (reason) j.metadata.cancelReason = String(reason);
|
|
526
|
+
return { ...j, metadata: { ...j.metadata } };
|
|
527
|
+
}
|
|
528
|
+
export function getExecJobV2(id) {
|
|
529
|
+
const j = _ebJobs.get(id);
|
|
530
|
+
if (!j) return null;
|
|
531
|
+
return { ...j, metadata: { ...j.metadata } };
|
|
532
|
+
}
|
|
533
|
+
export function listExecJobsV2() {
|
|
534
|
+
return [..._ebJobs.values()].map((j) => ({
|
|
535
|
+
...j,
|
|
536
|
+
metadata: { ...j.metadata },
|
|
537
|
+
}));
|
|
538
|
+
}
|
|
539
|
+
|
|
540
|
+
export function autoDegradeIdleBackendsV2({ now } = {}) {
|
|
541
|
+
const t = now ?? Date.now();
|
|
542
|
+
const flipped = [];
|
|
543
|
+
for (const b of _ebBackends.values())
|
|
544
|
+
if (
|
|
545
|
+
b.status === EXECBE_BACKEND_MATURITY_V2.ACTIVE &&
|
|
546
|
+
t - b.lastTouchedAt >= _ebBackendIdleMs
|
|
547
|
+
) {
|
|
548
|
+
b.status = EXECBE_BACKEND_MATURITY_V2.DEGRADED;
|
|
549
|
+
b.updatedAt = t;
|
|
550
|
+
flipped.push(b.id);
|
|
551
|
+
}
|
|
552
|
+
return { flipped, count: flipped.length };
|
|
553
|
+
}
|
|
554
|
+
export function autoFailStuckExecJobsV2({ now } = {}) {
|
|
555
|
+
const t = now ?? Date.now();
|
|
556
|
+
const flipped = [];
|
|
557
|
+
for (const j of _ebJobs.values())
|
|
558
|
+
if (
|
|
559
|
+
j.status === EXECBE_JOB_LIFECYCLE_V2.RUNNING &&
|
|
560
|
+
j.startedAt != null &&
|
|
561
|
+
t - j.startedAt >= _ebJobStuckMs
|
|
562
|
+
) {
|
|
563
|
+
j.status = EXECBE_JOB_LIFECYCLE_V2.FAILED;
|
|
564
|
+
j.updatedAt = t;
|
|
565
|
+
if (!j.settledAt) j.settledAt = t;
|
|
566
|
+
j.metadata.failReason = "auto-fail-stuck";
|
|
567
|
+
flipped.push(j.id);
|
|
568
|
+
}
|
|
569
|
+
return { flipped, count: flipped.length };
|
|
570
|
+
}
|
|
571
|
+
|
|
572
|
+
export function getExecutionBackendStatsV2() {
|
|
573
|
+
const backendsByStatus = {};
|
|
574
|
+
for (const s of Object.values(EXECBE_BACKEND_MATURITY_V2))
|
|
575
|
+
backendsByStatus[s] = 0;
|
|
576
|
+
for (const b of _ebBackends.values()) backendsByStatus[b.status]++;
|
|
577
|
+
const jobsByStatus = {};
|
|
578
|
+
for (const s of Object.values(EXECBE_JOB_LIFECYCLE_V2)) jobsByStatus[s] = 0;
|
|
579
|
+
for (const j of _ebJobs.values()) jobsByStatus[j.status]++;
|
|
580
|
+
return {
|
|
581
|
+
totalBackendsV2: _ebBackends.size,
|
|
582
|
+
totalJobsV2: _ebJobs.size,
|
|
583
|
+
maxActiveBackendsPerOwner: _ebMaxActivePerOwner,
|
|
584
|
+
maxPendingJobsPerBackend: _ebMaxPendingPerBackend,
|
|
585
|
+
backendIdleMs: _ebBackendIdleMs,
|
|
586
|
+
execJobStuckMs: _ebJobStuckMs,
|
|
587
|
+
backendsByStatus,
|
|
588
|
+
jobsByStatus,
|
|
589
|
+
};
|
|
590
|
+
}
|