@remnic/core 9.3.653 → 9.3.655

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 (273) hide show
  1. package/dist/access-cli.js +24 -24
  2. package/dist/access-http.d.ts +4 -4
  3. package/dist/access-http.js +17 -17
  4. package/dist/access-mcp.d.ts +4 -4
  5. package/dist/access-mcp.js +16 -16
  6. package/dist/access-schema.d.ts +12 -12
  7. package/dist/{access-service-CdJFd3_b.d.ts → access-service-BEJvriUt.d.ts} +11 -2
  8. package/dist/access-service.d.ts +4 -4
  9. package/dist/access-service.js +15 -15
  10. package/dist/action-confidence.d.ts +1 -1
  11. package/dist/active-memory-bridge.d.ts +1 -1
  12. package/dist/active-recall.d.ts +1 -1
  13. package/dist/active-recall.js +1 -1
  14. package/dist/behavior-learner.d.ts +1 -1
  15. package/dist/behavior-signals.d.ts +1 -1
  16. package/dist/bootstrap.d.ts +3 -3
  17. package/dist/briefing.d.ts +1 -1
  18. package/dist/briefing.js +3 -3
  19. package/dist/buffer-surprise-report.d.ts +1 -1
  20. package/dist/buffer.d.ts +1 -1
  21. package/dist/calibration.d.ts +1 -1
  22. package/dist/causal-behavior.d.ts +1 -1
  23. package/dist/causal-consolidation.d.ts +1 -1
  24. package/dist/causal-consolidation.js +4 -4
  25. package/dist/{chunk-GI45G4BK.js → chunk-2RCGZ67B.js} +4 -4
  26. package/dist/{chunk-BEMWL2FZ.js → chunk-54LOUIBE.js} +2 -2
  27. package/dist/{chunk-E3J6O6N7.js → chunk-55ZMNKMQ.js} +20 -9
  28. package/dist/{chunk-E3J6O6N7.js.map → chunk-55ZMNKMQ.js.map} +1 -1
  29. package/dist/{chunk-7WEB3FLJ.js → chunk-5PLUC5OB.js} +2 -2
  30. package/dist/{chunk-SPMZZUEJ.js → chunk-5QD3QD76.js} +2684 -401
  31. package/dist/chunk-5QD3QD76.js.map +1 -0
  32. package/dist/{chunk-WLGE6KEO.js → chunk-67G4T7KI.js} +3 -3
  33. package/dist/{chunk-JX2RINDR.js → chunk-6G5JEN55.js} +2 -2
  34. package/dist/{chunk-R3PQUPQ4.js → chunk-6IMKOIZ6.js} +85 -3
  35. package/dist/chunk-6IMKOIZ6.js.map +1 -0
  36. package/dist/{chunk-KJDKZVF3.js → chunk-A3Y37UWI.js} +3 -3
  37. package/dist/{chunk-CFOCZPIQ.js → chunk-BGKXTVNG.js} +2 -2
  38. package/dist/{chunk-QQHIQ7JD.js → chunk-COVZLGMR.js} +87 -18
  39. package/dist/chunk-COVZLGMR.js.map +1 -0
  40. package/dist/{chunk-JVRPJ7D4.js → chunk-EKQMQQ3U.js} +48 -12
  41. package/dist/chunk-EKQMQQ3U.js.map +1 -0
  42. package/dist/{chunk-H3PHZLMF.js → chunk-GKKAXVAJ.js} +20 -11
  43. package/dist/chunk-GKKAXVAJ.js.map +1 -0
  44. package/dist/{chunk-JBHXMCYN.js → chunk-GRYAECRV.js} +2 -2
  45. package/dist/{chunk-EHQLDFSH.js → chunk-IQ53ZSXV.js} +2 -2
  46. package/dist/{chunk-C63WC454.js → chunk-KOI765XP.js} +125 -1
  47. package/dist/chunk-KOI765XP.js.map +1 -0
  48. package/dist/{chunk-IVYSVAC6.js → chunk-KZZ4YAEC.js} +2 -2
  49. package/dist/{chunk-2DGQLOOM.js → chunk-M3VYPE2H.js} +1 -1
  50. package/dist/{chunk-2DGQLOOM.js.map → chunk-M3VYPE2H.js.map} +1 -1
  51. package/dist/{chunk-JF7SFXTG.js → chunk-NCSJKK23.js} +2 -2
  52. package/dist/{chunk-XMN6MMTU.js → chunk-NRBGRZW4.js} +2 -2
  53. package/dist/{chunk-NOBL7OUP.js → chunk-OKW6F5S5.js} +12 -5
  54. package/dist/{chunk-NOBL7OUP.js.map → chunk-OKW6F5S5.js.map} +1 -1
  55. package/dist/{chunk-BNFRL6QW.js → chunk-PTMJ2FH2.js} +2 -2
  56. package/dist/{chunk-KWM33SPU.js → chunk-PVE7KSQP.js} +2 -2
  57. package/dist/{chunk-EW52H5EM.js → chunk-QDVQ4AN2.js} +12 -5
  58. package/dist/chunk-QDVQ4AN2.js.map +1 -0
  59. package/dist/{chunk-PYWNNF2I.js → chunk-QRSKPI62.js} +99 -66
  60. package/dist/chunk-QRSKPI62.js.map +1 -0
  61. package/dist/{chunk-YM3LR4LS.js → chunk-SSSXWIBP.js} +5 -5
  62. package/dist/{chunk-C43KEWEV.js → chunk-TDZSSJV4.js} +1 -1
  63. package/dist/chunk-TDZSSJV4.js.map +1 -0
  64. package/dist/{chunk-Y7NWBBHV.js → chunk-TEO46GMM.js} +2 -2
  65. package/dist/{chunk-AJE7FJVE.js → chunk-UCEABZZN.js} +2 -2
  66. package/dist/{chunk-IENGGY2C.js → chunk-UCEDY5M7.js} +2 -2
  67. package/dist/{chunk-PRQXUSQV.js → chunk-UYNFWZWG.js} +2 -2
  68. package/dist/{chunk-V4UDXYGG.js → chunk-WDTUYOLS.js} +2 -2
  69. package/dist/{chunk-RZOBQ23O.js → chunk-XOFXKASO.js} +2 -2
  70. package/dist/chunk-XRKQOQLY.js +212 -0
  71. package/dist/chunk-XRKQOQLY.js.map +1 -0
  72. package/dist/{chunk-WTI35CVJ.js → chunk-YYN3LIYA.js} +5 -5
  73. package/dist/{cli-DDo7Qgs-.d.ts → cli-BGahB_d3.d.ts} +3 -3
  74. package/dist/cli.d.ts +5 -5
  75. package/dist/cli.js +29 -29
  76. package/dist/compounding/engine.d.ts +1 -1
  77. package/dist/compounding/engine.js +3 -3
  78. package/dist/compounding/preference-consolidator.d.ts +1 -1
  79. package/dist/compression-optimizer.d.ts +1 -1
  80. package/dist/config.d.ts +1 -1
  81. package/dist/config.js +1 -1
  82. package/dist/connectors/codex-materialize-runner.d.ts +1 -1
  83. package/dist/connectors/codex-materialize-runner.js +3 -3
  84. package/dist/connectors/codex-materialize.d.ts +1 -1
  85. package/dist/connectors/index.d.ts +1 -1
  86. package/dist/connectors/index.js +3 -3
  87. package/dist/consolidation-provenance-check.d.ts +1 -1
  88. package/dist/consolidation-undo.d.ts +1 -1
  89. package/dist/contradiction/index.d.ts +19 -1
  90. package/dist/contradiction/index.js +1 -1
  91. package/dist/conversation-index/backend.d.ts +1 -1
  92. package/dist/conversation-index/chunker.d.ts +1 -1
  93. package/dist/conversation-index/faiss-adapter.d.ts +1 -1
  94. package/dist/conversation-index/indexer.d.ts +1 -1
  95. package/dist/conversation-index/search.d.ts +1 -1
  96. package/dist/day-summary.d.ts +1 -1
  97. package/dist/delinearize.d.ts +1 -1
  98. package/dist/direct-answer-wiring.d.ts +1 -1
  99. package/dist/direct-answer.d.ts +1 -1
  100. package/dist/embedding-fallback.d.ts +1 -1
  101. package/dist/enrichment/index.d.ts +1 -1
  102. package/dist/entity-retrieval.d.ts +1 -1
  103. package/dist/entity-retrieval.js +3 -3
  104. package/dist/entity-schema.d.ts +1 -1
  105. package/dist/explicit-capture.d.ts +3 -3
  106. package/dist/explicit-capture.js +1 -1
  107. package/dist/extraction-judge-telemetry.d.ts +1 -1
  108. package/dist/extraction-judge-training.d.ts +1 -1
  109. package/dist/extraction-judge.d.ts +1 -1
  110. package/dist/extraction.d.ts +1 -1
  111. package/dist/fallback-llm.d.ts +1 -1
  112. package/dist/identity-continuity.d.ts +1 -1
  113. package/dist/importance.d.ts +1 -1
  114. package/dist/index.d.ts +8 -8
  115. package/dist/index.js +37 -35
  116. package/dist/index.js.map +1 -1
  117. package/dist/intent.d.ts +1 -1
  118. package/dist/lcm/engine.d.ts +1 -1
  119. package/dist/lcm/index.d.ts +1 -1
  120. package/dist/lcm/tools.d.ts +1 -1
  121. package/dist/lifecycle.d.ts +1 -1
  122. package/dist/live-connectors-runner.d.ts +1 -1
  123. package/dist/local-llm.d.ts +1 -1
  124. package/dist/maintenance/memory-governance.d.ts +1 -1
  125. package/dist/maintenance/memory-governance.js +3 -3
  126. package/dist/maintenance/rebuild-memory-lifecycle-ledger.js +3 -3
  127. package/dist/maintenance/rebuild-memory-projection.js +4 -4
  128. package/dist/mcp-memory-inspector-app.d.ts +4 -4
  129. package/dist/memory-action-policy.d.ts +1 -1
  130. package/dist/memory-cache.d.ts +1 -1
  131. package/dist/memory-lifecycle-ledger-utils.d.ts +1 -1
  132. package/dist/memory-projection-store.d.ts +1 -1
  133. package/dist/memory-provenance.d.ts +1 -1
  134. package/dist/memory-worth-outcomes.d.ts +1 -1
  135. package/dist/models-json.d.ts +1 -1
  136. package/dist/namespaces/migrate.d.ts +1 -1
  137. package/dist/namespaces/migrate.js +11 -11
  138. package/dist/namespaces/principal.d.ts +1 -1
  139. package/dist/namespaces/search.d.ts +15 -4
  140. package/dist/namespaces/search.js +7 -7
  141. package/dist/namespaces/storage.d.ts +52 -3
  142. package/dist/namespaces/storage.js +9 -5
  143. package/dist/native-knowledge.d.ts +1 -1
  144. package/dist/operator-toolkit.d.ts +1 -1
  145. package/dist/operator-toolkit.js +14 -14
  146. package/dist/{orchestrator-8fTZsa0y.d.ts → orchestrator-BgzZlWxH.d.ts} +500 -3
  147. package/dist/orchestrator.d.ts +3 -3
  148. package/dist/orchestrator.js +20 -20
  149. package/dist/patterns-cli.d.ts +1 -1
  150. package/dist/policy-runtime.d.ts +1 -1
  151. package/dist/qmd-recall-cache.d.ts +1 -1
  152. package/dist/qmd.d.ts +5 -1
  153. package/dist/qmd.js +2 -2
  154. package/dist/recall-disclosure-escalation.d.ts +1 -1
  155. package/dist/recall-explain-renderer.d.ts +1 -1
  156. package/dist/recall-explain-renderer.js +3 -3
  157. package/dist/recall-planner-llm.d.ts +1 -1
  158. package/dist/recall-state.d.ts +1 -1
  159. package/dist/recall-tag-filter.d.ts +1 -1
  160. package/dist/recall-xray-cli.d.ts +1 -1
  161. package/dist/recall-xray-cli.js +4 -4
  162. package/dist/recall-xray-renderer.d.ts +1 -1
  163. package/dist/recall-xray-renderer.js +3 -3
  164. package/dist/recall-xray.d.ts +1 -1
  165. package/dist/recall-xray.js +2 -2
  166. package/dist/{resolution-3SAP4SH2.js → resolution-IDTEBJFS.js} +2 -2
  167. package/dist/resolve-auth-token.d.ts +1 -1
  168. package/dist/resume-bundles.js +2 -2
  169. package/dist/retrieval-agents.d.ts +1 -1
  170. package/dist/retrieval-tiers.d.ts +1 -1
  171. package/dist/routing/engine.d.ts +1 -1
  172. package/dist/routing/store.d.ts +1 -1
  173. package/dist/schemas.d.ts +22 -22
  174. package/dist/search/embed-helper.d.ts +1 -1
  175. package/dist/search/factory.d.ts +1 -1
  176. package/dist/search/factory.js +6 -6
  177. package/dist/search/index.d.ts +1 -1
  178. package/dist/search/index.js +6 -6
  179. package/dist/search/lancedb-backend.d.ts +1 -1
  180. package/dist/search/lancedb-backend.js +2 -2
  181. package/dist/search/meilisearch-backend.d.ts +1 -1
  182. package/dist/search/meilisearch-backend.js +2 -2
  183. package/dist/search/noop-backend.d.ts +1 -1
  184. package/dist/search/orama-backend.d.ts +1 -1
  185. package/dist/search/orama-backend.js +2 -2
  186. package/dist/search/port.d.ts +17 -1
  187. package/dist/search/port.js +1 -1
  188. package/dist/search/remote-backend.d.ts +1 -1
  189. package/dist/{semantic-consolidation-DKdYzQOg.d.ts → semantic-consolidation-Z8d_uMq8.d.ts} +1 -1
  190. package/dist/semantic-consolidation.d.ts +2 -2
  191. package/dist/semantic-consolidation.js +4 -4
  192. package/dist/semantic-rule-promotion.js +3 -3
  193. package/dist/semantic-rule-verifier.d.ts +1 -1
  194. package/dist/semantic-rule-verifier.js +3 -3
  195. package/dist/session-observer-bands.d.ts +1 -1
  196. package/dist/session-observer-state.d.ts +1 -1
  197. package/dist/shared-context/manager.d.ts +1 -1
  198. package/dist/signal.d.ts +1 -1
  199. package/dist/storage.d.ts +1 -1
  200. package/dist/storage.js +2 -2
  201. package/dist/summarizer.d.ts +1 -1
  202. package/dist/summary-snapshot.d.ts +1 -1
  203. package/dist/temporal-supersession.d.ts +1 -1
  204. package/dist/temporal-validity.d.ts +1 -1
  205. package/dist/threading.d.ts +1 -1
  206. package/dist/tier-migration.d.ts +1 -1
  207. package/dist/tier-routing.d.ts +1 -1
  208. package/dist/topics.d.ts +1 -1
  209. package/dist/transcript.d.ts +1 -1
  210. package/dist/transfer/types.d.ts +12 -12
  211. package/dist/{types-D8yUmSik.d.ts → types-2OPlQWJG.d.ts} +23 -0
  212. package/dist/types.d.ts +1 -1
  213. package/dist/types.js +1 -1
  214. package/dist/utility-runtime.d.ts +1 -1
  215. package/dist/verified-recall.js +3 -3
  216. package/package.json +1 -1
  217. package/src/access-http.ts +7 -0
  218. package/src/access-mcp.ts +7 -0
  219. package/src/access-service.ts +12 -0
  220. package/src/cli.ts +104 -0
  221. package/src/config.test.ts +109 -0
  222. package/src/config.ts +164 -0
  223. package/src/contradiction/contradiction.test.ts +284 -0
  224. package/src/contradiction/resolution.ts +151 -4
  225. package/src/explicit-capture.ts +31 -10
  226. package/src/index.ts +10 -0
  227. package/src/maintenance/namespace-planner.test.ts +1120 -0
  228. package/src/maintenance/namespace-planner.ts +893 -0
  229. package/src/namespaces/catalog.test.ts +3356 -0
  230. package/src/namespaces/catalog.ts +2123 -0
  231. package/src/namespaces/search.test.ts +130 -2
  232. package/src/namespaces/search.ts +71 -10
  233. package/src/namespaces/storage.ts +210 -30
  234. package/src/orchestrator-flush.test.ts +720 -0
  235. package/src/orchestrator.ts +881 -239
  236. package/src/qmd-client.test.ts +59 -0
  237. package/src/qmd.ts +124 -84
  238. package/src/search/port.ts +16 -0
  239. package/src/types.ts +23 -0
  240. package/dist/chunk-C43KEWEV.js.map +0 -1
  241. package/dist/chunk-C63WC454.js.map +0 -1
  242. package/dist/chunk-EW52H5EM.js.map +0 -1
  243. package/dist/chunk-H3PHZLMF.js.map +0 -1
  244. package/dist/chunk-JVRPJ7D4.js.map +0 -1
  245. package/dist/chunk-ORGWWNJG.js +0 -131
  246. package/dist/chunk-ORGWWNJG.js.map +0 -1
  247. package/dist/chunk-PYWNNF2I.js.map +0 -1
  248. package/dist/chunk-QQHIQ7JD.js.map +0 -1
  249. package/dist/chunk-R3PQUPQ4.js.map +0 -1
  250. package/dist/chunk-SPMZZUEJ.js.map +0 -1
  251. /package/dist/{chunk-GI45G4BK.js.map → chunk-2RCGZ67B.js.map} +0 -0
  252. /package/dist/{chunk-BEMWL2FZ.js.map → chunk-54LOUIBE.js.map} +0 -0
  253. /package/dist/{chunk-7WEB3FLJ.js.map → chunk-5PLUC5OB.js.map} +0 -0
  254. /package/dist/{chunk-WLGE6KEO.js.map → chunk-67G4T7KI.js.map} +0 -0
  255. /package/dist/{chunk-JX2RINDR.js.map → chunk-6G5JEN55.js.map} +0 -0
  256. /package/dist/{chunk-KJDKZVF3.js.map → chunk-A3Y37UWI.js.map} +0 -0
  257. /package/dist/{chunk-CFOCZPIQ.js.map → chunk-BGKXTVNG.js.map} +0 -0
  258. /package/dist/{chunk-JBHXMCYN.js.map → chunk-GRYAECRV.js.map} +0 -0
  259. /package/dist/{chunk-EHQLDFSH.js.map → chunk-IQ53ZSXV.js.map} +0 -0
  260. /package/dist/{chunk-IVYSVAC6.js.map → chunk-KZZ4YAEC.js.map} +0 -0
  261. /package/dist/{chunk-JF7SFXTG.js.map → chunk-NCSJKK23.js.map} +0 -0
  262. /package/dist/{chunk-XMN6MMTU.js.map → chunk-NRBGRZW4.js.map} +0 -0
  263. /package/dist/{chunk-BNFRL6QW.js.map → chunk-PTMJ2FH2.js.map} +0 -0
  264. /package/dist/{chunk-KWM33SPU.js.map → chunk-PVE7KSQP.js.map} +0 -0
  265. /package/dist/{chunk-YM3LR4LS.js.map → chunk-SSSXWIBP.js.map} +0 -0
  266. /package/dist/{chunk-Y7NWBBHV.js.map → chunk-TEO46GMM.js.map} +0 -0
  267. /package/dist/{chunk-AJE7FJVE.js.map → chunk-UCEABZZN.js.map} +0 -0
  268. /package/dist/{chunk-IENGGY2C.js.map → chunk-UCEDY5M7.js.map} +0 -0
  269. /package/dist/{chunk-PRQXUSQV.js.map → chunk-UYNFWZWG.js.map} +0 -0
  270. /package/dist/{chunk-V4UDXYGG.js.map → chunk-WDTUYOLS.js.map} +0 -0
  271. /package/dist/{chunk-RZOBQ23O.js.map → chunk-XOFXKASO.js.map} +0 -0
  272. /package/dist/{chunk-WTI35CVJ.js.map → chunk-YYN3LIYA.js.map} +0 -0
  273. /package/dist/{resolution-3SAP4SH2.js.map → resolution-IDTEBJFS.js.map} +0 -0
@@ -0,0 +1,893 @@
1
+ import { createHash, randomUUID } from "node:crypto";
2
+ import type { Dirent } from "node:fs";
3
+ import { lstat, mkdir, open, readFile, readdir, rename, rm, rmdir, stat, utimes, writeFile } from "node:fs/promises";
4
+ import path from "node:path";
5
+
6
+ import type { NamespaceCatalog, NamespaceKind, NamespaceRecord } from "../namespaces/catalog.js";
7
+ import { hasMemoryData } from "../namespaces/catalog.js";
8
+ import { namespaceIdentityToken } from "../namespaces/identity.js";
9
+ import { resolveNamespaceStorageRoot } from "../namespaces/storage.js";
10
+ import { displayErrorDetail } from "../runtime/better-sqlite.js";
11
+ import type { PluginConfig } from "../types.js";
12
+
13
+ export type NamespaceMaintenanceJobName = string;
14
+
15
+ export type NamespaceMaintenanceSkipReason =
16
+ | "fanout_disabled"
17
+ | "empty_root"
18
+ | "branch_disabled"
19
+ | "project_disabled"
20
+ | "team_project_disabled"
21
+ | "catalog_read_failed"
22
+ | "unsafe_or_stale_root"
23
+ | "budget_exhausted"
24
+ | "lock_held"
25
+ | "batch_lock_incomplete"
26
+ | "throttled"
27
+ | "job_failed";
28
+
29
+ export interface NamespaceMaintenancePlannerOptions {
30
+ jobName: NamespaceMaintenanceJobName;
31
+ catalog?: NamespaceCatalog;
32
+ now?: Date;
33
+ budgetMode?: "cycle" | "unbounded";
34
+ }
35
+
36
+ export interface NamespaceMaintenanceCandidate {
37
+ namespace: string;
38
+ kind: NamespaceKind;
39
+ storageDir?: string;
40
+ source: "configured" | "catalog";
41
+ lastWriteAt?: string;
42
+ lastMaintenanceAt?: string;
43
+ }
44
+
45
+ export interface NamespaceMaintenancePlan {
46
+ jobName: NamespaceMaintenanceJobName;
47
+ generatedAt: string;
48
+ namespaces: NamespaceMaintenanceCandidate[];
49
+ skipped: NamespaceMaintenanceSkippedNamespace[];
50
+ budget: {
51
+ maxNamespacesPerCycle: number;
52
+ selected: number;
53
+ };
54
+ }
55
+
56
+ export interface NamespaceMaintenanceSkippedNamespace {
57
+ namespace: string;
58
+ kind?: NamespaceKind;
59
+ reason: NamespaceMaintenanceSkipReason;
60
+ detail?: string;
61
+ }
62
+
63
+ export interface NamespaceMaintenanceRunStatus {
64
+ namespace: string;
65
+ jobName: NamespaceMaintenanceJobName;
66
+ state: "ran" | "skipped" | "failed";
67
+ reason?: NamespaceMaintenanceSkipReason;
68
+ startedAt: string;
69
+ completedAt: string;
70
+ itemCount?: number;
71
+ error?: string;
72
+ }
73
+
74
+ export interface NamespaceMaintenanceSummary {
75
+ jobName: NamespaceMaintenanceJobName;
76
+ generatedAt: string;
77
+ ran: number;
78
+ skipped: number;
79
+ failed: number;
80
+ statuses: NamespaceMaintenanceRunStatus[];
81
+ }
82
+
83
+ export interface NamespaceMaintenanceBatchRunResult {
84
+ itemCount?: number;
85
+ itemCounts?: Record<string, number> | Map<string, number>;
86
+ }
87
+
88
+ export interface NamespaceMaintenanceBatchRunOptions {
89
+ requireAllLocks?: boolean;
90
+ skipReasonForError?: (error: unknown) => NamespaceMaintenanceSkipReason | null | undefined;
91
+ }
92
+
93
+ interface LockHandle {
94
+ path: string;
95
+ touch(): Promise<void>;
96
+ release(): Promise<void>;
97
+ }
98
+
99
+ const DEFAULT_MAX_NAMESPACES_PER_CYCLE = 20;
100
+ const DEFAULT_LOCK_STALE_MS = 10 * 60_000;
101
+ const LOCK_BASE = "maintenance-locks";
102
+ const STATUS_BASE = "namespace-maintenance-status";
103
+ const namespaceMaintenanceFs = { open, rm };
104
+
105
+ export function __setNamespaceMaintenanceFsForTest(overrides: Partial<typeof namespaceMaintenanceFs>): () => void {
106
+ const previous = { ...namespaceMaintenanceFs };
107
+ Object.assign(namespaceMaintenanceFs, overrides);
108
+ return () => {
109
+ Object.assign(namespaceMaintenanceFs, previous);
110
+ };
111
+ }
112
+
113
+ function configuredNamespaces(config: PluginConfig): string[] {
114
+ return Array.from(
115
+ new Set(
116
+ [config.defaultNamespace, config.sharedNamespace, ...config.namespacePolicies.map((policy) => policy.name)]
117
+ .map((value) => value.trim())
118
+ .filter(Boolean)
119
+ )
120
+ );
121
+ }
122
+
123
+ function inferConfiguredKind(config: PluginConfig, namespace: string): NamespaceKind {
124
+ if (namespace === config.defaultNamespace.trim()) return "default";
125
+ if (namespace === config.sharedNamespace.trim()) return "shared";
126
+ return "explicit";
127
+ }
128
+
129
+ function maxNamespacesPerCycle(config: PluginConfig): number {
130
+ return Math.max(
131
+ 1,
132
+ Math.floor(
133
+ typeof config.maintenanceMaxNamespacesPerCycle === "number" &&
134
+ Number.isFinite(config.maintenanceMaxNamespacesPerCycle)
135
+ ? config.maintenanceMaxNamespacesPerCycle
136
+ : DEFAULT_MAX_NAMESPACES_PER_CYCLE
137
+ )
138
+ );
139
+ }
140
+
141
+ function namespaceKindAllowed(config: PluginConfig, kind: NamespaceKind): boolean {
142
+ switch (kind) {
143
+ case "branch":
144
+ return config.maintenanceIncludeBranchNamespaces === true;
145
+ case "project":
146
+ return config.maintenanceIncludeProjectNamespaces !== false;
147
+ case "team-project":
148
+ return config.maintenanceIncludeTeamProjectNamespaces !== false;
149
+ default:
150
+ return true;
151
+ }
152
+ }
153
+
154
+ function disabledReasonForKind(kind: NamespaceKind): NamespaceMaintenanceSkipReason {
155
+ if (kind === "branch") return "branch_disabled";
156
+ if (kind === "project") return "project_disabled";
157
+ if (kind === "team-project") return "team_project_disabled";
158
+ return "fanout_disabled";
159
+ }
160
+
161
+ async function catalogRootIsLive(config: PluginConfig, record: NamespaceRecord): Promise<boolean> {
162
+ if (typeof record.storageDir !== "string" || record.storageDir.length === 0) {
163
+ return false;
164
+ }
165
+ try {
166
+ const liveRoot = await resolveNamespaceStorageRoot(config, record.namespace);
167
+ if (path.resolve(liveRoot) !== path.resolve(record.storageDir)) return false;
168
+ return hasMemoryData(liveRoot);
169
+ } catch {
170
+ return false;
171
+ }
172
+ }
173
+
174
+ function candidateSortKey(candidate: NamespaceMaintenanceCandidate): string {
175
+ const write = candidate.lastWriteAt ?? "";
176
+ return `${write}\u0000${candidate.namespace}`;
177
+ }
178
+
179
+ function candidatePriority(candidate: NamespaceMaintenanceCandidate): number {
180
+ if (candidate.kind === "default") return 0;
181
+ if (candidate.kind === "shared") return 1;
182
+ if (candidate.source === "configured") return 2;
183
+ if (candidate.kind === "team-project") return 3;
184
+ if (candidate.kind === "project") return 4;
185
+ if (candidate.kind === "self") return 5;
186
+ if (candidate.kind === "legacy") return 6;
187
+ if (candidate.kind === "branch") return 8;
188
+ return 7;
189
+ }
190
+
191
+ function sortCandidates(a: NamespaceMaintenanceCandidate, b: NamespaceMaintenanceCandidate): number {
192
+ const priority = candidatePriority(a) - candidatePriority(b);
193
+ if (priority !== 0) return priority;
194
+ const am = Date.parse(a.lastMaintenanceAt ?? "");
195
+ const bm = Date.parse(b.lastMaintenanceAt ?? "");
196
+ const aMaintained = Number.isFinite(am);
197
+ const bMaintained = Number.isFinite(bm);
198
+ if (aMaintained && bMaintained && am !== bm) return am - bm;
199
+ if (aMaintained !== bMaintained) return aMaintained ? 1 : -1;
200
+ const aw = Date.parse(a.lastWriteAt ?? "");
201
+ const bw = Date.parse(b.lastWriteAt ?? "");
202
+ const aValid = Number.isFinite(aw);
203
+ const bValid = Number.isFinite(bw);
204
+ if (aValid && bValid && aw !== bw) return bw - aw;
205
+ if (aValid !== bValid) return aValid ? -1 : 1;
206
+ const byKey = candidateSortKey(a).localeCompare(candidateSortKey(b));
207
+ if (byKey !== 0) return byKey;
208
+ return a.namespace.localeCompare(b.namespace);
209
+ }
210
+
211
+ export async function planNamespaceMaintenance(
212
+ config: PluginConfig,
213
+ options: NamespaceMaintenancePlannerOptions
214
+ ): Promise<NamespaceMaintenancePlan> {
215
+ const generatedAt = (options.now ?? new Date()).toISOString();
216
+ const configured = configuredNamespaces(config);
217
+ const byNamespace = new Map<string, NamespaceMaintenanceCandidate>();
218
+ const skipped: NamespaceMaintenanceSkippedNamespace[] = [];
219
+
220
+ for (const namespace of configured) {
221
+ const kind = inferConfiguredKind(config, namespace);
222
+ byNamespace.set(namespace, {
223
+ namespace,
224
+ kind,
225
+ source: "configured",
226
+ });
227
+ }
228
+
229
+ if (config.namespacesEnabled && config.maintenanceNamespaceFanoutEnabled !== false) {
230
+ const configuredSet = new Set(configured);
231
+ try {
232
+ const records = options.catalog?.enabled ? await options.catalog.listNamespaces() : [];
233
+ for (const record of records) {
234
+ const namespace = record.namespace.trim();
235
+ if (!namespace) continue;
236
+ const isConfigured = configuredSet.has(namespace);
237
+ const kind = isConfigured ? inferConfiguredKind(config, namespace) : record.kind;
238
+ if (!namespaceKindAllowed(config, kind)) {
239
+ skipped.push({
240
+ namespace,
241
+ kind,
242
+ reason: disabledReasonForKind(kind),
243
+ });
244
+ continue;
245
+ }
246
+ if (!isConfigured && !(await catalogRootIsLive(config, record))) {
247
+ skipped.push({
248
+ namespace,
249
+ kind,
250
+ reason: "unsafe_or_stale_root",
251
+ });
252
+ continue;
253
+ }
254
+ byNamespace.set(namespace, {
255
+ namespace,
256
+ kind,
257
+ storageDir: record.storageDir,
258
+ source: isConfigured ? "configured" : "catalog",
259
+ lastWriteAt: record.lastWriteAt,
260
+ lastMaintenanceAt: record.lastMaintenanceAt?.[options.jobName],
261
+ });
262
+ }
263
+ } catch (error) {
264
+ skipped.push({
265
+ namespace: "*",
266
+ reason: "catalog_read_failed",
267
+ detail: error instanceof Error ? error.message : String(error),
268
+ });
269
+ }
270
+ } else if (config.namespacesEnabled) {
271
+ skipped.push({
272
+ namespace: "*",
273
+ reason: "fanout_disabled",
274
+ });
275
+ }
276
+
277
+ if (options.budgetMode !== "unbounded") {
278
+ const latestStatusAtByNamespace = await readLatestStatusAtByNamespace(config, options.jobName);
279
+ for (const candidate of byNamespace.values()) {
280
+ if (!candidate.lastMaintenanceAt) {
281
+ candidate.lastMaintenanceAt = latestStatusAtByNamespace.get(candidate.namespace);
282
+ }
283
+ }
284
+ }
285
+
286
+ const candidates = [...byNamespace.values()]
287
+ .filter((candidate) => namespaceKindAllowed(config, candidate.kind))
288
+ .sort(sortCandidates);
289
+
290
+ const max = maxNamespacesPerCycle(config);
291
+ const applyCycleBudget = options.budgetMode !== "unbounded";
292
+ const selected = applyCycleBudget ? candidates.slice(0, max) : candidates;
293
+ if (applyCycleBudget) {
294
+ for (const candidate of candidates.slice(max)) {
295
+ skipped.push({
296
+ namespace: candidate.namespace,
297
+ kind: candidate.kind,
298
+ reason: "budget_exhausted",
299
+ });
300
+ }
301
+ }
302
+
303
+ return {
304
+ jobName: options.jobName,
305
+ generatedAt,
306
+ namespaces: selected,
307
+ skipped,
308
+ budget: {
309
+ maxNamespacesPerCycle: max,
310
+ selected: selected.length,
311
+ },
312
+ };
313
+ }
314
+
315
+ function stablePathSegment(value: string): string {
316
+ const sanitized = value.replace(/[^a-zA-Z0-9._-]/g, "_").slice(0, 128) || "unnamed";
317
+ if (sanitized.length <= 128 && sanitized === value) return sanitized;
318
+ return `${sanitized.slice(0, 80)}-${createHash("sha256").update(value).digest("hex").slice(0, 16)}`;
319
+ }
320
+
321
+ function namespacePathSegment(namespace: string): string {
322
+ const token = namespaceIdentityToken(namespace);
323
+ if (token.length <= 160) return token;
324
+ return `ns-${createHash("sha256").update(namespace).digest("hex")}`;
325
+ }
326
+
327
+ function lockPath(config: PluginConfig, jobName: string, namespace: string): string {
328
+ return path.join(
329
+ config.memoryDir,
330
+ "state",
331
+ LOCK_BASE,
332
+ stablePathSegment(jobName),
333
+ `${namespacePathSegment(namespace)}.lock`
334
+ );
335
+ }
336
+
337
+ function namespaceMaintenanceLockStaleMs(config: PluginConfig): number {
338
+ if (
339
+ typeof config.maintenanceNamespaceLockStaleMs === "number" &&
340
+ Number.isFinite(config.maintenanceNamespaceLockStaleMs) &&
341
+ config.maintenanceNamespaceLockStaleMs > 0
342
+ ) {
343
+ return Math.floor(config.maintenanceNamespaceLockStaleMs);
344
+ }
345
+ return DEFAULT_LOCK_STALE_MS;
346
+ }
347
+
348
+ function namespaceMaintenanceLockHeartbeatMs(config: PluginConfig): number {
349
+ const staleMs = namespaceMaintenanceLockStaleMs(config);
350
+ return Math.max(1, Math.min(30_000, Math.floor(staleMs / 3) || 1));
351
+ }
352
+
353
+ function errorCode(error: unknown): string | undefined {
354
+ return typeof error === "object" && error !== null && "code" in error
355
+ ? (error as { code?: string }).code
356
+ : undefined;
357
+ }
358
+
359
+ async function withNamespaceMaintenanceLockHeartbeat<T>(
360
+ config: PluginConfig,
361
+ locks: LockHandle | LockHandle[],
362
+ task: () => Promise<T>,
363
+ ): Promise<T> {
364
+ const activeLocks = Array.isArray(locks) ? locks : [locks];
365
+ const interval = setInterval(() => {
366
+ for (const lock of activeLocks) {
367
+ void lock.touch().catch(() => undefined);
368
+ }
369
+ }, namespaceMaintenanceLockHeartbeatMs(config));
370
+ interval.unref?.();
371
+ try {
372
+ return await task();
373
+ } finally {
374
+ clearInterval(interval);
375
+ }
376
+ }
377
+
378
+ async function removeStaleLockDirectory(filePath: string): Promise<void> {
379
+ let entries: Dirent[];
380
+ try {
381
+ entries = await readdir(filePath, { withFileTypes: true });
382
+ } catch (error) {
383
+ if (errorCode(error) === "ENOENT") return;
384
+ throw error;
385
+ }
386
+ for (const entry of entries) {
387
+ if (!entry.isFile()) continue;
388
+ await namespaceMaintenanceFs.rm(path.join(filePath, entry.name), { force: true });
389
+ }
390
+ try {
391
+ await rmdir(filePath);
392
+ } catch (error) {
393
+ if (errorCode(error) === "ENOENT") return;
394
+ throw error;
395
+ }
396
+ }
397
+
398
+ async function tryAcquireNamespaceMaintenanceLock(
399
+ config: PluginConfig,
400
+ jobName: string,
401
+ namespace: string
402
+ ): Promise<LockHandle | null> {
403
+ const filePath = lockPath(config, jobName, namespace);
404
+ await mkdir(path.dirname(filePath), { recursive: true });
405
+ try {
406
+ const lockId = randomUUID();
407
+ await mkdir(filePath);
408
+ const ownerPath = path.join(filePath, `${lockId}.json`);
409
+ let handle: Awaited<ReturnType<typeof open>> | undefined;
410
+ try {
411
+ handle = await namespaceMaintenanceFs.open(ownerPath, "wx");
412
+ await handle.writeFile(
413
+ `${JSON.stringify({
414
+ lockId,
415
+ pid: process.pid,
416
+ jobName,
417
+ namespace,
418
+ acquiredAt: new Date().toISOString(),
419
+ })}\n`,
420
+ "utf8"
421
+ );
422
+ await handle.close();
423
+ } catch (setupError) {
424
+ await handle?.close().catch(() => undefined);
425
+ await namespaceMaintenanceFs.rm(ownerPath, { force: true }).catch(() => undefined);
426
+ await rmdir(filePath).catch(() => undefined);
427
+ throw setupError;
428
+ }
429
+ return {
430
+ path: filePath,
431
+ async touch() {
432
+ try {
433
+ const parsed = JSON.parse(await readFile(ownerPath, "utf8")) as { lockId?: unknown };
434
+ if (parsed.lockId === lockId) {
435
+ const now = new Date();
436
+ await utimes(ownerPath, now, now);
437
+ await utimes(filePath, now, now);
438
+ }
439
+ } catch {}
440
+ },
441
+ async release() {
442
+ try {
443
+ const parsed = JSON.parse(await readFile(ownerPath, "utf8")) as { lockId?: unknown };
444
+ if (parsed.lockId === lockId) {
445
+ await namespaceMaintenanceFs.rm(ownerPath, { force: true });
446
+ await rmdir(filePath).catch(() => undefined);
447
+ }
448
+ } catch {}
449
+ },
450
+ };
451
+ } catch (error) {
452
+ if (errorCode(error) === "EEXIST") {
453
+ const staleMs =
454
+ namespaceMaintenanceLockStaleMs(config);
455
+ try {
456
+ const s = await lstat(filePath);
457
+ if (s.isSymbolicLink()) {
458
+ return null;
459
+ }
460
+ if ((s.isFile() || s.isDirectory()) && Date.now() - s.mtimeMs > staleMs) {
461
+ try {
462
+ if (s.isDirectory()) {
463
+ await removeStaleLockDirectory(filePath);
464
+ } else {
465
+ await namespaceMaintenanceFs.rm(filePath, { force: true });
466
+ }
467
+ } catch (removeError) {
468
+ if (errorCode(removeError) === "ENOENT") {
469
+ return tryAcquireNamespaceMaintenanceLock(config, jobName, namespace);
470
+ }
471
+ if (errorCode(removeError) === "ENOTEMPTY") {
472
+ return null;
473
+ }
474
+ throw removeError;
475
+ }
476
+ return tryAcquireNamespaceMaintenanceLock(config, jobName, namespace);
477
+ }
478
+ } catch (statError) {
479
+ if (errorCode(statError) === "ENOENT") {
480
+ return tryAcquireNamespaceMaintenanceLock(config, jobName, namespace);
481
+ }
482
+ throw statError;
483
+ }
484
+ return null;
485
+ }
486
+ throw error;
487
+ }
488
+ }
489
+
490
+ function statusBasePath(config: PluginConfig): string {
491
+ return path.join(config.memoryDir, "state", STATUS_BASE);
492
+ }
493
+
494
+ function statusPath(config: PluginConfig, jobName: string, namespace: string): string {
495
+ return path.join(statusBasePath(config), stablePathSegment(jobName), `${namespacePathSegment(namespace)}.json`);
496
+ }
497
+
498
+ function lastRanStatusPath(config: PluginConfig, jobName: string, namespace: string): string {
499
+ return path.join(statusBasePath(config), stablePathSegment(jobName), `${namespacePathSegment(namespace)}.last-ran.json`);
500
+ }
501
+
502
+ function parseStatus(value: unknown): NamespaceMaintenanceRunStatus | null {
503
+ if (!value || typeof value !== "object" || Array.isArray(value)) return null;
504
+ const v = value as Partial<NamespaceMaintenanceRunStatus>;
505
+ if (
506
+ typeof v.namespace === "string" &&
507
+ typeof v.jobName === "string" &&
508
+ (v.state === "ran" || v.state === "skipped" || v.state === "failed") &&
509
+ typeof v.startedAt === "string" &&
510
+ typeof v.completedAt === "string"
511
+ ) {
512
+ return v as NamespaceMaintenanceRunStatus;
513
+ }
514
+ return null;
515
+ }
516
+
517
+ async function readLatestStatusAtByNamespace(config: PluginConfig, jobName: string): Promise<Map<string, string>> {
518
+ const latest = new Map<string, string>();
519
+ const latestMs = new Map<string, number>();
520
+ for (const status of [...(await readStatusFiles(config)), ...(await readLastRanStatusFiles(config))]) {
521
+ if (status.state !== "ran") continue;
522
+ if (status.jobName !== jobName) continue;
523
+ const completedAtMs = Date.parse(status.completedAt);
524
+ if (!Number.isFinite(completedAtMs)) continue;
525
+ const previousMs = latestMs.get(status.namespace);
526
+ if (previousMs !== undefined && previousMs >= completedAtMs) continue;
527
+ latestMs.set(status.namespace, completedAtMs);
528
+ latest.set(status.namespace, status.completedAt);
529
+ }
530
+ return latest;
531
+ }
532
+
533
+ async function readStatusFile(filePath: string): Promise<NamespaceMaintenanceRunStatus | null> {
534
+ try {
535
+ const parsed = JSON.parse(await readFile(filePath, "utf8")) as unknown;
536
+ return parseStatus(parsed);
537
+ } catch {
538
+ return null;
539
+ }
540
+ }
541
+
542
+ async function readStatusFiles(config: PluginConfig): Promise<NamespaceMaintenanceRunStatus[]> {
543
+ if (typeof config.memoryDir !== "string" || config.memoryDir.length === 0) {
544
+ return [];
545
+ }
546
+ const root = statusBasePath(config);
547
+ const statuses: NamespaceMaintenanceRunStatus[] = [];
548
+ let jobDirs: Dirent[];
549
+ try {
550
+ jobDirs = await readdir(root, { withFileTypes: true });
551
+ } catch {
552
+ return statuses;
553
+ }
554
+ for (const jobDir of jobDirs) {
555
+ if (!jobDir.isDirectory()) continue;
556
+ let files: Dirent[];
557
+ try {
558
+ files = await readdir(path.join(root, jobDir.name), { withFileTypes: true });
559
+ } catch {
560
+ continue;
561
+ }
562
+ for (const file of files) {
563
+ if (!file.isFile() || !file.name.endsWith(".json")) continue;
564
+ if (file.name.endsWith(".last-ran.json")) continue;
565
+ const status = await readStatusFile(path.join(root, jobDir.name, file.name));
566
+ if (status) statuses.push(status);
567
+ }
568
+ }
569
+ return statuses;
570
+ }
571
+
572
+ async function readLastRanStatusFiles(config: PluginConfig): Promise<NamespaceMaintenanceRunStatus[]> {
573
+ if (typeof config.memoryDir !== "string" || config.memoryDir.length === 0) {
574
+ return [];
575
+ }
576
+ const root = statusBasePath(config);
577
+ const statuses: NamespaceMaintenanceRunStatus[] = [];
578
+ let jobDirs: Dirent[];
579
+ try {
580
+ jobDirs = await readdir(root, { withFileTypes: true });
581
+ } catch {
582
+ return statuses;
583
+ }
584
+ for (const jobDir of jobDirs) {
585
+ if (!jobDir.isDirectory()) continue;
586
+ let files: Dirent[];
587
+ try {
588
+ files = await readdir(path.join(root, jobDir.name), { withFileTypes: true });
589
+ } catch {
590
+ continue;
591
+ }
592
+ for (const file of files) {
593
+ if (!file.isFile() || !file.name.endsWith(".last-ran.json")) continue;
594
+ const status = await readStatusFile(path.join(root, jobDir.name, file.name));
595
+ if (status) statuses.push(status);
596
+ }
597
+ }
598
+ return statuses;
599
+ }
600
+
601
+ async function writeStatusPayload(target: string, status: NamespaceMaintenanceRunStatus): Promise<void> {
602
+ const dir = path.dirname(target);
603
+ await mkdir(dir, { recursive: true });
604
+ const temp = `${target}.${process.pid}.${Date.now()}.${Math.random().toString(36).slice(2)}.tmp`;
605
+ const payload = {
606
+ version: 1,
607
+ ...status,
608
+ };
609
+ await writeFile(temp, `${JSON.stringify(payload, null, 2)}\n`, "utf8");
610
+ await rename(temp, target);
611
+ }
612
+
613
+ async function writeStatusFile(config: PluginConfig, status: NamespaceMaintenanceRunStatus): Promise<void> {
614
+ await writeStatusPayload(statusPath(config, status.jobName, status.namespace), status);
615
+ if (status.state === "ran") {
616
+ await writeStatusPayload(lastRanStatusPath(config, status.jobName, status.namespace), status);
617
+ }
618
+ }
619
+
620
+ async function recordNamespaceMaintenanceStatusSafely(
621
+ config: PluginConfig,
622
+ status: NamespaceMaintenanceRunStatus
623
+ ): Promise<void> {
624
+ try {
625
+ await writeStatusFile(config, status);
626
+ } catch {
627
+ // Observability must not fail the maintenance operation.
628
+ }
629
+ }
630
+
631
+ function maintenanceErrorDetail(error: unknown): string {
632
+ return displayErrorDetail(error) || "Error";
633
+ }
634
+
635
+ export async function readNamespaceMaintenanceStatuses(config: PluginConfig): Promise<NamespaceMaintenanceRunStatus[]> {
636
+ return (await readStatusFiles(config)).sort((a, b) => {
637
+ const byJob = a.jobName.localeCompare(b.jobName);
638
+ if (byJob !== 0) return byJob;
639
+ return a.namespace.localeCompare(b.namespace);
640
+ });
641
+ }
642
+
643
+ export async function runNamespaceMaintenancePlan(
644
+ config: PluginConfig,
645
+ plan: NamespaceMaintenancePlan,
646
+ runner: (candidate: NamespaceMaintenanceCandidate) => Promise<{ itemCount?: number } | undefined>,
647
+ catalog?: NamespaceCatalog
648
+ ): Promise<NamespaceMaintenanceSummary> {
649
+ const statuses: NamespaceMaintenanceRunStatus[] = [];
650
+
651
+ for (const skipped of plan.skipped) {
652
+ if (skipped.namespace === "*") continue;
653
+ const now = new Date().toISOString();
654
+ const status: NamespaceMaintenanceRunStatus = {
655
+ namespace: skipped.namespace,
656
+ jobName: plan.jobName,
657
+ state: "skipped",
658
+ reason: skipped.reason,
659
+ startedAt: now,
660
+ completedAt: now,
661
+ };
662
+ statuses.push(status);
663
+ await recordNamespaceMaintenanceStatusSafely(config, status);
664
+ }
665
+
666
+ for (const candidate of plan.namespaces) {
667
+ const startedAt = new Date().toISOString();
668
+ const lock = await tryAcquireNamespaceMaintenanceLock(config, plan.jobName, candidate.namespace);
669
+ if (!lock) {
670
+ const completedAt = new Date().toISOString();
671
+ const status: NamespaceMaintenanceRunStatus = {
672
+ namespace: candidate.namespace,
673
+ jobName: plan.jobName,
674
+ state: "skipped",
675
+ reason: "lock_held",
676
+ startedAt,
677
+ completedAt,
678
+ };
679
+ statuses.push(status);
680
+ await recordNamespaceMaintenanceStatusSafely(config, status);
681
+ continue;
682
+ }
683
+ try {
684
+ const result = await withNamespaceMaintenanceLockHeartbeat(config, lock, () => runner(candidate));
685
+ const completedAt = new Date().toISOString();
686
+ const status: NamespaceMaintenanceRunStatus = {
687
+ namespace: candidate.namespace,
688
+ jobName: plan.jobName,
689
+ state: "ran",
690
+ startedAt,
691
+ completedAt,
692
+ itemCount: result?.itemCount,
693
+ };
694
+ statuses.push(status);
695
+ await recordNamespaceMaintenanceStatusSafely(config, status);
696
+ try {
697
+ await catalog?.markMaintenance(candidate.namespace, plan.jobName, new Date(completedAt));
698
+ } catch {
699
+ // Catalog maintenance touches are best-effort status metadata.
700
+ }
701
+ } catch (error) {
702
+ const completedAt = new Date().toISOString();
703
+ const status: NamespaceMaintenanceRunStatus = {
704
+ namespace: candidate.namespace,
705
+ jobName: plan.jobName,
706
+ state: "failed",
707
+ reason: "job_failed",
708
+ startedAt,
709
+ completedAt,
710
+ error: maintenanceErrorDetail(error),
711
+ };
712
+ statuses.push(status);
713
+ await recordNamespaceMaintenanceStatusSafely(config, status);
714
+ } finally {
715
+ await lock.release().catch(() => undefined);
716
+ }
717
+ }
718
+
719
+ return {
720
+ jobName: plan.jobName,
721
+ generatedAt: new Date().toISOString(),
722
+ ran: statuses.filter((s) => s.state === "ran").length,
723
+ skipped: statuses.filter((s) => s.state === "skipped").length,
724
+ failed: statuses.filter((s) => s.state === "failed").length,
725
+ statuses,
726
+ };
727
+ }
728
+
729
+ export async function runNamespaceMaintenanceBatchPlan(
730
+ config: PluginConfig,
731
+ plan: NamespaceMaintenancePlan,
732
+ runner: (candidates: NamespaceMaintenanceCandidate[]) => Promise<NamespaceMaintenanceBatchRunResult | undefined>,
733
+ catalog?: NamespaceCatalog,
734
+ options: NamespaceMaintenanceBatchRunOptions = {}
735
+ ): Promise<NamespaceMaintenanceSummary> {
736
+ const statuses: NamespaceMaintenanceRunStatus[] = [];
737
+
738
+ for (const skipped of plan.skipped) {
739
+ if (skipped.namespace === "*") continue;
740
+ const now = new Date().toISOString();
741
+ const status: NamespaceMaintenanceRunStatus = {
742
+ namespace: skipped.namespace,
743
+ jobName: plan.jobName,
744
+ state: "skipped",
745
+ reason: skipped.reason,
746
+ startedAt: now,
747
+ completedAt: now,
748
+ };
749
+ statuses.push(status);
750
+ await recordNamespaceMaintenanceStatusSafely(config, status);
751
+ }
752
+
753
+ const acquired: Array<{
754
+ candidate: NamespaceMaintenanceCandidate;
755
+ lock: LockHandle;
756
+ startedAt: string;
757
+ }> = [];
758
+
759
+ try {
760
+ for (const candidate of plan.namespaces) {
761
+ const startedAt = new Date().toISOString();
762
+ const lock = await tryAcquireNamespaceMaintenanceLock(config, plan.jobName, candidate.namespace);
763
+ if (!lock) {
764
+ const completedAt = new Date().toISOString();
765
+ const status: NamespaceMaintenanceRunStatus = {
766
+ namespace: candidate.namespace,
767
+ jobName: plan.jobName,
768
+ state: "skipped",
769
+ reason: "lock_held",
770
+ startedAt,
771
+ completedAt,
772
+ };
773
+ statuses.push(status);
774
+ await recordNamespaceMaintenanceStatusSafely(config, status);
775
+ continue;
776
+ }
777
+ acquired.push({ candidate, lock, startedAt });
778
+ }
779
+ } catch (error) {
780
+ await Promise.all(acquired.map(({ lock }) => lock.release().catch(() => undefined)));
781
+ throw error;
782
+ }
783
+
784
+ if (options.requireAllLocks && acquired.length > 0 && acquired.length < plan.namespaces.length) {
785
+ for (const { candidate, startedAt } of acquired) {
786
+ const completedAt = new Date().toISOString();
787
+ const status: NamespaceMaintenanceRunStatus = {
788
+ namespace: candidate.namespace,
789
+ jobName: plan.jobName,
790
+ state: "skipped",
791
+ reason: "batch_lock_incomplete",
792
+ startedAt,
793
+ completedAt,
794
+ };
795
+ statuses.push(status);
796
+ await recordNamespaceMaintenanceStatusSafely(config, status);
797
+ }
798
+ await Promise.all(acquired.map(({ lock }) => lock.release().catch(() => undefined)));
799
+ return {
800
+ jobName: plan.jobName,
801
+ generatedAt: new Date().toISOString(),
802
+ ran: statuses.filter((s) => s.state === "ran").length,
803
+ skipped: statuses.filter((s) => s.state === "skipped").length,
804
+ failed: statuses.filter((s) => s.state === "failed").length,
805
+ statuses,
806
+ };
807
+ }
808
+
809
+ if (acquired.length === 0) {
810
+ return {
811
+ jobName: plan.jobName,
812
+ generatedAt: new Date().toISOString(),
813
+ ran: statuses.filter((s) => s.state === "ran").length,
814
+ skipped: statuses.filter((s) => s.state === "skipped").length,
815
+ failed: statuses.filter((s) => s.state === "failed").length,
816
+ statuses,
817
+ };
818
+ }
819
+
820
+ try {
821
+ const result = await withNamespaceMaintenanceLockHeartbeat(
822
+ config,
823
+ acquired.map(({ lock }) => lock),
824
+ () => runner(acquired.map(({ candidate }) => candidate)),
825
+ );
826
+ for (const { candidate, startedAt } of acquired) {
827
+ const completedAt = new Date().toISOString();
828
+ const status: NamespaceMaintenanceRunStatus = {
829
+ namespace: candidate.namespace,
830
+ jobName: plan.jobName,
831
+ state: "ran",
832
+ startedAt,
833
+ completedAt,
834
+ itemCount: itemCountForNamespace(result, candidate.namespace),
835
+ };
836
+ statuses.push(status);
837
+ await recordNamespaceMaintenanceStatusSafely(config, status);
838
+ try {
839
+ await catalog?.markMaintenance(candidate.namespace, plan.jobName, new Date(completedAt));
840
+ } catch {
841
+ // Catalog maintenance touches are best-effort status metadata.
842
+ }
843
+ }
844
+ } catch (error) {
845
+ const skipReason = options.skipReasonForError?.(error);
846
+ for (const { candidate, startedAt } of acquired) {
847
+ const completedAt = new Date().toISOString();
848
+ const status: NamespaceMaintenanceRunStatus = skipReason
849
+ ? {
850
+ namespace: candidate.namespace,
851
+ jobName: plan.jobName,
852
+ state: "skipped",
853
+ reason: skipReason,
854
+ startedAt,
855
+ completedAt,
856
+ }
857
+ : {
858
+ namespace: candidate.namespace,
859
+ jobName: plan.jobName,
860
+ state: "failed",
861
+ reason: "job_failed",
862
+ startedAt,
863
+ completedAt,
864
+ error: maintenanceErrorDetail(error),
865
+ };
866
+ statuses.push(status);
867
+ await recordNamespaceMaintenanceStatusSafely(config, status);
868
+ }
869
+ } finally {
870
+ await Promise.all(acquired.map(({ lock }) => lock.release().catch(() => undefined)));
871
+ }
872
+
873
+ return {
874
+ jobName: plan.jobName,
875
+ generatedAt: new Date().toISOString(),
876
+ ran: statuses.filter((s) => s.state === "ran").length,
877
+ skipped: statuses.filter((s) => s.state === "skipped").length,
878
+ failed: statuses.filter((s) => s.state === "failed").length,
879
+ statuses,
880
+ };
881
+ }
882
+
883
+ function itemCountForNamespace(
884
+ result: NamespaceMaintenanceBatchRunResult | undefined,
885
+ namespace: string,
886
+ ): number | undefined {
887
+ const itemCounts = result?.itemCounts;
888
+ if (itemCounts instanceof Map) return itemCounts.get(namespace);
889
+ if (itemCounts && Object.prototype.hasOwnProperty.call(itemCounts, namespace)) {
890
+ return itemCounts[namespace];
891
+ }
892
+ return result?.itemCount;
893
+ }