avorelo 0.1.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 (258) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +56 -0
  3. package/bin/avorelo +9 -0
  4. package/package.json +135 -0
  5. package/scripts/README.md +40 -0
  6. package/scripts/cco-dashboard.js +252 -0
  7. package/scripts/cco-status.js +430 -0
  8. package/scripts/lib/activation/account-state.js +37 -0
  9. package/scripts/lib/activation/activation-runner.js +546 -0
  10. package/scripts/lib/activation/activation-self-healing.js +480 -0
  11. package/scripts/lib/activation/activation-state.js +83 -0
  12. package/scripts/lib/activation/activation-summary.js +191 -0
  13. package/scripts/lib/activation/adapters/claude-code.js +77 -0
  14. package/scripts/lib/activation/adapters/codex-cli.js +52 -0
  15. package/scripts/lib/activation/adapters/cursor.js +37 -0
  16. package/scripts/lib/activation/adapters/github-agent.js +39 -0
  17. package/scripts/lib/activation/adapters/terminal.js +42 -0
  18. package/scripts/lib/activation/adapters/vscode.js +39 -0
  19. package/scripts/lib/activation/adapters/windsurf.js +37 -0
  20. package/scripts/lib/activation/ai-surface-detector.js +151 -0
  21. package/scripts/lib/activation/connect-account.js +145 -0
  22. package/scripts/lib/activation/detect-environment.js +75 -0
  23. package/scripts/lib/activation/detect-hosts.js +62 -0
  24. package/scripts/lib/activation/format-activation-output.js +109 -0
  25. package/scripts/lib/activation/next-action.js +43 -0
  26. package/scripts/lib/activation/repair-engine.js +219 -0
  27. package/scripts/lib/activation-distribution-readiness.js +507 -0
  28. package/scripts/lib/adapter-conformance.js +176 -0
  29. package/scripts/lib/adapter-readiness.js +417 -0
  30. package/scripts/lib/adapter-safety-boundaries.js +335 -0
  31. package/scripts/lib/adapter-technical-readiness-gate.js +205 -0
  32. package/scripts/lib/agent-access-governance.js +455 -0
  33. package/scripts/lib/agent-enforcement.js +765 -0
  34. package/scripts/lib/agent-policy-profile.js +210 -0
  35. package/scripts/lib/agent-security/action-evaluator.js +507 -0
  36. package/scripts/lib/agent-security/adapter-registry.js +98 -0
  37. package/scripts/lib/agent-security/auto-policy.js +139 -0
  38. package/scripts/lib/agent-security/bounded-scan.js +93 -0
  39. package/scripts/lib/agent-security/enforcement-adapter.js +174 -0
  40. package/scripts/lib/agent-security/enforcement-engine.js +1129 -0
  41. package/scripts/lib/agent-security/file-write-adapter.js +183 -0
  42. package/scripts/lib/agent-security/file-write-rules.js +178 -0
  43. package/scripts/lib/agent-security/index.js +3342 -0
  44. package/scripts/lib/agent-security/instruction-risk.js +181 -0
  45. package/scripts/lib/agent-security/mcp-action-adapter.js +185 -0
  46. package/scripts/lib/agent-security/mcp-action-rules.js +184 -0
  47. package/scripts/lib/agent-security/package-action-adapter.js +175 -0
  48. package/scripts/lib/agent-security/package-action-rules.js +233 -0
  49. package/scripts/lib/agent-security/performance.js +148 -0
  50. package/scripts/lib/agent-security/permission-minimizer.js +403 -0
  51. package/scripts/lib/agent-security/scan-cache.js +74 -0
  52. package/scripts/lib/agent-security/source-trust.js +146 -0
  53. package/scripts/lib/ai-install-prompt.js +288 -0
  54. package/scripts/lib/ai-workspace-hygiene.js +1499 -0
  55. package/scripts/lib/alpha-activation.js +520 -0
  56. package/scripts/lib/alpha-feedback.js +263 -0
  57. package/scripts/lib/alpha-readiness-gate.js +332 -0
  58. package/scripts/lib/anti-gaming.js +169 -0
  59. package/scripts/lib/artifact-health.js +431 -0
  60. package/scripts/lib/attribution.js +180 -0
  61. package/scripts/lib/audit.js +289 -0
  62. package/scripts/lib/avorelo-skill-registry.js +810 -0
  63. package/scripts/lib/batch-jobs.js +71 -0
  64. package/scripts/lib/brain-pack.js +578 -0
  65. package/scripts/lib/brand-boundary.js +424 -0
  66. package/scripts/lib/brand.js +74 -0
  67. package/scripts/lib/browser-capability.js +1048 -0
  68. package/scripts/lib/browser-proof-preflight.js +321 -0
  69. package/scripts/lib/cache-readiness.js +187 -0
  70. package/scripts/lib/canonical-reentry.js +162 -0
  71. package/scripts/lib/capability-packs.js +314 -0
  72. package/scripts/lib/capability-recommender.js +512 -0
  73. package/scripts/lib/capability-registry.js +1059 -0
  74. package/scripts/lib/carry-forward-surfacing.js +194 -0
  75. package/scripts/lib/ccusage-adapter.js +188 -0
  76. package/scripts/lib/company-loop.js +1149 -0
  77. package/scripts/lib/config.js +637 -0
  78. package/scripts/lib/context-acquisition-plan.js +287 -0
  79. package/scripts/lib/context-budget-guard.js +170 -0
  80. package/scripts/lib/context-budget-scanner.js +257 -0
  81. package/scripts/lib/context-optimizer.js +715 -0
  82. package/scripts/lib/context-reduction-plan.js +178 -0
  83. package/scripts/lib/context-safety.js +88 -0
  84. package/scripts/lib/context-savings-engine.js +158 -0
  85. package/scripts/lib/cost-evidence.js +254 -0
  86. package/scripts/lib/cross-host-install-plan.js +308 -0
  87. package/scripts/lib/cross-host-install-readiness.js +237 -0
  88. package/scripts/lib/cross-host-value-flow.js +268 -0
  89. package/scripts/lib/dashboard.js +900 -0
  90. package/scripts/lib/design-partner-feedback.js +346 -0
  91. package/scripts/lib/entitlements.js +100 -0
  92. package/scripts/lib/execution-packet.js +559 -0
  93. package/scripts/lib/experimentation-events.js +547 -0
  94. package/scripts/lib/external-capability-compliance.js +107 -0
  95. package/scripts/lib/external-user-simulation.js +166 -0
  96. package/scripts/lib/failure-recovery-readiness.js +81 -0
  97. package/scripts/lib/failure-recovery.js +419 -0
  98. package/scripts/lib/feedback-intelligence.js +537 -0
  99. package/scripts/lib/feedback-signals.js +205 -0
  100. package/scripts/lib/file-integrity.js +68 -0
  101. package/scripts/lib/fsx.js +127 -0
  102. package/scripts/lib/full-readiness-gate.js +451 -0
  103. package/scripts/lib/guidance-builder.js +174 -0
  104. package/scripts/lib/hook-apply.js +1019 -0
  105. package/scripts/lib/hook-baseline.js +310 -0
  106. package/scripts/lib/hook-config-preview.js +275 -0
  107. package/scripts/lib/hook-contracts.js +290 -0
  108. package/scripts/lib/hook-safety-boundary-readiness.js +80 -0
  109. package/scripts/lib/host-capability-matrix.js +351 -0
  110. package/scripts/lib/host-support-context.js +254 -0
  111. package/scripts/lib/http-hook-action.js +538 -0
  112. package/scripts/lib/install-ai-readiness.js +84 -0
  113. package/scripts/lib/install-intake-risk.js +1037 -0
  114. package/scripts/lib/install-journey-intelligence.js +329 -0
  115. package/scripts/lib/intervention-guidance.js +57 -0
  116. package/scripts/lib/known-limitations.js +115 -0
  117. package/scripts/lib/l8-path-truth.js +146 -0
  118. package/scripts/lib/launch-hardening-gate.js +436 -0
  119. package/scripts/lib/launch-readiness.js +628 -0
  120. package/scripts/lib/learning-memory.js +686 -0
  121. package/scripts/lib/lifecycle-hooks.js +802 -0
  122. package/scripts/lib/local-package-smoke.js +423 -0
  123. package/scripts/lib/local-pricing.js +299 -0
  124. package/scripts/lib/mcp-enforcement.js +311 -0
  125. package/scripts/lib/mcp-least-privilege-policy.js +303 -0
  126. package/scripts/lib/mcp-tool-inventory.js +388 -0
  127. package/scripts/lib/mcp-tool-risk.js +0 -0
  128. package/scripts/lib/memory.js +335 -0
  129. package/scripts/lib/metrics.js +699 -0
  130. package/scripts/lib/micro-proof.js +133 -0
  131. package/scripts/lib/next-run-context.js +436 -0
  132. package/scripts/lib/operating-value.js +1648 -0
  133. package/scripts/lib/optimization-v3.js +122 -0
  134. package/scripts/lib/orchestration/adapters/_shared.js +49 -0
  135. package/scripts/lib/orchestration/adapters/aider.js +18 -0
  136. package/scripts/lib/orchestration/adapters/claude-code.js +35 -0
  137. package/scripts/lib/orchestration/adapters/codex.js +35 -0
  138. package/scripts/lib/orchestration/adapters/gemini-cli.js +18 -0
  139. package/scripts/lib/orchestration/adapters/git.js +25 -0
  140. package/scripts/lib/orchestration/adapters/index.js +31 -0
  141. package/scripts/lib/orchestration/adapters/lm-studio.js +18 -0
  142. package/scripts/lib/orchestration/adapters/ollama.js +18 -0
  143. package/scripts/lib/orchestration/adapters/opencode.js +18 -0
  144. package/scripts/lib/orchestration/adapters/openrouter.js +18 -0
  145. package/scripts/lib/orchestration/adapters/test-runner.js +25 -0
  146. package/scripts/lib/orchestration/cli.js +438 -0
  147. package/scripts/lib/orchestration/execution-manager.js +279 -0
  148. package/scripts/lib/orchestration/handoff.js +314 -0
  149. package/scripts/lib/orchestration/index.js +456 -0
  150. package/scripts/lib/orchestration/inventory.js +47 -0
  151. package/scripts/lib/orchestration/model-discovery.js +498 -0
  152. package/scripts/lib/orchestration/model-profiler.js +170 -0
  153. package/scripts/lib/orchestration/model-profiles.js +252 -0
  154. package/scripts/lib/orchestration/model-refresh-policy.js +72 -0
  155. package/scripts/lib/orchestration/proof-writer.js +349 -0
  156. package/scripts/lib/orchestration/provider-discovery/aider.js +49 -0
  157. package/scripts/lib/orchestration/provider-discovery/claude-code.js +56 -0
  158. package/scripts/lib/orchestration/provider-discovery/codex.js +49 -0
  159. package/scripts/lib/orchestration/provider-discovery/common.js +186 -0
  160. package/scripts/lib/orchestration/provider-discovery/gemini.js +106 -0
  161. package/scripts/lib/orchestration/provider-discovery/lm-studio.js +118 -0
  162. package/scripts/lib/orchestration/provider-discovery/models-dev.js +12 -0
  163. package/scripts/lib/orchestration/provider-discovery/ollama.js +100 -0
  164. package/scripts/lib/orchestration/provider-discovery/opencode.js +47 -0
  165. package/scripts/lib/orchestration/provider-discovery/openrouter.js +44 -0
  166. package/scripts/lib/orchestration/risk-classifier.js +130 -0
  167. package/scripts/lib/orchestration/routing-policy.js +486 -0
  168. package/scripts/lib/orchestration/settings.js +112 -0
  169. package/scripts/lib/orchestration/state.js +165 -0
  170. package/scripts/lib/orchestration/verification-manager.js +138 -0
  171. package/scripts/lib/output-profiles.js +146 -0
  172. package/scripts/lib/package-content-audit.js +368 -0
  173. package/scripts/lib/package-runtime.js +278 -0
  174. package/scripts/lib/plan-surface.js +53 -0
  175. package/scripts/lib/plans.js +2318 -0
  176. package/scripts/lib/policy-provider.js +27 -0
  177. package/scripts/lib/prelaunch-activation-readiness.js +409 -0
  178. package/scripts/lib/prelaunch-evidence-store.js +816 -0
  179. package/scripts/lib/prelaunch-intelligence.js +869 -0
  180. package/scripts/lib/pricing-experiment.js +118 -0
  181. package/scripts/lib/pro-moment-events.js +77 -0
  182. package/scripts/lib/pro-moment-state.js +227 -0
  183. package/scripts/lib/pro-moments.js +1216 -0
  184. package/scripts/lib/product-learning-events.js +629 -0
  185. package/scripts/lib/project-profile.js +555 -0
  186. package/scripts/lib/prompt-compiler.js +280 -0
  187. package/scripts/lib/prompt-lint.js +32 -0
  188. package/scripts/lib/prompt-suggestions.js +52 -0
  189. package/scripts/lib/proof-canonical.js +398 -0
  190. package/scripts/lib/proof-drilldown.js +383 -0
  191. package/scripts/lib/proof-events.js +342 -0
  192. package/scripts/lib/proof-history.js +243 -0
  193. package/scripts/lib/proof-metrics.js +296 -0
  194. package/scripts/lib/proof-outcome-evidence.js +134 -0
  195. package/scripts/lib/proof-receipt.js +335 -0
  196. package/scripts/lib/proof-record.js +461 -0
  197. package/scripts/lib/public-activation-distribution-gate.js +258 -0
  198. package/scripts/lib/public-cli.js +3891 -0
  199. package/scripts/lib/public-distribution-truth.js +211 -0
  200. package/scripts/lib/public-install-claim-checker.js +294 -0
  201. package/scripts/lib/publish-provenance-readiness.js +283 -0
  202. package/scripts/lib/readiness-delta.js +218 -0
  203. package/scripts/lib/readiness-evidence-closure.js +196 -0
  204. package/scripts/lib/reentry-memory-capture.js +241 -0
  205. package/scripts/lib/reentry-memory-retrieval.js +302 -0
  206. package/scripts/lib/reentry-memory-status.js +146 -0
  207. package/scripts/lib/reentry-memory-store.js +178 -0
  208. package/scripts/lib/reentry-state.js +66 -0
  209. package/scripts/lib/release-candidate-bundle.js +166 -0
  210. package/scripts/lib/remediation.js +81 -0
  211. package/scripts/lib/repo-map.js +391 -0
  212. package/scripts/lib/run-improvements-lifecycle.js +330 -0
  213. package/scripts/lib/run-improvements.js +789 -0
  214. package/scripts/lib/runtime-decision-policy.js +387 -0
  215. package/scripts/lib/safe-path-engine.js +705 -0
  216. package/scripts/lib/safe-run-controller.js +887 -0
  217. package/scripts/lib/score.js +262 -0
  218. package/scripts/lib/seamless-enforcement.js +329 -0
  219. package/scripts/lib/seamless-outcome.js +689 -0
  220. package/scripts/lib/seamless-reality-gate.js +5043 -0
  221. package/scripts/lib/security-risk-classifier.js +511 -0
  222. package/scripts/lib/security-scan.js +384 -0
  223. package/scripts/lib/session-context-optimizer.js +1211 -0
  224. package/scripts/lib/session-timing.js +315 -0
  225. package/scripts/lib/skill-hygiene.js +805 -0
  226. package/scripts/lib/skill-packs.js +161 -0
  227. package/scripts/lib/skills-operating-layer.js +580 -0
  228. package/scripts/lib/smart-work-routing.js +768 -0
  229. package/scripts/lib/source-catalog.js +700 -0
  230. package/scripts/lib/status-value-summary.js +32 -0
  231. package/scripts/lib/support-bundle.js +578 -0
  232. package/scripts/lib/task-continuation.js +440 -0
  233. package/scripts/lib/test-helpers.js +15 -0
  234. package/scripts/lib/tier.js +38 -0
  235. package/scripts/lib/token-context-quality-gate.js +370 -0
  236. package/scripts/lib/token-cost-capture.js +187 -0
  237. package/scripts/lib/token-cost-intelligence.js +358 -0
  238. package/scripts/lib/token-efficiency-evidence.js +213 -0
  239. package/scripts/lib/token-evidence.js +699 -0
  240. package/scripts/lib/tokenish.js +17 -0
  241. package/scripts/lib/tool-output-sandbox.js +304 -0
  242. package/scripts/lib/trust-audit.js +136 -0
  243. package/scripts/lib/unified-events.js +396 -0
  244. package/scripts/lib/upgrade-interruption-recovery.js +407 -0
  245. package/scripts/lib/usage-ledger.js +201 -0
  246. package/scripts/lib/value-ledger.js +130 -0
  247. package/scripts/lib/value-proof-calibration.js +531 -0
  248. package/scripts/lib/visual-qa.js +231 -0
  249. package/scripts/lib/voice-alpha.js +29 -0
  250. package/scripts/lib/work-aware-orchestration.js +976 -0
  251. package/scripts/lib/work-control-receipts.js +577 -0
  252. package/scripts/lib/work-ledger.js +1123 -0
  253. package/scripts/lib/work-panel-preview.js +352 -0
  254. package/scripts/lib/workflow-discipline.js +280 -0
  255. package/scripts/lib/workflow-signals.js +419 -0
  256. package/scripts/lib/workspace-map.js +281 -0
  257. package/scripts/lib/workspace-registry.js +1367 -0
  258. package/scripts/lib/workspace-resolver.js +480 -0
@@ -0,0 +1,342 @@
1
+ "use strict";
2
+
3
+ /**
4
+ * Proof Event Taxonomy Mapping
5
+ *
6
+ * Maps existing Wuz reason codes and metric events to the 7 proof event families
7
+ * and 4 proof dimensions. This is a mapping layer, not a replacement for existing events.
8
+ */
9
+
10
+ const PROOF_DIMENSIONS = ["outcome", "efficiency", "trust", "control"];
11
+
12
+ const PROOF_EVENT_FAMILIES = [
13
+ "context_optimization",
14
+ "tool_model_efficiency",
15
+ "safety_control",
16
+ "recovery",
17
+ "trust",
18
+ "outcome",
19
+ "ux_exposure",
20
+ ];
21
+
22
+ /**
23
+ * Maps reason codes to proof event family and dimension.
24
+ * Each entry: { family, dimension, proofWorthy, calmCopy }
25
+ *
26
+ * proofWorthy: true if this event should surface as a micro-proof moment.
27
+ * calmCopy: the user-facing one-liner if surfaced as micro-proof.
28
+ */
29
+ const REASON_CODE_PROOF_MAP = {
30
+ // Context optimization -> efficiency
31
+ CTX_LARGE_OUTPUT: {
32
+ family: "context_optimization",
33
+ dimension: "efficiency",
34
+ proofWorthy: true,
35
+ calmCopy: "Wuz stored a large output out-of-band to reduce context pressure.",
36
+ },
37
+ CTX_SCAN_SKIPPED_UNCHANGED: {
38
+ family: "context_optimization",
39
+ dimension: "efficiency",
40
+ proofWorthy: false,
41
+ calmCopy: null,
42
+ },
43
+ CTX_SNAPSHOT: {
44
+ family: "context_optimization",
45
+ dimension: "efficiency",
46
+ proofWorthy: true,
47
+ calmCopy: "Wuz preserved a context snapshot before compaction.",
48
+ },
49
+ CTX_COST_PRESSURE_SUBAGENT: {
50
+ family: "context_optimization",
51
+ dimension: "efficiency",
52
+ proofWorthy: false,
53
+ calmCopy: null,
54
+ },
55
+
56
+ // Tool/model efficiency -> efficiency
57
+ LOOP_REPEATED_TOOL: {
58
+ family: "tool_model_efficiency",
59
+ dimension: "efficiency",
60
+ proofWorthy: true,
61
+ calmCopy: "Wuz interrupted a repeated tool loop to prevent wasted cycles.",
62
+ },
63
+ FAIL_RETRY_GUARD: {
64
+ family: "recovery",
65
+ dimension: "trust",
66
+ proofWorthy: true,
67
+ calmCopy: "Wuz prevented a blind retry after a tool failure.",
68
+ },
69
+
70
+ // Safety/control -> control
71
+ SEC_CURL_PIPE_SH: {
72
+ family: "safety_control",
73
+ dimension: "control",
74
+ proofWorthy: true,
75
+ calmCopy: "Wuz blocked a remote script piped into shell.",
76
+ },
77
+ SEC_REMOTE_FETCH: {
78
+ family: "safety_control",
79
+ dimension: "control",
80
+ proofWorthy: true,
81
+ calmCopy: "Wuz flagged a remote fetch pattern for review.",
82
+ },
83
+ SEC_REMOTE_INCLUDE: {
84
+ family: "safety_control",
85
+ dimension: "control",
86
+ proofWorthy: true,
87
+ calmCopy: "Wuz flagged a remote include pattern for review.",
88
+ },
89
+ SEC_UNPINNED_GIT_DEP: {
90
+ family: "safety_control",
91
+ dimension: "control",
92
+ proofWorthy: false,
93
+ calmCopy: null,
94
+ },
95
+ SEC_OVERBROAD_PERMISSION: {
96
+ family: "safety_control",
97
+ dimension: "control",
98
+ proofWorthy: false,
99
+ calmCopy: null,
100
+ },
101
+ SEC_SKILL_SOURCE_UNKNOWN: {
102
+ family: "safety_control",
103
+ dimension: "control",
104
+ proofWorthy: true,
105
+ calmCopy: "Wuz flagged an unknown skill source for review.",
106
+ },
107
+ SEC_SKILL_SOURCE_UNAPPROVED: {
108
+ family: "safety_control",
109
+ dimension: "control",
110
+ proofWorthy: true,
111
+ calmCopy: "Wuz blocked an unapproved skill source.",
112
+ },
113
+ SEC_SKILL_INTAKE_BLOCK: {
114
+ family: "safety_control",
115
+ dimension: "control",
116
+ proofWorthy: true,
117
+ calmCopy: "Wuz blocked a skill that did not pass intake policy.",
118
+ },
119
+ SEC_SKILL_INTAKE_WARN: {
120
+ family: "safety_control",
121
+ dimension: "control",
122
+ proofWorthy: false,
123
+ calmCopy: null,
124
+ },
125
+ SEC_SKILL_INTAKE_ALLOW: {
126
+ family: "safety_control",
127
+ dimension: "control",
128
+ proofWorthy: false,
129
+ calmCopy: null,
130
+ },
131
+ SEC_PRERUN_SCAN_STALE: {
132
+ family: "safety_control",
133
+ dimension: "control",
134
+ proofWorthy: false,
135
+ calmCopy: null,
136
+ },
137
+ SEC_HTTP_ENDPOINT_BLOCKED: {
138
+ family: "safety_control",
139
+ dimension: "control",
140
+ proofWorthy: true,
141
+ calmCopy: "Wuz blocked a request to an unsafe HTTP endpoint.",
142
+ },
143
+ SEC_HTTP_PRIVATE_NETWORK_BLOCKED: {
144
+ family: "safety_control",
145
+ dimension: "control",
146
+ proofWorthy: true,
147
+ calmCopy: "Wuz blocked a request to a private network address.",
148
+ },
149
+ SEC_OUTPUT_SENSITIVE_DATA: {
150
+ family: "safety_control",
151
+ dimension: "control",
152
+ proofWorthy: true,
153
+ calmCopy: "Wuz detected a sensitive data pattern in tool output.",
154
+ },
155
+ TEAM_BLOCK_PATTERN: {
156
+ family: "safety_control",
157
+ dimension: "control",
158
+ proofWorthy: true,
159
+ calmCopy: "Wuz blocked a command matching team policy.",
160
+ },
161
+
162
+ // Permission -> control
163
+ PERM_FATIGUE_CONSOLIDATED: {
164
+ family: "safety_control",
165
+ dimension: "control",
166
+ proofWorthy: true,
167
+ calmCopy: "Wuz consolidated repeated safe permission requests.",
168
+ },
169
+ PERM_REQ_DENY: {
170
+ family: "safety_control",
171
+ dimension: "control",
172
+ proofWorthy: true,
173
+ calmCopy: "Wuz denied a permission request based on policy.",
174
+ },
175
+ PERM_REQ_ASK: {
176
+ family: "safety_control",
177
+ dimension: "control",
178
+ proofWorthy: false,
179
+ calmCopy: null,
180
+ },
181
+
182
+ // Trust -> trust
183
+ SEC_REMEDIATION_JOB_CREATED: {
184
+ family: "trust",
185
+ dimension: "trust",
186
+ proofWorthy: true,
187
+ calmCopy: "Wuz generated a remediation job for a security finding.",
188
+ },
189
+ SEC_SUPPRESSION_AUDIT: {
190
+ family: "trust",
191
+ dimension: "trust",
192
+ proofWorthy: false,
193
+ calmCopy: null,
194
+ },
195
+ SEC_TRUST_UNKNOWN_HIGH: {
196
+ family: "trust",
197
+ dimension: "trust",
198
+ proofWorthy: true,
199
+ calmCopy: "Wuz flagged high-risk changes in trusted-surface files.",
200
+ },
201
+ SEC_TRUST_AUDIT: {
202
+ family: "trust",
203
+ dimension: "trust",
204
+ proofWorthy: false,
205
+ calmCopy: null,
206
+ },
207
+
208
+ // Recovery -> trust
209
+ STOP_SNAPSHOT: {
210
+ family: "recovery",
211
+ dimension: "trust",
212
+ proofWorthy: true,
213
+ calmCopy: "Wuz preserved progress for safe re-entry.",
214
+ },
215
+
216
+ // Outcome -> outcome
217
+ TASK_COMPLETED: {
218
+ family: "outcome",
219
+ dimension: "outcome",
220
+ proofWorthy: false,
221
+ calmCopy: null,
222
+ },
223
+ REPORT_WRITTEN: {
224
+ family: "outcome",
225
+ dimension: "outcome",
226
+ proofWorthy: false,
227
+ calmCopy: null,
228
+ },
229
+
230
+ // UX exposure (proof-system-specific events)
231
+ PROOF_MICRO_RENDERED: {
232
+ family: "ux_exposure",
233
+ dimension: "outcome",
234
+ proofWorthy: false,
235
+ calmCopy: null,
236
+ },
237
+ PROOF_RECEIPT_GENERATED: {
238
+ family: "ux_exposure",
239
+ dimension: "outcome",
240
+ proofWorthy: false,
241
+ calmCopy: null,
242
+ },
243
+ PROOF_DRILLDOWN_OPENED: {
244
+ family: "ux_exposure",
245
+ dimension: "outcome",
246
+ proofWorthy: false,
247
+ calmCopy: null,
248
+ },
249
+ };
250
+
251
+ /**
252
+ * Classify a reason code into its proof dimension.
253
+ * Returns null if the code is unknown.
254
+ */
255
+ function classifyReasonCode(code) {
256
+ const entry = REASON_CODE_PROOF_MAP[code];
257
+ if (!entry) return null;
258
+ return {
259
+ code,
260
+ family: entry.family,
261
+ dimension: entry.dimension,
262
+ proofWorthy: entry.proofWorthy,
263
+ calmCopy: entry.calmCopy,
264
+ };
265
+ }
266
+
267
+ /**
268
+ * Classify all reason codes from a metric entry.
269
+ * Returns array of classified codes (unknown codes are excluded).
270
+ */
271
+ function classifyMetricReasonCodes(metric) {
272
+ const codes = metric?.reasonCodes || [];
273
+ return codes.map(classifyReasonCode).filter(Boolean);
274
+ }
275
+
276
+ /**
277
+ * Extract proof-worthy events from a list of metrics.
278
+ * Returns only events that should surface to the user.
279
+ */
280
+ function extractProofWorthyEvents(metrics) {
281
+ const results = [];
282
+ for (const metric of metrics || []) {
283
+ const classified = classifyMetricReasonCodes(metric);
284
+ for (const c of classified) {
285
+ if (c.proofWorthy) {
286
+ results.push({
287
+ ...c,
288
+ ts: metric.ts,
289
+ sessionId: metric.sessionId,
290
+ event: metric.event,
291
+ tool: metric.tool,
292
+ action: metric.action,
293
+ });
294
+ }
295
+ }
296
+ }
297
+ return results;
298
+ }
299
+
300
+ /**
301
+ * Count metrics by proof dimension.
302
+ */
303
+ function countByDimension(metrics) {
304
+ const counts = { outcome: 0, efficiency: 0, trust: 0, control: 0 };
305
+ for (const metric of metrics || []) {
306
+ const classified = classifyMetricReasonCodes(metric);
307
+ for (const c of classified) {
308
+ if (counts[c.dimension] !== undefined) {
309
+ counts[c.dimension]++;
310
+ }
311
+ }
312
+ }
313
+ return counts;
314
+ }
315
+
316
+ /**
317
+ * Count metrics by event family.
318
+ */
319
+ function countByFamily(metrics) {
320
+ const counts = {};
321
+ for (const family of PROOF_EVENT_FAMILIES) counts[family] = 0;
322
+ for (const metric of metrics || []) {
323
+ const classified = classifyMetricReasonCodes(metric);
324
+ for (const c of classified) {
325
+ if (counts[c.family] !== undefined) {
326
+ counts[c.family]++;
327
+ }
328
+ }
329
+ }
330
+ return counts;
331
+ }
332
+
333
+ module.exports = {
334
+ PROOF_DIMENSIONS,
335
+ PROOF_EVENT_FAMILIES,
336
+ REASON_CODE_PROOF_MAP,
337
+ classifyReasonCode,
338
+ classifyMetricReasonCodes,
339
+ extractProofWorthyEvents,
340
+ countByDimension,
341
+ countByFamily,
342
+ };
@@ -0,0 +1,243 @@
1
+ "use strict";
2
+
3
+ const fs = require("fs");
4
+ const path = require("path");
5
+
6
+ const { ensureCcoDirs, nowIso } = require("./fsx");
7
+ const { getHistoryRetention, canExportReports } = require("./entitlements");
8
+ const { redactPayload } = require("./product-learning-events");
9
+
10
+ const PROOF_RECEIPTS_REL = ".claude/cco/receipts";
11
+ const PROOF_EXPORTS_REL = ".claude/cco/exports";
12
+ const DEFAULT_HISTORY_LIMIT = 10;
13
+
14
+ function tsOrFallback(value, fallback = 0) {
15
+ const parsed = Date.parse(value || "");
16
+ return Number.isFinite(parsed) ? parsed : fallback;
17
+ }
18
+
19
+ function redactFreeformText(value) {
20
+ const text = String(value || "");
21
+ const sanitized = redactPayload({ text }).text;
22
+ return sanitized
23
+ .replace(/[A-Za-z]:\\[^\s,;]+/g, "[redacted-path]")
24
+ .replace(/(^|[\s(])\/[A-Za-z0-9._\-\/]+/g, (match, prefix) => `${prefix}[redacted-path]`)
25
+ .replace(/(^|[\s(])\.claude\/[A-Za-z0-9._\-\/]+/g, (match, prefix) => `${prefix}[redacted-path]`)
26
+ .replace(/(^|[\s(])artifacts\/[A-Za-z0-9._\-\/]+/g, (match, prefix) => `${prefix}[redacted-path]`)
27
+ .replace(/(^|[\s(])\.env(?:\.[A-Za-z0-9._-]+)?/g, (match, prefix) => `${prefix}[redacted-path]`);
28
+ }
29
+
30
+ function safeSummary(value) {
31
+ const text = redactFreeformText(value);
32
+ if (!text) return null;
33
+ return text.length > 240 ? `${text.slice(0, 237)}...` : text;
34
+ }
35
+
36
+ function projectReceiptHistoryEntry(filename, receipt, stat) {
37
+ const result = receipt?.result || {};
38
+ const trustControl = receipt?.trust_control || {};
39
+ return {
40
+ receiptId: safeSummary(receipt?.session_id || filename.replace(/\.json$/i, "")),
41
+ endedAt: receipt?.ended_at || receipt?.started_at || new Date(stat.mtimeMs).toISOString(),
42
+ status: safeSummary(result.status || "unknown"),
43
+ summary: safeSummary(result.summary || "Proof receipt recorded."),
44
+ nextAction: safeSummary(
45
+ result.next_bounded_action
46
+ || trustControl.next_bounded_action
47
+ || null,
48
+ ),
49
+ claimBoundary: safeSummary(result.claim_boundary || trustControl.claim_boundary || null),
50
+ material: receipt?.material === true,
51
+ source: "proof_receipt",
52
+ redacted: true,
53
+ };
54
+ }
55
+
56
+ function listReceiptFiles(cwd) {
57
+ const dir = path.join(cwd, PROOF_RECEIPTS_REL);
58
+ try {
59
+ return fs.readdirSync(dir)
60
+ .filter((file) => file.endsWith(".json"))
61
+ .map((file) => {
62
+ const abs = path.join(dir, file);
63
+ return {
64
+ file,
65
+ abs,
66
+ stat: fs.statSync(abs),
67
+ };
68
+ });
69
+ } catch {
70
+ return [];
71
+ }
72
+ }
73
+
74
+ function readProofHistory(cwd, options = {}) {
75
+ const retention = getHistoryRetention(options.plan);
76
+ const maxReports = Math.max(1, Number(retention.maxReports || 1));
77
+ const limit = Math.max(
78
+ 1,
79
+ Math.min(
80
+ Number(options.limit || DEFAULT_HISTORY_LIMIT),
81
+ maxReports,
82
+ ),
83
+ );
84
+ const receipts = listReceiptFiles(cwd)
85
+ .sort((left, right) => right.stat.mtimeMs - left.stat.mtimeMs);
86
+
87
+ const entries = [];
88
+ let corruptCount = 0;
89
+
90
+ for (const receiptFile of receipts) {
91
+ if (entries.length >= maxReports) break;
92
+ try {
93
+ const parsed = JSON.parse(fs.readFileSync(receiptFile.abs, "utf8"));
94
+ entries.push(projectReceiptHistoryEntry(receiptFile.file, parsed, receiptFile.stat));
95
+ } catch {
96
+ corruptCount += 1;
97
+ }
98
+ }
99
+
100
+ entries.sort((left, right) => tsOrFallback(right.endedAt) - tsOrFallback(left.endedAt));
101
+
102
+ return {
103
+ redacted: true,
104
+ sourcePath: PROOF_RECEIPTS_REL,
105
+ retention,
106
+ totalReceiptsScanned: receipts.length,
107
+ corruptReceipts: corruptCount,
108
+ partial: corruptCount > 0,
109
+ available: entries.length > 0,
110
+ entries: entries.slice(0, limit),
111
+ totalRetainedEntries: entries.length,
112
+ };
113
+ }
114
+
115
+ function formatProofHistoryText(history, options = {}) {
116
+ if (!history?.available) {
117
+ return "Proof history is not available yet.";
118
+ }
119
+
120
+ const limit = Math.max(1, Number(options.limit || history.entries.length || 1));
121
+ const lines = [
122
+ "Avorelo Proof History",
123
+ `Entries shown: ${Math.min(history.entries.length, limit)} of ${history.totalRetainedEntries}`,
124
+ ];
125
+
126
+ for (const entry of history.entries.slice(0, limit)) {
127
+ lines.push(
128
+ `- ${entry.receiptId} | ${entry.status} | ${entry.endedAt}`,
129
+ );
130
+ lines.push(` Summary: ${entry.summary}`);
131
+ if (entry.nextAction) {
132
+ lines.push(` Next: ${entry.nextAction}`);
133
+ }
134
+ }
135
+
136
+ if (history.partial) {
137
+ lines.push("Caveat: Some local proof receipts were unreadable and were skipped.");
138
+ }
139
+
140
+ lines.push("Caveat: History is redacted and limited to safe local receipt summaries.");
141
+ return lines.join("\n");
142
+ }
143
+
144
+ function formatProofExportText(result) {
145
+ const lines = [
146
+ "Avorelo Proof Export",
147
+ `JSON: ${result.jsonRel}`,
148
+ `Markdown: ${result.mdRel}`,
149
+ `Entries exported: ${result.count}`,
150
+ ];
151
+ if (result.partial) {
152
+ lines.push("Caveat: Some local proof receipts were unreadable and were skipped.");
153
+ }
154
+ lines.push("Caveat: Export is redacted and excludes raw prompts, code, secrets, and sensitive paths.");
155
+ return lines.join("\n");
156
+ }
157
+
158
+ function buildProofHistoryMarkdown(history) {
159
+ const lines = [
160
+ "# Avorelo Proof History Export",
161
+ "",
162
+ `Generated: ${nowIso()}`,
163
+ `Entries: ${history.entries.length}`,
164
+ "",
165
+ ];
166
+
167
+ for (const entry of history.entries) {
168
+ lines.push(`## ${entry.receiptId}`);
169
+ lines.push(`- Status: ${entry.status}`);
170
+ lines.push(`- Ended: ${entry.endedAt}`);
171
+ lines.push(`- Summary: ${entry.summary}`);
172
+ if (entry.nextAction) {
173
+ lines.push(`- Next: ${entry.nextAction}`);
174
+ }
175
+ if (entry.claimBoundary) {
176
+ lines.push(`- Claim boundary: ${entry.claimBoundary}`);
177
+ }
178
+ lines.push("");
179
+ }
180
+
181
+ lines.push("Caveat: Export is redacted and excludes raw prompts, code, secrets, and sensitive paths.");
182
+ return lines.join("\n");
183
+ }
184
+
185
+ function exportProofHistoryBundle(cwd, options = {}) {
186
+ const plan = options.plan || "free";
187
+ if (!canExportReports(plan)) {
188
+ return {
189
+ ok: false,
190
+ reason: "plan_lacks_export",
191
+ };
192
+ }
193
+
194
+ const history = readProofHistory(cwd, {
195
+ plan,
196
+ limit: options.limit || getHistoryRetention(plan).maxReports || DEFAULT_HISTORY_LIMIT,
197
+ });
198
+ const stamp = nowIso().replace(/[:.]/g, "-");
199
+ const jsonRel = path.posix.join(PROOF_EXPORTS_REL, `proof-history-${stamp}.json`);
200
+ const mdRel = path.posix.join(PROOF_EXPORTS_REL, `proof-history-${stamp}.md`);
201
+ const jsonAbs = path.join(cwd, jsonRel);
202
+ const mdAbs = path.join(cwd, mdRel);
203
+
204
+ ensureCcoDirs(cwd);
205
+ fs.mkdirSync(path.dirname(jsonAbs), { recursive: true });
206
+
207
+ fs.writeFileSync(jsonAbs, JSON.stringify({
208
+ version: 1,
209
+ generatedAt: nowIso(),
210
+ plan,
211
+ redacted: true,
212
+ sourcePath: PROOF_RECEIPTS_REL,
213
+ count: history.entries.length,
214
+ retention: history.retention,
215
+ partial: history.partial,
216
+ caveats: [
217
+ "Redacted local proof history export.",
218
+ "No raw prompts, raw code, secrets, or sensitive paths are included.",
219
+ ],
220
+ entries: history.entries,
221
+ }, null, 2), "utf8");
222
+
223
+ fs.writeFileSync(mdAbs, buildProofHistoryMarkdown(history), "utf8");
224
+
225
+ return {
226
+ ok: true,
227
+ count: history.entries.length,
228
+ partial: history.partial,
229
+ jsonRel,
230
+ mdRel,
231
+ history,
232
+ };
233
+ }
234
+
235
+ module.exports = {
236
+ DEFAULT_HISTORY_LIMIT,
237
+ PROOF_EXPORTS_REL,
238
+ PROOF_RECEIPTS_REL,
239
+ exportProofHistoryBundle,
240
+ formatProofExportText,
241
+ formatProofHistoryText,
242
+ readProofHistory,
243
+ };