@principles/pd-cli 1.73.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.
Files changed (298) hide show
  1. package/README.md +90 -0
  2. package/dist/commands/artifact.d.ts +14 -0
  3. package/dist/commands/artifact.d.ts.map +1 -0
  4. package/dist/commands/artifact.js +67 -0
  5. package/dist/commands/artifact.js.map +1 -0
  6. package/dist/commands/candidate.d.ts +83 -0
  7. package/dist/commands/candidate.d.ts.map +1 -0
  8. package/dist/commands/candidate.js +891 -0
  9. package/dist/commands/candidate.js.map +1 -0
  10. package/dist/commands/central-sync.d.ts +10 -0
  11. package/dist/commands/central-sync.d.ts.map +1 -0
  12. package/dist/commands/central-sync.js +32 -0
  13. package/dist/commands/central-sync.js.map +1 -0
  14. package/dist/commands/console.d.ts +9 -0
  15. package/dist/commands/console.d.ts.map +1 -0
  16. package/dist/commands/console.js +114 -0
  17. package/dist/commands/console.js.map +1 -0
  18. package/dist/commands/context.d.ts +7 -0
  19. package/dist/commands/context.d.ts.map +1 -0
  20. package/dist/commands/context.js +55 -0
  21. package/dist/commands/context.js.map +1 -0
  22. package/dist/commands/demo-story-a.d.ts +12 -0
  23. package/dist/commands/demo-story-a.d.ts.map +1 -0
  24. package/dist/commands/demo-story-a.js +175 -0
  25. package/dist/commands/demo-story-a.js.map +1 -0
  26. package/dist/commands/diagnose.d.ts +35 -0
  27. package/dist/commands/diagnose.d.ts.map +1 -0
  28. package/dist/commands/diagnose.js +390 -0
  29. package/dist/commands/diagnose.js.map +1 -0
  30. package/dist/commands/evolution-tasks-list.d.ts +15 -0
  31. package/dist/commands/evolution-tasks-list.d.ts.map +1 -0
  32. package/dist/commands/evolution-tasks-list.js +34 -0
  33. package/dist/commands/evolution-tasks-list.js.map +1 -0
  34. package/dist/commands/evolution-tasks-show.d.ts +14 -0
  35. package/dist/commands/evolution-tasks-show.d.ts.map +1 -0
  36. package/dist/commands/evolution-tasks-show.js +52 -0
  37. package/dist/commands/evolution-tasks-show.js.map +1 -0
  38. package/dist/commands/flow.d.ts +7 -0
  39. package/dist/commands/flow.d.ts.map +1 -0
  40. package/dist/commands/flow.js +57 -0
  41. package/dist/commands/flow.js.map +1 -0
  42. package/dist/commands/health.d.ts +16 -0
  43. package/dist/commands/health.d.ts.map +1 -0
  44. package/dist/commands/health.js +150 -0
  45. package/dist/commands/health.js.map +1 -0
  46. package/dist/commands/history.d.ts +11 -0
  47. package/dist/commands/history.d.ts.map +1 -0
  48. package/dist/commands/history.js +50 -0
  49. package/dist/commands/history.js.map +1 -0
  50. package/dist/commands/legacy-cleanup.d.ts +27 -0
  51. package/dist/commands/legacy-cleanup.d.ts.map +1 -0
  52. package/dist/commands/legacy-cleanup.js +171 -0
  53. package/dist/commands/legacy-cleanup.js.map +1 -0
  54. package/dist/commands/legacy-import.d.ts +7 -0
  55. package/dist/commands/legacy-import.d.ts.map +1 -0
  56. package/dist/commands/legacy-import.js +86 -0
  57. package/dist/commands/legacy-import.js.map +1 -0
  58. package/dist/commands/pain-record.d.ts +10 -0
  59. package/dist/commands/pain-record.d.ts.map +1 -0
  60. package/dist/commands/pain-record.js +162 -0
  61. package/dist/commands/pain-record.js.map +1 -0
  62. package/dist/commands/proven-channel-baseline.d.ts +12 -0
  63. package/dist/commands/proven-channel-baseline.d.ts.map +1 -0
  64. package/dist/commands/proven-channel-baseline.js +97 -0
  65. package/dist/commands/proven-channel-baseline.js.map +1 -0
  66. package/dist/commands/remediation-output.d.ts +40 -0
  67. package/dist/commands/remediation-output.d.ts.map +1 -0
  68. package/dist/commands/remediation-output.js +23 -0
  69. package/dist/commands/remediation-output.js.map +1 -0
  70. package/dist/commands/run.d.ts +10 -0
  71. package/dist/commands/run.d.ts.map +1 -0
  72. package/dist/commands/run.js +68 -0
  73. package/dist/commands/run.js.map +1 -0
  74. package/dist/commands/runtime-activation.d.ts +11 -0
  75. package/dist/commands/runtime-activation.d.ts.map +1 -0
  76. package/dist/commands/runtime-activation.js +150 -0
  77. package/dist/commands/runtime-activation.js.map +1 -0
  78. package/dist/commands/runtime-canary.d.ts +30 -0
  79. package/dist/commands/runtime-canary.d.ts.map +1 -0
  80. package/dist/commands/runtime-canary.js +343 -0
  81. package/dist/commands/runtime-canary.js.map +1 -0
  82. package/dist/commands/runtime-diagnostics-export.d.ts +20 -0
  83. package/dist/commands/runtime-diagnostics-export.d.ts.map +1 -0
  84. package/dist/commands/runtime-diagnostics-export.js +177 -0
  85. package/dist/commands/runtime-diagnostics-export.js.map +1 -0
  86. package/dist/commands/runtime-features.d.ts +26 -0
  87. package/dist/commands/runtime-features.d.ts.map +1 -0
  88. package/dist/commands/runtime-features.js +70 -0
  89. package/dist/commands/runtime-features.js.map +1 -0
  90. package/dist/commands/runtime-gfi-snapshot.d.ts +7 -0
  91. package/dist/commands/runtime-gfi-snapshot.d.ts.map +1 -0
  92. package/dist/commands/runtime-gfi-snapshot.js +101 -0
  93. package/dist/commands/runtime-gfi-snapshot.js.map +1 -0
  94. package/dist/commands/runtime-health-snapshot.d.ts +7 -0
  95. package/dist/commands/runtime-health-snapshot.d.ts.map +1 -0
  96. package/dist/commands/runtime-health-snapshot.js +93 -0
  97. package/dist/commands/runtime-health-snapshot.js.map +1 -0
  98. package/dist/commands/runtime-idle-trigger.d.ts +12 -0
  99. package/dist/commands/runtime-idle-trigger.d.ts.map +1 -0
  100. package/dist/commands/runtime-idle-trigger.js +102 -0
  101. package/dist/commands/runtime-idle-trigger.js.map +1 -0
  102. package/dist/commands/runtime-internalization-enqueue-successors.d.ts +9 -0
  103. package/dist/commands/runtime-internalization-enqueue-successors.d.ts.map +1 -0
  104. package/dist/commands/runtime-internalization-enqueue-successors.js +393 -0
  105. package/dist/commands/runtime-internalization-enqueue-successors.js.map +1 -0
  106. package/dist/commands/runtime-internalization-integrity-repair.d.ts +9 -0
  107. package/dist/commands/runtime-internalization-integrity-repair.d.ts.map +1 -0
  108. package/dist/commands/runtime-internalization-integrity-repair.js +54 -0
  109. package/dist/commands/runtime-internalization-integrity-repair.js.map +1 -0
  110. package/dist/commands/runtime-internalization-integrity.d.ts +7 -0
  111. package/dist/commands/runtime-internalization-integrity.d.ts.map +1 -0
  112. package/dist/commands/runtime-internalization-integrity.js +53 -0
  113. package/dist/commands/runtime-internalization-integrity.js.map +1 -0
  114. package/dist/commands/runtime-internalization-queue.d.ts +7 -0
  115. package/dist/commands/runtime-internalization-queue.d.ts.map +1 -0
  116. package/dist/commands/runtime-internalization-queue.js +85 -0
  117. package/dist/commands/runtime-internalization-queue.js.map +1 -0
  118. package/dist/commands/runtime-internalization-run-once.d.ts +12 -0
  119. package/dist/commands/runtime-internalization-run-once.d.ts.map +1 -0
  120. package/dist/commands/runtime-internalization-run-once.js +546 -0
  121. package/dist/commands/runtime-internalization-run-once.js.map +1 -0
  122. package/dist/commands/runtime-internalization-wake-once.d.ts +8 -0
  123. package/dist/commands/runtime-internalization-wake-once.d.ts.map +1 -0
  124. package/dist/commands/runtime-internalization-wake-once.js +72 -0
  125. package/dist/commands/runtime-internalization-wake-once.js.map +1 -0
  126. package/dist/commands/runtime-pain-flood-simulation.d.ts +10 -0
  127. package/dist/commands/runtime-pain-flood-simulation.d.ts.map +1 -0
  128. package/dist/commands/runtime-pain-flood-simulation.js +104 -0
  129. package/dist/commands/runtime-pain-flood-simulation.js.map +1 -0
  130. package/dist/commands/runtime-pruning.d.ts +45 -0
  131. package/dist/commands/runtime-pruning.d.ts.map +1 -0
  132. package/dist/commands/runtime-pruning.js +355 -0
  133. package/dist/commands/runtime-pruning.js.map +1 -0
  134. package/dist/commands/runtime-recovery.d.ts +9 -0
  135. package/dist/commands/runtime-recovery.d.ts.map +1 -0
  136. package/dist/commands/runtime-recovery.js +94 -0
  137. package/dist/commands/runtime-recovery.js.map +1 -0
  138. package/dist/commands/runtime-synthetic-baseline.d.ts +7 -0
  139. package/dist/commands/runtime-synthetic-baseline.d.ts.map +1 -0
  140. package/dist/commands/runtime-synthetic-baseline.js +59 -0
  141. package/dist/commands/runtime-synthetic-baseline.js.map +1 -0
  142. package/dist/commands/runtime-uat.d.ts +52 -0
  143. package/dist/commands/runtime-uat.d.ts.map +1 -0
  144. package/dist/commands/runtime-uat.js +274 -0
  145. package/dist/commands/runtime-uat.js.map +1 -0
  146. package/dist/commands/runtime.d.ts +20 -0
  147. package/dist/commands/runtime.d.ts.map +1 -0
  148. package/dist/commands/runtime.js +256 -0
  149. package/dist/commands/runtime.js.map +1 -0
  150. package/dist/commands/samples-list.d.ts +11 -0
  151. package/dist/commands/samples-list.d.ts.map +1 -0
  152. package/dist/commands/samples-list.js +37 -0
  153. package/dist/commands/samples-list.js.map +1 -0
  154. package/dist/commands/samples-review.d.ts +14 -0
  155. package/dist/commands/samples-review.d.ts.map +1 -0
  156. package/dist/commands/samples-review.js +22 -0
  157. package/dist/commands/samples-review.js.map +1 -0
  158. package/dist/commands/task.d.ts +14 -0
  159. package/dist/commands/task.d.ts.map +1 -0
  160. package/dist/commands/task.js +92 -0
  161. package/dist/commands/task.js.map +1 -0
  162. package/dist/commands/trace.d.ts +19 -0
  163. package/dist/commands/trace.d.ts.map +1 -0
  164. package/dist/commands/trace.js +154 -0
  165. package/dist/commands/trace.js.map +1 -0
  166. package/dist/commands/trajectory.d.ts +11 -0
  167. package/dist/commands/trajectory.d.ts.map +1 -0
  168. package/dist/commands/trajectory.js +47 -0
  169. package/dist/commands/trajectory.js.map +1 -0
  170. package/dist/index.d.ts +9 -0
  171. package/dist/index.d.ts.map +1 -0
  172. package/dist/index.js +736 -0
  173. package/dist/index.js.map +1 -0
  174. package/dist/legacy/legacy-import.d.ts +15 -0
  175. package/dist/legacy/legacy-import.d.ts.map +1 -0
  176. package/dist/legacy/legacy-import.js +141 -0
  177. package/dist/legacy/legacy-import.js.map +1 -0
  178. package/dist/legacy/session-history-import.d.ts +26 -0
  179. package/dist/legacy/session-history-import.d.ts.map +1 -0
  180. package/dist/legacy/session-history-import.js +151 -0
  181. package/dist/legacy/session-history-import.js.map +1 -0
  182. package/dist/principle-tree-ledger-adapter.d.ts +12 -0
  183. package/dist/principle-tree-ledger-adapter.d.ts.map +1 -0
  184. package/dist/principle-tree-ledger-adapter.js +12 -0
  185. package/dist/principle-tree-ledger-adapter.js.map +1 -0
  186. package/dist/resolve-workspace.d.ts +12 -0
  187. package/dist/resolve-workspace.d.ts.map +1 -0
  188. package/dist/resolve-workspace.js +20 -0
  189. package/dist/resolve-workspace.js.map +1 -0
  190. package/dist/services/demo-story-a-runner.d.ts +8 -0
  191. package/dist/services/demo-story-a-runner.d.ts.map +1 -0
  192. package/dist/services/demo-story-a-runner.js +369 -0
  193. package/dist/services/demo-story-a-runner.js.map +1 -0
  194. package/dist/services/feature-flag-loader.d.ts +6 -0
  195. package/dist/services/feature-flag-loader.d.ts.map +1 -0
  196. package/dist/services/feature-flag-loader.js +54 -0
  197. package/dist/services/feature-flag-loader.js.map +1 -0
  198. package/dist/services/pain-flood-simulation-runner.d.ts +10 -0
  199. package/dist/services/pain-flood-simulation-runner.d.ts.map +1 -0
  200. package/dist/services/pain-flood-simulation-runner.js +289 -0
  201. package/dist/services/pain-flood-simulation-runner.js.map +1 -0
  202. package/dist/services/proven-channel-baseline-runner.d.ts +12 -0
  203. package/dist/services/proven-channel-baseline-runner.d.ts.map +1 -0
  204. package/dist/services/proven-channel-baseline-runner.js +114 -0
  205. package/dist/services/proven-channel-baseline-runner.js.map +1 -0
  206. package/dist/services/synthetic-baseline-runner.d.ts +8 -0
  207. package/dist/services/synthetic-baseline-runner.d.ts.map +1 -0
  208. package/dist/services/synthetic-baseline-runner.js +251 -0
  209. package/dist/services/synthetic-baseline-runner.js.map +1 -0
  210. package/package.json +35 -0
  211. package/src/commands/artifact.ts +82 -0
  212. package/src/commands/candidate.ts +1117 -0
  213. package/src/commands/central-sync.ts +44 -0
  214. package/src/commands/console.ts +121 -0
  215. package/src/commands/context.ts +72 -0
  216. package/src/commands/demo-story-a.ts +195 -0
  217. package/src/commands/diagnose.ts +452 -0
  218. package/src/commands/evolution-tasks-list.ts +44 -0
  219. package/src/commands/evolution-tasks-show.ts +60 -0
  220. package/src/commands/flow.ts +60 -0
  221. package/src/commands/health.ts +189 -0
  222. package/src/commands/history.ts +63 -0
  223. package/src/commands/legacy-cleanup.ts +206 -0
  224. package/src/commands/legacy-import.ts +104 -0
  225. package/src/commands/pain-record.ts +167 -0
  226. package/src/commands/proven-channel-baseline.ts +113 -0
  227. package/src/commands/remediation-output.ts +66 -0
  228. package/src/commands/run.ts +89 -0
  229. package/src/commands/runtime-activation.ts +176 -0
  230. package/src/commands/runtime-canary.ts +371 -0
  231. package/src/commands/runtime-diagnostics-export.ts +229 -0
  232. package/src/commands/runtime-features.ts +103 -0
  233. package/src/commands/runtime-gfi-snapshot.ts +135 -0
  234. package/src/commands/runtime-health-snapshot.ts +106 -0
  235. package/src/commands/runtime-internalization-enqueue-successors.ts +479 -0
  236. package/src/commands/runtime-internalization-integrity-repair.ts +69 -0
  237. package/src/commands/runtime-internalization-integrity.ts +63 -0
  238. package/src/commands/runtime-internalization-queue.ts +106 -0
  239. package/src/commands/runtime-internalization-run-once.ts +658 -0
  240. package/src/commands/runtime-internalization-wake-once.ts +87 -0
  241. package/src/commands/runtime-pain-flood-simulation.ts +121 -0
  242. package/src/commands/runtime-pruning.ts +438 -0
  243. package/src/commands/runtime-recovery.ts +107 -0
  244. package/src/commands/runtime-synthetic-baseline.ts +70 -0
  245. package/src/commands/runtime-uat.ts +339 -0
  246. package/src/commands/runtime.ts +281 -0
  247. package/src/commands/samples-list.ts +43 -0
  248. package/src/commands/samples-review.ts +32 -0
  249. package/src/commands/task.ts +130 -0
  250. package/src/commands/trace.ts +174 -0
  251. package/src/commands/trajectory.ts +64 -0
  252. package/src/index.ts +829 -0
  253. package/src/legacy/legacy-import.ts +179 -0
  254. package/src/legacy/session-history-import.ts +231 -0
  255. package/src/principle-tree-ledger-adapter.ts +13 -0
  256. package/src/resolve-workspace.ts +20 -0
  257. package/src/services/demo-story-a-runner.ts +472 -0
  258. package/src/services/feature-flag-loader.ts +73 -0
  259. package/src/services/pain-flood-simulation-runner.ts +354 -0
  260. package/src/services/proven-channel-baseline-runner.ts +150 -0
  261. package/src/services/synthetic-baseline-runner.ts +291 -0
  262. package/tests/commands/candidate-audit-repair.test.ts +338 -0
  263. package/tests/commands/candidate-intake.test.ts +589 -0
  264. package/tests/commands/candidate-internalization-backfill.test.ts +480 -0
  265. package/tests/commands/candidate-internalize.test.ts +272 -0
  266. package/tests/commands/candidate-route.test.ts +328 -0
  267. package/tests/commands/candidate-show.test.ts +95 -0
  268. package/tests/commands/cli-command-tree.test.ts +64 -0
  269. package/tests/commands/context.test.ts +114 -0
  270. package/tests/commands/demo-story-a.test.ts +255 -0
  271. package/tests/commands/diagnose.test.ts +792 -0
  272. package/tests/commands/health.test.ts +330 -0
  273. package/tests/commands/pain-record.test.ts +316 -0
  274. package/tests/commands/plugin-config-resolution-cutover.test.ts +220 -0
  275. package/tests/commands/proven-channel-baseline.test.ts +441 -0
  276. package/tests/commands/runtime-activation.test.ts +168 -0
  277. package/tests/commands/runtime-canary.test.ts +369 -0
  278. package/tests/commands/runtime-diagnostics-export.test.ts +170 -0
  279. package/tests/commands/runtime-features.test.ts +114 -0
  280. package/tests/commands/runtime-health-snapshot.test.ts +357 -0
  281. package/tests/commands/runtime-internalization-enqueue-successors.test.ts +803 -0
  282. package/tests/commands/runtime-internalization-integrity-repair.test.ts +169 -0
  283. package/tests/commands/runtime-internalization-integrity.test.ts +102 -0
  284. package/tests/commands/runtime-internalization-queue.test.ts +252 -0
  285. package/tests/commands/runtime-internalization-run-once.test.ts +1318 -0
  286. package/tests/commands/runtime-internalization-wake-once.test.ts +170 -0
  287. package/tests/commands/runtime-internalization.test.ts +52 -0
  288. package/tests/commands/runtime-pain-flood-simulation.test.ts +418 -0
  289. package/tests/commands/runtime-pruning.test.ts +693 -0
  290. package/tests/commands/runtime-recovery.test.ts +96 -0
  291. package/tests/commands/runtime-synthetic-baseline.test.ts +249 -0
  292. package/tests/commands/runtime-uat.test.ts +397 -0
  293. package/tests/commands/runtime.test.ts +262 -0
  294. package/tests/commands/trace.test.ts +314 -0
  295. package/tests/e2e/candidate-intake-e2e.test.ts +316 -0
  296. package/tests/services/feature-flag-loader.test.ts +207 -0
  297. package/tests/services/proven-channel-baseline-runner.test.ts +30 -0
  298. package/tsconfig.json +26 -0
@@ -0,0 +1,371 @@
1
+ import * as path from 'path';
2
+ import { OperatorHealthReadModel, SchemaConformanceReadModel, PruningReadModel, createInternalizationQueueReadModel, auditCandidateLedgerConsistency, buildGfiWorkspaceSnapshot, classifyGfiWorkspaceHealth } from '@principles/core/runtime-v2';
3
+ import type { OperatorHealthSnapshot, SchemaConformanceResult, OrphanDetectionResult, InternalizationQueueSnapshot, GfiWorkspaceSnapshot, CandidateAuditResult } from '@principles/core/runtime-v2';
4
+ import { resolveWorkspaceDir } from '../resolve-workspace.js';
5
+ import { loadEffectiveFeatureFlags } from '../services/feature-flag-loader.js';
6
+
7
+ export interface CanaryCheck {
8
+ name: string;
9
+ status: 'healthy' | 'degraded' | 'error';
10
+ summary: string;
11
+ details?: unknown;
12
+ error?: string;
13
+ }
14
+
15
+ export interface CanaryOutput {
16
+ overallStatus: 'healthy' | 'degraded' | 'error';
17
+ checks: CanaryCheck[];
18
+ recommendedNextActions: string[];
19
+ generatedAt: string;
20
+ internalizationQueueSummary?: InternalizationQueueSummary;
21
+ }
22
+
23
+ export interface InternalizationQueueSummary {
24
+ readyCount: number;
25
+ retryWaitCount: number;
26
+ pendingCount: number;
27
+ nextReadyTaskKind: string | null;
28
+ nextReadyTaskId: string | null;
29
+ noReadyReason: string | null;
30
+ }
31
+
32
+ interface CanaryOptions {
33
+ workspace?: string;
34
+ json?: boolean;
35
+ }
36
+
37
+ function computeOverallStatus(checks: CanaryCheck[]): 'healthy' | 'degraded' | 'error' {
38
+ let hasError = false;
39
+ let hasDegraded = false;
40
+ for (const check of checks) {
41
+ if (check.status === 'error') hasError = true;
42
+ if (check.status === 'degraded') hasDegraded = true;
43
+ }
44
+ if (hasError) return 'error';
45
+ if (hasDegraded) return 'degraded';
46
+ return 'healthy';
47
+ }
48
+
49
+ function buildRecommendedActions(checks: CanaryCheck[]): string[] {
50
+ const actions: string[] = [];
51
+ for (const check of checks) {
52
+ if (check.status === 'healthy') continue;
53
+ switch (check.name) {
54
+ case 'schema_conformance':
55
+ actions.push('Run workspace initialization to migrate schema: open DB with writable SqliteConnection.');
56
+ break;
57
+ case 'candidate_audit':
58
+ actions.push('Run `pd candidate audit --workspace <path> --json` for details.');
59
+ break;
60
+ case 'gfi_snapshot': {
61
+ const gfiDetails = check.details as { warnings?: string[]; configPath?: string } | undefined;
62
+ if (gfiDetails && gfiDetails.warnings && gfiDetails.warnings.length > 0) {
63
+ actions.push(`Review .pd/feature-flags.yaml and rerun \`pd runtime features --workspace <path> --json\`.`);
64
+ } else {
65
+ actions.push('Investigate GFI sessions — consider cleanup or session lifecycle review.');
66
+ }
67
+ break;
68
+ }
69
+ case 'pruning_orphans':
70
+ actions.push('Run `pd runtime pruning orphans --workspace <path> --dry-run` to inspect orphan candidates.');
71
+ break;
72
+ case 'internalization_queue': {
73
+ const queueDetails = check.details as InternalizationQueueSnapshot | undefined;
74
+ if (queueDetails && queueDetails.noReadyTasks?.reason === 'no_candidates' && queueDetails.noReadyTasks.inspectedCount === 0) {
75
+ actions.push('No internalization tasks found. If candidates exist, run `pd candidate internalization backfill --dry-run` to check, then `--confirm` to create dreamer tasks.');
76
+ } else {
77
+ actions.push('Check internalization queue for blocked or dependency-failed tasks.');
78
+ }
79
+ break;
80
+ }
81
+ case 'runtime_health': {
82
+ const details = check.details as OperatorHealthSnapshot | undefined;
83
+ if (details && details.totalTaskCount === 0) {
84
+ actions.push('Runtime V2 pipeline has never been exercised. Run `pd pain record --reason "test" --workspace <path>` to trigger the pain-to-principle chain.');
85
+ } else if (details && details.painChain.lastSuccessfulChain === null) {
86
+ actions.push('Run `pd runtime uat --workspace <path> --count 3` to establish baseline.');
87
+ } else {
88
+ actions.push('Review runtime health snapshot for specific failure categories.');
89
+ }
90
+ break;
91
+ }
92
+ case 'pd_shim_info':
93
+ actions.push('Verify pd CLI installation and sync-plugin configuration.');
94
+ break;
95
+ }
96
+ }
97
+ return actions;
98
+ }
99
+
100
+ export async function runCanaryChecks(workspaceDir: string): Promise<CanaryOutput> {
101
+ const generatedAt = new Date().toISOString();
102
+
103
+ const checkPromises: Promise<CanaryCheck>[] = [
104
+ (async (): Promise<CanaryCheck> => {
105
+ try {
106
+ const model = new SchemaConformanceReadModel({ workspaceDir });
107
+ const result: SchemaConformanceResult = model.check();
108
+ const status = result.overallStatus === 'ok' ? 'healthy' : result.overallStatus === 'degraded' ? 'degraded' : 'error';
109
+ const missingCount = Object.values(result.tables).filter(t => !t.exists || t.missingColumns.length > 0).length;
110
+ return {
111
+ name: 'schema_conformance',
112
+ status,
113
+ summary: status === 'healthy'
114
+ ? 'All tables and columns conform to expected schema.'
115
+ : `${missingCount} table(s) have missing columns or do not exist. Migrations needed: ${result.migrationsNeeded.join(', ')}`,
116
+ details: result,
117
+ };
118
+ } catch (err) {
119
+ return { name: 'schema_conformance', status: 'error', summary: 'Schema conformance check failed.', error: String(err) };
120
+ }
121
+ })(),
122
+ (async (): Promise<CanaryCheck> => {
123
+ try {
124
+ const result: CandidateAuditResult = await auditCandidateLedgerConsistency(workspaceDir);
125
+ const status = result.status === 'ok' ? 'healthy' : result.status === 'degraded' ? 'degraded' : 'error';
126
+ return {
127
+ name: 'candidate_audit',
128
+ status,
129
+ summary: status === 'healthy'
130
+ ? 'Candidate/ledger consistency OK.'
131
+ : `Audit status: ${result.status}. Orphan candidates: ${result.orphanCandidateCount}, Missing ledger: ${result.missingLedgerCount}`,
132
+ details: result,
133
+ };
134
+ } catch (err) {
135
+ return { name: 'candidate_audit', status: 'error', summary: 'Candidate audit check failed.', error: String(err) };
136
+ }
137
+ })(),
138
+ (async (): Promise<CanaryCheck> => {
139
+ try {
140
+ const featureFlags = loadEffectiveFeatureFlags(workspaceDir);
141
+ const gfiFlag = featureFlags.flags.gfi;
142
+ if (!gfiFlag || !gfiFlag.enabled) {
143
+ if (featureFlags.warnings.length > 0) {
144
+ return {
145
+ name: 'gfi_snapshot',
146
+ status: 'degraded',
147
+ summary: `GFI disabled but config has warnings: ${featureFlags.warnings.join('; ')}`,
148
+ details: { warnings: featureFlags.warnings, configPath: featureFlags.configPath },
149
+ };
150
+ }
151
+ return {
152
+ name: 'gfi_snapshot',
153
+ status: 'healthy',
154
+ summary: 'GFI feature flag disabled — skipping snapshot.',
155
+ };
156
+ }
157
+ const sessionDir = path.join(workspaceDir, '.state', 'sessions');
158
+ const fs = await import('fs');
159
+ const sessions: { sessionId: string; currentGfi: number; lastActivityAt: number; consecutiveErrors: number }[] = [];
160
+ if (fs.existsSync(sessionDir)) {
161
+ for (const file of fs.readdirSync(sessionDir)) {
162
+ if (!file.endsWith('.json')) continue;
163
+ try {
164
+ const raw = fs.readFileSync(path.join(sessionDir, file), 'utf8');
165
+ const parsed = JSON.parse(raw);
166
+ if (parsed?.sessionId) {
167
+ sessions.push({
168
+ sessionId: parsed.sessionId,
169
+ currentGfi: parsed.currentGfi ?? 0,
170
+ lastActivityAt: parsed.lastActivityAt ?? parsed.lastControlActivityAt ?? 0,
171
+ consecutiveErrors: parsed.consecutiveErrors ?? 0,
172
+ });
173
+ }
174
+ } catch { /* skip malformed */ }
175
+ }
176
+ }
177
+ const snapshot: GfiWorkspaceSnapshot = buildGfiWorkspaceSnapshot({ sessions, nowMs: Date.now() });
178
+ const health = classifyGfiWorkspaceHealth(snapshot);
179
+ const {status} = health;
180
+ return {
181
+ name: 'gfi_snapshot',
182
+ status,
183
+ summary: status === 'healthy'
184
+ ? `GFI snapshot OK. ${snapshot.activeSessionCount} active, ${snapshot.staleSessionCount} stale sessions.`
185
+ : `GFI degraded: ${health.reason}`,
186
+ details: { ...snapshot, healthAssessment: health },
187
+ };
188
+ } catch (err) {
189
+ return { name: 'gfi_snapshot', status: 'error', summary: 'GFI snapshot check failed.', error: String(err) };
190
+ }
191
+ })(),
192
+ (async (): Promise<CanaryCheck> => {
193
+ try {
194
+ const model = new PruningReadModel({ workspaceDir });
195
+ const result: OrphanDetectionResult = model.getOrphanDerivedCandidates();
196
+ const status = result.candidates.length > 0 ? 'degraded' : 'healthy';
197
+ return {
198
+ name: 'pruning_orphans',
199
+ status,
200
+ summary: status === 'healthy'
201
+ ? 'No orphan derived candidates found.'
202
+ : `${result.candidates.length} orphan derived candidate(s) found. dbReadable: ${result.dbReadable}`,
203
+ details: { orphanDerivedCandidateCount: result.candidates.length, dbReadable: result.dbReadable, samples: result.candidates.slice(0, 5) },
204
+ };
205
+ } catch (err) {
206
+ return { name: 'pruning_orphans', status: 'error', summary: 'Pruning orphan check failed.', error: String(err) };
207
+ }
208
+ })(),
209
+ (async (): Promise<CanaryCheck> => {
210
+ try {
211
+ const featureFlags = loadEffectiveFeatureFlags(workspaceDir);
212
+ const enabledChannels = new Set(
213
+ Object.values(featureFlags.flags)
214
+ .filter(f => f.enabled)
215
+ .map(f => f.id),
216
+ );
217
+ const { readModel, close } = await createInternalizationQueueReadModel({ workspaceDir, readonly: true, enabledChannels });
218
+ try {
219
+ const snapshot: InternalizationQueueSnapshot = await readModel.getSnapshot();
220
+ const hasBlocked = snapshot.blockedSummary.count > 0;
221
+ const hasDepFailed = snapshot.dependencyFailedSummary.count > 0;
222
+ const hasInvalid = snapshot.invalidMetadataCount > 0;
223
+ const hasLeaseConflicts = snapshot.leaseConflictSummary.count > 0;
224
+ // Suppressed tasks (disabled channels / non-MVP kinds) do NOT degrade health
225
+ const status = (hasBlocked || hasDepFailed || hasInvalid || hasLeaseConflicts) ? 'degraded' : 'healthy';
226
+ const suppressedNote = snapshot.suppressedTasks.length > 0
227
+ ? ` (${snapshot.suppressedTasks.length} suppressed non-MVP)`
228
+ : '';
229
+ return {
230
+ name: 'internalization_queue',
231
+ status,
232
+ summary: status === 'healthy'
233
+ ? `Queue OK. ${snapshot.readyTasks.length} ready, ${snapshot.pendingCount} pending, ${snapshot.retryWaitCount} retry_wait${suppressedNote}.`
234
+ : `Queue degraded: ${snapshot.blockedSummary.count} blocked, ${snapshot.dependencyFailedSummary.count} dep-failed, ${snapshot.invalidMetadataCount} invalid metadata, ${snapshot.leaseConflictSummary.count} lease conflicts${suppressedNote}.`,
235
+ details: snapshot,
236
+ };
237
+ } finally {
238
+ await close();
239
+ }
240
+ } catch (err) {
241
+ return { name: 'internalization_queue', status: 'error', summary: 'Internalization queue check failed.', error: String(err) };
242
+ }
243
+ })(),
244
+ (async (): Promise<CanaryCheck> => {
245
+ try {
246
+ const model = new OperatorHealthReadModel({ workspaceDir });
247
+ try {
248
+ const snapshot: OperatorHealthSnapshot = await model.getSnapshot();
249
+ const status = snapshot.overallStatus === 'healthy' ? 'healthy' : snapshot.overallStatus === 'degraded' ? 'degraded' : 'error';
250
+ return {
251
+ name: 'runtime_health',
252
+ status,
253
+ summary: status === 'healthy'
254
+ ? 'Runtime health OK.'
255
+ : `Runtime health: ${snapshot.overallStatus}. Actions: ${snapshot.recommendedActions.join('; ')}`,
256
+ details: snapshot,
257
+ };
258
+ } finally {
259
+ await model.close();
260
+ }
261
+ } catch (err) {
262
+ return { name: 'runtime_health', status: 'error', summary: 'Runtime health check failed.', error: String(err) };
263
+ }
264
+ })(),
265
+ (async (): Promise<CanaryCheck> => {
266
+ try {
267
+ const fs = await import('fs');
268
+ const cliPath = path.resolve(process.argv[1] ?? '');
269
+ const cliExists = fs.existsSync(cliPath);
270
+ const status = cliExists ? 'healthy' : 'degraded';
271
+ return {
272
+ name: 'pd_shim_info',
273
+ status,
274
+ summary: status === 'healthy'
275
+ ? `PD CLI accessible at ${cliPath}`
276
+ : 'PD CLI entrypoint not found — verify installation.',
277
+ details: { cliPath, exists: cliExists, version: '0.1.0' },
278
+ };
279
+ } catch (err) {
280
+ return { name: 'pd_shim_info', status: 'error', summary: 'PD shim check failed.', error: String(err) };
281
+ }
282
+ })(),
283
+ ];
284
+
285
+ const checkResults = await Promise.allSettled(checkPromises);
286
+ const checks: CanaryCheck[] = checkResults.map((result, index) => {
287
+ if (result.status === 'fulfilled') return result.value;
288
+ return {
289
+ name: `check_${index}`,
290
+ status: 'error' as const,
291
+ summary: 'Check failed unexpectedly.',
292
+ error: result.reason ? String(result.reason) : 'Unknown error',
293
+ };
294
+ });
295
+
296
+ const overallStatus = computeOverallStatus(checks);
297
+ const recommendedNextActions = buildRecommendedActions(checks);
298
+
299
+ const internalizationQueueCheck = checks.find(c => c.name === 'internalization_queue');
300
+ const internalizationQueueSummary: InternalizationQueueSummary | undefined =
301
+ internalizationQueueCheck && internalizationQueueCheck.details
302
+ ? (() => {
303
+ const snapshot = internalizationQueueCheck.details as InternalizationQueueSnapshot;
304
+ const firstReady = snapshot.readyTasks[0] ?? null;
305
+ return {
306
+ readyCount: snapshot.readyTasks.length,
307
+ retryWaitCount: snapshot.retryWaitCount,
308
+ pendingCount: snapshot.pendingCount,
309
+ nextReadyTaskKind: firstReady ? firstReady.taskKind : null,
310
+ nextReadyTaskId: firstReady ? firstReady.taskId : null,
311
+ noReadyReason: snapshot.noReadyTasks?.reason ?? null,
312
+ };
313
+ })()
314
+ : undefined;
315
+
316
+ return {
317
+ overallStatus,
318
+ checks,
319
+ recommendedNextActions,
320
+ generatedAt,
321
+ ...(internalizationQueueSummary ? { internalizationQueueSummary } : {}),
322
+ };
323
+ }
324
+
325
+ function formatTextOutput(output: CanaryOutput): string {
326
+ const lines: string[] = [];
327
+ const icon = output.overallStatus === 'healthy' ? '✓' : '✗';
328
+
329
+ lines.push('PD Control Plane Canary');
330
+ lines.push(`generatedAt: ${output.generatedAt}`);
331
+ lines.push(`OVERALL: ${icon} ${output.overallStatus.toUpperCase()}`);
332
+ lines.push('');
333
+
334
+ for (const check of output.checks) {
335
+ const checkIcon = check.status === 'healthy' ? '✓' : check.status === 'degraded' ? '⚠' : '✗';
336
+ lines.push(` ${checkIcon} ${check.name}: ${check.summary}`);
337
+ if (check.error) {
338
+ lines.push(` error: ${check.error}`);
339
+ }
340
+ }
341
+
342
+ if (output.recommendedNextActions.length > 0) {
343
+ lines.push('');
344
+ lines.push('Recommended next actions:');
345
+ for (const action of output.recommendedNextActions) {
346
+ lines.push(` [!] ${action}`);
347
+ }
348
+ }
349
+
350
+ return lines.join('\n');
351
+ }
352
+
353
+ export async function handleRuntimeCanary(opts: CanaryOptions): Promise<void> {
354
+ const workspaceDir = opts.workspace
355
+ ? path.resolve(opts.workspace)
356
+ : resolveWorkspaceDir();
357
+
358
+ const output = await runCanaryChecks(workspaceDir);
359
+
360
+ if (opts.json) {
361
+ console.log(JSON.stringify(output, null, 2));
362
+ } else {
363
+ console.log(formatTextOutput(output));
364
+ }
365
+
366
+ if (output.overallStatus !== 'healthy') {
367
+ console.error('');
368
+ console.error(`FAIL: overallStatus=${output.overallStatus}`);
369
+ process.exitCode = 1;
370
+ }
371
+ }
@@ -0,0 +1,229 @@
1
+ import * as path from 'path';
2
+ import * as fs from 'fs';
3
+ import {
4
+ OperatorHealthReadModel,
5
+ SchemaConformanceReadModel,
6
+ PruningReadModel,
7
+ createInternalizationQueueReadModel,
8
+ InternalizationChainIntegrityReadModel,
9
+ auditCandidateLedgerConsistency,
10
+ buildGfiWorkspaceSnapshot,
11
+ } from '@principles/core/runtime-v2';
12
+ import { runCanaryChecks } from './runtime-canary.js';
13
+ import { resolveWorkspaceDir } from '../resolve-workspace.js';
14
+
15
+ export interface BundleManifestArtifact {
16
+ name: string;
17
+ path: string;
18
+ status: 'ok' | 'failed';
19
+ error?: string;
20
+ }
21
+
22
+ export interface BundleManifest {
23
+ generatedAt: string;
24
+ workspace: string;
25
+ outputDir: string;
26
+ artifacts: BundleManifestArtifact[];
27
+ }
28
+
29
+ export interface DiagnosticsExportOptions {
30
+ workspace?: string;
31
+ out?: string;
32
+ json?: boolean;
33
+ }
34
+
35
+ function validateOutputPath(outDir: string, workspaceDir: string): string {
36
+ const resolvedOut = path.resolve(outDir);
37
+ const resolvedWs = path.resolve(workspaceDir);
38
+ const relative = path.relative(resolvedWs, resolvedOut);
39
+ if (relative.startsWith('..') || path.isAbsolute(relative)) {
40
+ throw new Error('Output path must be within workspace directory');
41
+ }
42
+ return resolvedOut;
43
+ }
44
+
45
+ function sanitizeForExport(data: unknown): unknown {
46
+ if (data === null || data === undefined) return data;
47
+ if (typeof data === 'string') {
48
+ return data.replace(/(?:api[_-]?key|secret|token|password|authorization|bearer)\s*[:=]\s*\S+/gi, '[REDACTED]');
49
+ }
50
+ if (Array.isArray(data)) return data.map(sanitizeForExport);
51
+ if (typeof data === 'object') {
52
+ const result: Record<string, unknown> = {};
53
+ for (const [key, value] of Object.entries(data as Record<string, unknown>)) {
54
+ const lowerKey = key.toLowerCase();
55
+ if (/key|secret|token|password|auth|bearer|credential/i.test(lowerKey)) {
56
+ result[key] = '[REDACTED]';
57
+ } else {
58
+ result[key] = sanitizeForExport(value);
59
+ }
60
+ }
61
+ return result;
62
+ }
63
+ return data;
64
+ }
65
+
66
+ interface CollectArtifactContext {
67
+ name: string;
68
+ fileName: string;
69
+ outputDir: string;
70
+ collector: () => Promise<unknown> | unknown;
71
+ artifacts: BundleManifestArtifact[];
72
+ errors: { artifact: string; error: string }[];
73
+ }
74
+
75
+ async function collectArtifact(ctx: CollectArtifactContext): Promise<void> {
76
+ try {
77
+ const data = await ctx.collector();
78
+ const sanitized = sanitizeForExport(data);
79
+ const filePath = path.join(ctx.outputDir, ctx.fileName);
80
+ fs.writeFileSync(filePath, JSON.stringify(sanitized, null, 2), 'utf8');
81
+ ctx.artifacts.push({ name: ctx.name, path: ctx.fileName, status: 'ok' });
82
+ } catch (err) {
83
+ ctx.artifacts.push({ name: ctx.name, path: ctx.fileName, status: 'failed', error: String(err) });
84
+ ctx.errors.push({ artifact: ctx.name, error: String(err) });
85
+ }
86
+ }
87
+
88
+ export async function exportDiagnosticsBundle(workspaceDir: string, outDir: string): Promise<BundleManifest> {
89
+ const generatedAt = new Date().toISOString();
90
+ const resolvedOut = validateOutputPath(outDir, workspaceDir);
91
+ const artifacts: BundleManifestArtifact[] = [];
92
+ const errors: { artifact: string; error: string }[] = [];
93
+
94
+ fs.mkdirSync(resolvedOut, { recursive: true });
95
+
96
+ const baseCtx = { outputDir: resolvedOut, artifacts, errors };
97
+
98
+ await collectArtifact({ ...baseCtx, name: 'runtime-health', fileName: 'runtime-health.json', collector: async () => {
99
+ const model = new OperatorHealthReadModel({ workspaceDir });
100
+ try {
101
+ return await model.getSnapshot();
102
+ } finally {
103
+ await model.close();
104
+ }
105
+ } });
106
+
107
+ await collectArtifact({ ...baseCtx, name: 'canary', fileName: 'canary.json', collector: async () => {
108
+ return await runCanaryChecks(workspaceDir);
109
+ } });
110
+
111
+ await collectArtifact({ ...baseCtx, name: 'schema-conformance', fileName: 'schema-conformance.json', collector: () => {
112
+ const model = new SchemaConformanceReadModel({ workspaceDir });
113
+ return model.check();
114
+ } });
115
+
116
+ await collectArtifact({ ...baseCtx, name: 'candidate-audit', fileName: 'candidate-audit.json', collector: async () => {
117
+ return await auditCandidateLedgerConsistency(workspaceDir);
118
+ } });
119
+
120
+ await collectArtifact({ ...baseCtx, name: 'gfi-snapshot', fileName: 'gfi-snapshot.json', collector: () => {
121
+ const sessionDir = path.join(workspaceDir, '.state', 'sessions');
122
+ const sessions: { sessionId: string; currentGfi: number; lastActivityAt: number; consecutiveErrors: number }[] = [];
123
+ if (fs.existsSync(sessionDir)) {
124
+ for (const file of fs.readdirSync(sessionDir)) {
125
+ if (!file.endsWith('.json')) continue;
126
+ try {
127
+ const raw = fs.readFileSync(path.join(sessionDir, file), 'utf8');
128
+ const parsed = JSON.parse(raw);
129
+ if (parsed?.sessionId) {
130
+ sessions.push({
131
+ sessionId: parsed.sessionId,
132
+ currentGfi: parsed.currentGfi ?? 0,
133
+ lastActivityAt: parsed.lastActivityAt ?? parsed.lastControlActivityAt ?? 0,
134
+ consecutiveErrors: parsed.consecutiveErrors ?? 0,
135
+ });
136
+ }
137
+ } catch { /* skip malformed */ }
138
+ }
139
+ }
140
+ return buildGfiWorkspaceSnapshot({ sessions, nowMs: Date.now() });
141
+ } });
142
+
143
+ await collectArtifact({ ...baseCtx, name: 'pruning-orphans', fileName: 'pruning-orphans.json', collector: () => {
144
+ const model = new PruningReadModel({ workspaceDir });
145
+ return model.getOrphanDerivedCandidates();
146
+ } });
147
+
148
+ await collectArtifact({ ...baseCtx, name: 'internalization-queue', fileName: 'internalization-queue.json', collector: async () => {
149
+ const { readModel, close } = await createInternalizationQueueReadModel({ workspaceDir, readonly: true });
150
+ try {
151
+ return await readModel.getSnapshot();
152
+ } finally {
153
+ await close();
154
+ }
155
+ } });
156
+
157
+ await collectArtifact({ ...baseCtx, name: 'internalization-integrity', fileName: 'internalization-integrity.json', collector: () => {
158
+ const model = new InternalizationChainIntegrityReadModel({ workspaceDir });
159
+ return model.check();
160
+ } });
161
+
162
+ await collectArtifact({ ...baseCtx, name: 'recent-session-summary', fileName: 'recent-session-summary.json', collector: () => {
163
+ const sessionDir = path.join(workspaceDir, '.state', 'sessions');
164
+ const summaries: { sessionId: string; status: string; lastActivityAt: number }[] = [];
165
+ if (fs.existsSync(sessionDir)) {
166
+ for (const file of fs.readdirSync(sessionDir)) {
167
+ if (!file.endsWith('.json')) continue;
168
+ try {
169
+ const raw = fs.readFileSync(path.join(sessionDir, file), 'utf8');
170
+ const parsed = JSON.parse(raw);
171
+ if (parsed?.sessionId) {
172
+ summaries.push({
173
+ sessionId: parsed.sessionId,
174
+ status: parsed.status ?? 'unknown',
175
+ lastActivityAt: parsed.lastActivityAt ?? parsed.lastControlActivityAt ?? 0,
176
+ });
177
+ }
178
+ } catch { /* skip */ }
179
+ }
180
+ }
181
+ return { sessionCount: summaries.length, sessions: summaries.slice(0, 20) };
182
+ } });
183
+
184
+ if (errors.length > 0) {
185
+ await collectArtifact({ outputDir: resolvedOut, artifacts, errors: [], name: 'errors', fileName: 'errors.json', collector: () => errors });
186
+ }
187
+
188
+ const manifest: BundleManifest = {
189
+ generatedAt,
190
+ workspace: workspaceDir,
191
+ outputDir: resolvedOut,
192
+ artifacts,
193
+ };
194
+
195
+ fs.writeFileSync(
196
+ path.join(resolvedOut, 'manifest.json'),
197
+ JSON.stringify(manifest, null, 2),
198
+ 'utf8',
199
+ );
200
+
201
+ return manifest;
202
+ }
203
+
204
+ export async function handleRuntimeDiagnosticsExport(opts: DiagnosticsExportOptions): Promise<void> {
205
+ const workspaceDir = opts.workspace
206
+ ? path.resolve(opts.workspace)
207
+ : resolveWorkspaceDir();
208
+
209
+ const outDir = opts.out
210
+ ? path.resolve(workspaceDir, opts.out)
211
+ : path.resolve(workspaceDir, '.state', 'control-plane-observation', 'snapshots');
212
+
213
+ const manifest = await exportDiagnosticsBundle(workspaceDir, outDir);
214
+
215
+ if (opts.json) {
216
+ console.log(JSON.stringify(manifest, null, 2));
217
+ } else {
218
+ console.log(`Diagnostic bundle exported to: ${manifest.outputDir}`);
219
+ console.log(`Artifacts: ${manifest.artifacts.length}`);
220
+ for (const artifact of manifest.artifacts) {
221
+ const icon = artifact.status === 'ok' ? '✓' : '✗';
222
+ console.log(` ${icon} ${artifact.name} (${artifact.path})`);
223
+ }
224
+ const failedCount = manifest.artifacts.filter(a => a.status === 'failed').length;
225
+ if (failedCount > 0) {
226
+ console.log(`\n${failedCount} artifact(s) failed to generate. See errors.json for details.`);
227
+ }
228
+ }
229
+ }