oxe-cc 1.0.0 → 1.3.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 (322) hide show
  1. package/.cursor/commands/oxe-ask.md +3 -3
  2. package/.cursor/commands/oxe-capabilities.md +3 -3
  3. package/.cursor/commands/oxe-checkpoint.md +3 -3
  4. package/.cursor/commands/oxe-compact.md +3 -3
  5. package/.cursor/commands/oxe-dashboard.md +3 -3
  6. package/.cursor/commands/oxe-debug.md +3 -3
  7. package/.cursor/commands/oxe-discuss.md +3 -3
  8. package/.cursor/commands/oxe-execute.md +7 -4
  9. package/.cursor/commands/oxe-forensics.md +3 -3
  10. package/.cursor/commands/oxe-help.md +3 -3
  11. package/.cursor/commands/oxe-loop.md +3 -3
  12. package/.cursor/commands/oxe-milestone.md +3 -3
  13. package/.cursor/commands/oxe-next.md +3 -3
  14. package/.cursor/commands/oxe-obs.md +3 -3
  15. package/.cursor/commands/oxe-plan-agent.md +3 -3
  16. package/.cursor/commands/oxe-plan.md +3 -3
  17. package/.cursor/commands/oxe-project.md +3 -3
  18. package/.cursor/commands/oxe-quick.md +3 -3
  19. package/.cursor/commands/oxe-research.md +3 -3
  20. package/.cursor/commands/oxe-retro.md +3 -3
  21. package/.cursor/commands/oxe-review-pr.md +3 -3
  22. package/.cursor/commands/oxe-route.md +3 -3
  23. package/.cursor/commands/oxe-scan.md +3 -3
  24. package/.cursor/commands/oxe-security.md +3 -3
  25. package/.cursor/commands/oxe-session.md +4 -4
  26. package/.cursor/commands/oxe-ship.md +45 -0
  27. package/.cursor/commands/oxe-skill.md +3 -3
  28. package/.cursor/commands/oxe-spec.md +3 -3
  29. package/.cursor/commands/oxe-ui-review.md +3 -3
  30. package/.cursor/commands/oxe-ui-spec.md +3 -3
  31. package/.cursor/commands/oxe-update.md +3 -3
  32. package/.cursor/commands/oxe-validate-gaps.md +3 -3
  33. package/.cursor/commands/oxe-verify.md +6 -3
  34. package/.cursor/commands/oxe-workstream.md +3 -3
  35. package/.cursor/commands/oxe.md +6 -6
  36. package/.github/copilot-instructions.md +94 -4
  37. package/.github/prompts/oxe-ask.prompt.md +3 -3
  38. package/.github/prompts/oxe-capabilities.prompt.md +3 -3
  39. package/.github/prompts/oxe-checkpoint.prompt.md +3 -3
  40. package/.github/prompts/oxe-compact.prompt.md +3 -3
  41. package/.github/prompts/oxe-dashboard.prompt.md +3 -3
  42. package/.github/prompts/oxe-debug.prompt.md +3 -3
  43. package/.github/prompts/oxe-discuss.prompt.md +3 -3
  44. package/.github/prompts/oxe-execute.prompt.md +7 -4
  45. package/.github/prompts/oxe-forensics.prompt.md +3 -3
  46. package/.github/prompts/oxe-help.prompt.md +3 -3
  47. package/.github/prompts/oxe-loop.prompt.md +3 -3
  48. package/.github/prompts/oxe-milestone.prompt.md +3 -3
  49. package/.github/prompts/oxe-next.prompt.md +3 -3
  50. package/.github/prompts/oxe-obs.prompt.md +3 -3
  51. package/.github/prompts/oxe-plan-agent.prompt.md +3 -3
  52. package/.github/prompts/oxe-plan.prompt.md +3 -3
  53. package/.github/prompts/oxe-project.prompt.md +3 -3
  54. package/.github/prompts/oxe-quick.prompt.md +3 -3
  55. package/.github/prompts/oxe-research.prompt.md +3 -3
  56. package/.github/prompts/oxe-retro.prompt.md +3 -3
  57. package/.github/prompts/oxe-review-pr.prompt.md +3 -3
  58. package/.github/prompts/oxe-route.prompt.md +3 -3
  59. package/.github/prompts/oxe-scan.prompt.md +3 -3
  60. package/.github/prompts/oxe-security.prompt.md +3 -3
  61. package/.github/prompts/oxe-session.prompt.md +4 -4
  62. package/.github/prompts/oxe-ship.prompt.md +45 -0
  63. package/.github/prompts/oxe-skill.prompt.md +3 -3
  64. package/.github/prompts/oxe-spec.prompt.md +3 -3
  65. package/.github/prompts/oxe-ui-review.prompt.md +3 -3
  66. package/.github/prompts/oxe-ui-spec.prompt.md +3 -3
  67. package/.github/prompts/oxe-update.prompt.md +3 -3
  68. package/.github/prompts/oxe-validate-gaps.prompt.md +3 -3
  69. package/.github/prompts/oxe-verify.prompt.md +6 -3
  70. package/.github/prompts/oxe-workstream.prompt.md +3 -3
  71. package/.github/prompts/oxe.prompt.md +5 -5
  72. package/AGENTS.md +43 -28
  73. package/CHANGELOG.md +193 -0
  74. package/README.md +610 -529
  75. package/bin/banner.txt +1 -1
  76. package/bin/lib/oxe-agent-install.cjs +69 -69
  77. package/bin/lib/oxe-azure.cjs +1445 -1445
  78. package/bin/lib/oxe-context-engine.cjs +867 -867
  79. package/bin/lib/oxe-dashboard.cjs +76 -28
  80. package/bin/lib/oxe-operational.cjs +2144 -1340
  81. package/bin/lib/oxe-project-health.cjs +483 -1
  82. package/bin/lib/oxe-runtime-semantics.cjs +12 -0
  83. package/bin/oxe-cc.js +554 -152
  84. package/commands/oxe/ask.md +7 -3
  85. package/commands/oxe/capabilities.md +2 -2
  86. package/commands/oxe/checkpoint.md +3 -3
  87. package/commands/oxe/compact.md +3 -3
  88. package/commands/oxe/dashboard.md +2 -2
  89. package/commands/oxe/debug.md +3 -3
  90. package/commands/oxe/discuss.md +2 -2
  91. package/commands/oxe/execute.md +7 -4
  92. package/commands/oxe/forensics.md +3 -3
  93. package/commands/oxe/help.md +2 -2
  94. package/commands/oxe/loop.md +3 -3
  95. package/commands/oxe/milestone.md +3 -3
  96. package/commands/oxe/next.md +3 -3
  97. package/commands/oxe/obs.md +3 -3
  98. package/commands/oxe/oxe.md +5 -5
  99. package/commands/oxe/plan-agent.md +2 -2
  100. package/commands/oxe/plan.md +2 -2
  101. package/commands/oxe/project.md +3 -3
  102. package/commands/oxe/quick.md +2 -2
  103. package/commands/oxe/research.md +3 -3
  104. package/commands/oxe/retro.md +3 -3
  105. package/commands/oxe/review-pr.md +3 -3
  106. package/commands/oxe/route.md +3 -3
  107. package/commands/oxe/scan.md +3 -3
  108. package/commands/oxe/security.md +3 -3
  109. package/commands/oxe/session.md +4 -4
  110. package/commands/oxe/ship.md +49 -0
  111. package/commands/oxe/skill.md +2 -2
  112. package/commands/oxe/spec.md +4 -4
  113. package/commands/oxe/ui-review.md +3 -3
  114. package/commands/oxe/ui-spec.md +3 -3
  115. package/commands/oxe/update.md +2 -2
  116. package/commands/oxe/validate-gaps.md +3 -3
  117. package/commands/oxe/verify.md +7 -4
  118. package/commands/oxe/workstream.md +3 -3
  119. package/lib/runtime/audit/audit-trail.d.ts +71 -0
  120. package/lib/runtime/audit/audit-trail.js +154 -0
  121. package/lib/runtime/audit/index.d.ts +2 -0
  122. package/lib/runtime/audit/index.js +18 -0
  123. package/lib/runtime/audit/policy-pack.d.ts +15 -0
  124. package/lib/runtime/audit/policy-pack.js +57 -0
  125. package/lib/runtime/context/context-pack-builder.d.ts +15 -0
  126. package/lib/runtime/context/context-pack-builder.js +42 -0
  127. package/lib/runtime/context/context-pack-store.d.ts +38 -0
  128. package/lib/runtime/context/context-pack-store.js +142 -0
  129. package/lib/runtime/context/context-profiles.d.ts +11 -0
  130. package/lib/runtime/context/context-profiles.js +51 -0
  131. package/lib/runtime/context/index.d.ts +2 -0
  132. package/lib/runtime/context/index.js +2 -0
  133. package/lib/runtime/decision/decision-engine.d.ts +43 -0
  134. package/lib/runtime/decision/decision-engine.js +127 -0
  135. package/lib/runtime/decision/decision-memo.d.ts +53 -0
  136. package/lib/runtime/decision/decision-memo.js +173 -0
  137. package/lib/runtime/decision/index.d.ts +2 -0
  138. package/lib/runtime/decision/index.js +18 -0
  139. package/lib/runtime/delivery/branch-manager.d.ts +1 -0
  140. package/lib/runtime/delivery/branch-manager.js +7 -0
  141. package/lib/runtime/delivery/ci-checks.js +34 -1
  142. package/lib/runtime/delivery/delivery-records.d.ts +34 -0
  143. package/lib/runtime/delivery/delivery-records.js +48 -0
  144. package/lib/runtime/delivery/index.d.ts +2 -0
  145. package/lib/runtime/delivery/index.js +2 -0
  146. package/lib/runtime/delivery/promotion-pipeline.d.ts +63 -0
  147. package/lib/runtime/delivery/promotion-pipeline.js +224 -0
  148. package/lib/runtime/gate/gate-manager.d.ts +41 -0
  149. package/lib/runtime/gate/gate-manager.js +108 -1
  150. package/lib/runtime/index.d.ts +5 -2
  151. package/lib/runtime/index.js +7 -1
  152. package/lib/runtime/models/gate-decision.d.ts +4 -1
  153. package/lib/runtime/models/workspace.d.ts +3 -0
  154. package/lib/runtime/plugins/capability-adapter.d.ts +12 -0
  155. package/lib/runtime/plugins/capability-adapter.js +204 -0
  156. package/lib/runtime/plugins/capability-matrix.d.ts +25 -0
  157. package/lib/runtime/plugins/capability-matrix.js +90 -0
  158. package/lib/runtime/plugins/index.d.ts +3 -0
  159. package/lib/runtime/plugins/index.js +3 -0
  160. package/lib/runtime/plugins/plugin-abi.d.ts +2 -0
  161. package/lib/runtime/plugins/plugin-manifest.d.ts +22 -0
  162. package/lib/runtime/plugins/plugin-manifest.js +95 -0
  163. package/lib/runtime/plugins/plugin-registry.d.ts +46 -0
  164. package/lib/runtime/plugins/plugin-registry.js +84 -2
  165. package/lib/runtime/policy/policy-engine.d.ts +47 -1
  166. package/lib/runtime/policy/policy-engine.js +172 -9
  167. package/lib/runtime/projection/projection-engine.d.ts +9 -1
  168. package/lib/runtime/projection/projection-engine.js +73 -3
  169. package/lib/runtime/reducers/run-state-reducer.d.ts +26 -0
  170. package/lib/runtime/reducers/run-state-reducer.js +117 -1
  171. package/lib/runtime/scheduler/agent-registry.d.ts +44 -0
  172. package/lib/runtime/scheduler/agent-registry.js +96 -0
  173. package/lib/runtime/scheduler/agent-roles.d.ts +54 -0
  174. package/lib/runtime/scheduler/agent-roles.js +62 -0
  175. package/lib/runtime/scheduler/index.d.ts +3 -0
  176. package/lib/runtime/scheduler/index.js +3 -0
  177. package/lib/runtime/scheduler/multi-agent-coordinator.d.ts +45 -1
  178. package/lib/runtime/scheduler/multi-agent-coordinator.js +234 -35
  179. package/lib/runtime/scheduler/run-journal.d.ts +18 -0
  180. package/lib/runtime/scheduler/run-journal.js +54 -0
  181. package/lib/runtime/scheduler/scheduler.d.ts +29 -1
  182. package/lib/runtime/scheduler/scheduler.js +387 -14
  183. package/lib/runtime/verification/index.d.ts +1 -0
  184. package/lib/runtime/verification/index.js +1 -0
  185. package/lib/runtime/verification/verification-compiler.d.ts +43 -0
  186. package/lib/runtime/verification/verification-compiler.js +137 -0
  187. package/lib/runtime/verification/verification-manifest.d.ts +67 -0
  188. package/lib/runtime/verification/verification-manifest.js +179 -0
  189. package/lib/runtime/workspace/strategies/ephemeral-container.d.ts +1 -0
  190. package/lib/runtime/workspace/strategies/ephemeral-container.js +4 -0
  191. package/lib/runtime/workspace/strategies/git-worktree.d.ts +1 -0
  192. package/lib/runtime/workspace/strategies/git-worktree.js +2 -0
  193. package/lib/runtime/workspace/strategies/inplace.d.ts +1 -0
  194. package/lib/runtime/workspace/strategies/inplace.js +2 -0
  195. package/lib/runtime/workspace/workspace-manager.d.ts +2 -1
  196. package/lib/sdk/README.md +9 -9
  197. package/lib/sdk/index.cjs +33 -24
  198. package/lib/sdk/index.d.ts +149 -14
  199. package/oxe/templates/ACTIVE-RUN.template.json +32 -32
  200. package/oxe/templates/CAPABILITIES.template.md +7 -7
  201. package/oxe/templates/CAPABILITY.template.md +45 -45
  202. package/oxe/templates/CHECKPOINTS.template.md +7 -7
  203. package/oxe/templates/EXECUTION-RUNTIME.template.md +68 -68
  204. package/oxe/templates/HYPOTHESES.template.md +33 -33
  205. package/oxe/templates/LESSONS-METRICS.template.json +13 -13
  206. package/oxe/templates/NOTES.template.md +16 -16
  207. package/oxe/templates/PLAN-REVIEW.template.md +31 -31
  208. package/oxe/templates/SESSION.template.md +34 -34
  209. package/oxe/templates/SKILL.template.md +26 -26
  210. package/oxe/templates/STATE.md +55 -55
  211. package/oxe/templates/WORKFLOW_AUTHORING.md +18 -18
  212. package/oxe/workflows/ask.md +96 -92
  213. package/oxe/workflows/capabilities.md +25 -25
  214. package/oxe/workflows/checkpoint.md +14 -10
  215. package/oxe/workflows/dashboard.md +33 -33
  216. package/oxe/workflows/debug.md +19 -15
  217. package/oxe/workflows/discuss.md +12 -12
  218. package/oxe/workflows/execute.md +44 -2
  219. package/oxe/workflows/forensics.md +13 -9
  220. package/oxe/workflows/help.md +352 -304
  221. package/oxe/workflows/loop.md +17 -13
  222. package/oxe/workflows/next.md +22 -22
  223. package/oxe/workflows/obs.md +4 -0
  224. package/oxe/workflows/oxe.md +64 -31
  225. package/oxe/workflows/plan-agent.md +9 -9
  226. package/oxe/workflows/project.md +6 -1
  227. package/oxe/workflows/quick.md +10 -10
  228. package/oxe/workflows/references/reasoning-discovery.md +28 -28
  229. package/oxe/workflows/references/reasoning-execution.md +29 -29
  230. package/oxe/workflows/references/reasoning-planning.md +32 -32
  231. package/oxe/workflows/references/reasoning-review.md +29 -29
  232. package/oxe/workflows/references/reasoning-status.md +24 -24
  233. package/oxe/workflows/references/robustness-elevation.md +295 -295
  234. package/oxe/workflows/references/workflow-runtime-contracts.json +952 -907
  235. package/oxe/workflows/research.md +32 -28
  236. package/oxe/workflows/retro.md +4 -0
  237. package/oxe/workflows/review-pr.md +15 -11
  238. package/oxe/workflows/route.md +16 -16
  239. package/oxe/workflows/scan.md +4 -0
  240. package/oxe/workflows/security.md +14 -10
  241. package/oxe/workflows/session.md +213 -197
  242. package/oxe/workflows/ship.md +142 -0
  243. package/oxe/workflows/skill.md +44 -44
  244. package/oxe/workflows/spec.md +15 -0
  245. package/oxe/workflows/ui-review.md +20 -16
  246. package/oxe/workflows/ui-spec.md +7 -3
  247. package/oxe/workflows/validate-gaps.md +13 -9
  248. package/oxe/workflows/verify-audit.md +73 -73
  249. package/oxe/workflows/verify.md +52 -3
  250. package/package.json +92 -92
  251. package/packages/runtime/package.json +17 -17
  252. package/packages/runtime/src/audit/audit-trail.ts +243 -0
  253. package/packages/runtime/src/audit/index.ts +2 -0
  254. package/packages/runtime/src/audit/policy-pack.ts +62 -0
  255. package/packages/runtime/src/compiler/graph-compiler.ts +245 -245
  256. package/packages/runtime/src/compiler/index.ts +1 -1
  257. package/packages/runtime/src/context/context-pack-builder.ts +259 -193
  258. package/packages/runtime/src/context/context-pack-store.ts +197 -0
  259. package/packages/runtime/src/context/context-profiles.ts +60 -0
  260. package/packages/runtime/src/context/index.ts +3 -1
  261. package/packages/runtime/src/decision/decision-engine.ts +174 -0
  262. package/packages/runtime/src/decision/decision-memo.ts +211 -0
  263. package/packages/runtime/src/decision/index.ts +2 -0
  264. package/packages/runtime/src/delivery/branch-manager.ts +91 -84
  265. package/packages/runtime/src/delivery/ci-checks.ts +285 -252
  266. package/packages/runtime/src/delivery/delivery-records.ts +75 -0
  267. package/packages/runtime/src/delivery/index.ts +5 -3
  268. package/packages/runtime/src/delivery/pr-manager.ts +112 -112
  269. package/packages/runtime/src/delivery/promotion-pipeline.ts +334 -0
  270. package/packages/runtime/src/events/bus.ts +92 -92
  271. package/packages/runtime/src/events/catalog.ts +29 -29
  272. package/packages/runtime/src/events/envelope.ts +14 -14
  273. package/packages/runtime/src/events/index.ts +3 -3
  274. package/packages/runtime/src/evidence/evidence-store.ts +130 -130
  275. package/packages/runtime/src/evidence/index.ts +1 -1
  276. package/packages/runtime/src/gate/gate-manager.ts +289 -137
  277. package/packages/runtime/src/gate/index.ts +1 -1
  278. package/packages/runtime/src/index.ts +41 -32
  279. package/packages/runtime/src/models/attempt.ts +19 -19
  280. package/packages/runtime/src/models/evidence.ts +21 -21
  281. package/packages/runtime/src/models/gate-decision.ts +25 -21
  282. package/packages/runtime/src/models/index.ts +8 -8
  283. package/packages/runtime/src/models/run.ts +24 -24
  284. package/packages/runtime/src/models/session.ts +11 -11
  285. package/packages/runtime/src/models/verification-result.ts +10 -10
  286. package/packages/runtime/src/models/work-item.ts +25 -25
  287. package/packages/runtime/src/models/workspace.ts +31 -28
  288. package/packages/runtime/src/plugins/capability-adapter.ts +206 -0
  289. package/packages/runtime/src/plugins/capability-matrix.ts +126 -0
  290. package/packages/runtime/src/plugins/index.ts +5 -2
  291. package/packages/runtime/src/plugins/plugin-abi.ts +97 -95
  292. package/packages/runtime/src/plugins/plugin-manifest.ts +118 -0
  293. package/packages/runtime/src/plugins/plugin-registry.ts +232 -119
  294. package/packages/runtime/src/policy/index.ts +1 -1
  295. package/packages/runtime/src/policy/policy-engine.ts +330 -113
  296. package/packages/runtime/src/projection/index.ts +1 -1
  297. package/packages/runtime/src/projection/projection-engine.ts +328 -249
  298. package/packages/runtime/src/reducers/debug-reducer.ts +36 -36
  299. package/packages/runtime/src/reducers/index.ts +2 -2
  300. package/packages/runtime/src/reducers/run-state-reducer.ts +269 -127
  301. package/packages/runtime/src/scheduler/agent-registry.ts +132 -0
  302. package/packages/runtime/src/scheduler/agent-roles.ts +109 -0
  303. package/packages/runtime/src/scheduler/index.ts +4 -1
  304. package/packages/runtime/src/scheduler/multi-agent-coordinator.ts +521 -231
  305. package/packages/runtime/src/scheduler/run-journal.ts +62 -0
  306. package/packages/runtime/src/scheduler/scheduler.ts +722 -281
  307. package/packages/runtime/src/verification/index.ts +2 -1
  308. package/packages/runtime/src/verification/verification-compiler.ts +436 -225
  309. package/packages/runtime/src/verification/verification-manifest.ts +252 -0
  310. package/packages/runtime/src/workspace/index.ts +5 -5
  311. package/packages/runtime/src/workspace/strategies/ephemeral-container.ts +126 -121
  312. package/packages/runtime/src/workspace/strategies/git-worktree.ts +79 -77
  313. package/packages/runtime/src/workspace/strategies/inplace.ts +38 -35
  314. package/packages/runtime/src/workspace/workspace-manager.ts +16 -15
  315. package/packages/runtime/tsconfig.json +17 -17
  316. package/vscode-extension/.vscodeignore +7 -7
  317. package/vscode-extension/oxe-agents-1.0.0.vsix +0 -0
  318. package/vscode-extension/package.json +185 -185
  319. package/vscode-extension/src/extension.js +310 -310
  320. package/vscode-extension/src/shared/contextLoader.js +137 -137
  321. package/vscode-extension/src/shared/contractBuilder.js +159 -159
  322. package/vscode-extension/src/shared/stateReader.js +101 -101
@@ -0,0 +1,60 @@
1
+ import type { ContextArtifact } from './context-pack-builder';
2
+
3
+ export type ContextDecisionType = 'execute' | 'verify' | 'plan' | 'review' | 'debug' | 'migration';
4
+
5
+ export interface ContextProfile {
6
+ decision_type: ContextDecisionType;
7
+ artifact_kind_weights: Record<ContextArtifact['kind'], number>;
8
+ quality_threshold: number;
9
+ max_artifacts: number;
10
+ max_tokens: number;
11
+ }
12
+
13
+ export const DEFAULT_PROFILES: Record<ContextDecisionType, ContextProfile> = {
14
+ execute: {
15
+ decision_type: 'execute',
16
+ artifact_kind_weights: { evidence: 0.8, lesson: 0.3, file: 0.6, summary: 0.4 },
17
+ quality_threshold: 0.6,
18
+ max_artifacts: 20,
19
+ max_tokens: 8000,
20
+ },
21
+ verify: {
22
+ decision_type: 'verify',
23
+ artifact_kind_weights: { evidence: 0.9, lesson: 0.2, file: 0.5, summary: 0.5 },
24
+ quality_threshold: 0.7,
25
+ max_artifacts: 15,
26
+ max_tokens: 6000,
27
+ },
28
+ plan: {
29
+ decision_type: 'plan',
30
+ artifact_kind_weights: { evidence: 0.4, lesson: 0.9, file: 0.7, summary: 0.6 },
31
+ quality_threshold: 0.5,
32
+ max_artifacts: 25,
33
+ max_tokens: 10000,
34
+ },
35
+ review: {
36
+ decision_type: 'review',
37
+ artifact_kind_weights: { evidence: 0.6, lesson: 0.5, file: 0.8, summary: 0.5 },
38
+ quality_threshold: 0.55,
39
+ max_artifacts: 20,
40
+ max_tokens: 8000,
41
+ },
42
+ debug: {
43
+ decision_type: 'debug',
44
+ artifact_kind_weights: { evidence: 0.9, lesson: 0.6, file: 0.7, summary: 0.4 },
45
+ quality_threshold: 0.6,
46
+ max_artifacts: 20,
47
+ max_tokens: 8000,
48
+ },
49
+ migration: {
50
+ decision_type: 'migration',
51
+ artifact_kind_weights: { evidence: 0.7, lesson: 0.7, file: 0.9, summary: 0.5 },
52
+ quality_threshold: 0.65,
53
+ max_artifacts: 30,
54
+ max_tokens: 12000,
55
+ },
56
+ };
57
+
58
+ export function getProfile(decisionType: ContextDecisionType): ContextProfile {
59
+ return DEFAULT_PROFILES[decisionType];
60
+ }
@@ -1 +1,3 @@
1
- export * from './context-pack-builder';
1
+ export * from './context-pack-builder';
2
+ export * from './context-pack-store';
3
+ export * from './context-profiles';
@@ -0,0 +1,174 @@
1
+ import crypto from 'crypto';
2
+ import path from 'path';
3
+ import fs from 'fs';
4
+ import type { DecisionMemo } from './decision-memo';
5
+
6
+ export type DecisionType =
7
+ | 'proceed'
8
+ | 'retry'
9
+ | 'escalate_gate'
10
+ | 'skip'
11
+ | 'abort'
12
+ | 'promote_lesson';
13
+
14
+ export type DecisionSignal =
15
+ | 'policy_allowed'
16
+ | 'policy_denied'
17
+ | 'gate_pending'
18
+ | 'gate_approved'
19
+ | 'evidence_sufficient'
20
+ | 'evidence_missing'
21
+ | 'retry_budget_available'
22
+ | 'retry_budget_exhausted'
23
+ | 'lesson_match'
24
+ | 'risk_high';
25
+
26
+ export type SeniorityLevel = 'junior' | 'standard' | 'senior' | 'expert';
27
+
28
+ export interface DecisionRecord {
29
+ decision_id: string;
30
+ work_item_id: string | null;
31
+ run_id: string;
32
+ type: DecisionType;
33
+ seniority: SeniorityLevel;
34
+ confidence: number;
35
+ signals: DecisionSignal[];
36
+ rationale: string;
37
+ timestamp: string;
38
+ memo?: DecisionMemo;
39
+ }
40
+
41
+ export interface DecisionLog {
42
+ run_id: string;
43
+ decisions: DecisionRecord[];
44
+ }
45
+
46
+ export interface DecisionInput {
47
+ work_item_id?: string;
48
+ run_id: string;
49
+ policy_allowed: boolean;
50
+ gate_pending: boolean;
51
+ gate_approved: boolean;
52
+ retry_count: number;
53
+ max_retries: number;
54
+ evidence_count: number;
55
+ risk_level: 'none' | 'low' | 'medium' | 'high' | 'critical';
56
+ lesson_match: boolean;
57
+ memo?: DecisionMemo;
58
+ }
59
+
60
+ function computeSeniority(confidence: number): SeniorityLevel {
61
+ if (confidence >= 0.9) return 'expert';
62
+ if (confidence >= 0.75) return 'senior';
63
+ if (confidence >= 0.5) return 'standard';
64
+ return 'junior';
65
+ }
66
+
67
+ export class DecisionEngine {
68
+ evaluate(input: DecisionInput): DecisionRecord {
69
+ const signals: DecisionSignal[] = [];
70
+ let type: DecisionType = 'proceed';
71
+ let confidence = 0.8;
72
+ let rationale = '';
73
+
74
+ if (!input.policy_allowed) {
75
+ signals.push('policy_denied');
76
+ type = 'abort';
77
+ confidence = 1.0;
78
+ rationale = 'Policy denied execution — aborting without retry.';
79
+ } else {
80
+ signals.push('policy_allowed');
81
+
82
+ if (input.gate_pending && !input.gate_approved) {
83
+ signals.push('gate_pending');
84
+ type = 'escalate_gate';
85
+ confidence = 0.95;
86
+ rationale = 'Human gate pending — escalating for approval before proceeding.';
87
+ } else {
88
+ if (input.gate_approved) signals.push('gate_approved');
89
+
90
+ if (input.risk_level === 'high' || input.risk_level === 'critical') {
91
+ signals.push('risk_high');
92
+ confidence = Math.max(0.4, confidence - 0.3);
93
+ rationale += 'High residual risk detected. ';
94
+ }
95
+
96
+ if (input.retry_count >= input.max_retries) {
97
+ signals.push('retry_budget_exhausted');
98
+ type = 'abort';
99
+ confidence = 0.9;
100
+ rationale += `Retry budget exhausted (${input.retry_count}/${input.max_retries}).`;
101
+ } else if (input.retry_count > 0) {
102
+ signals.push('retry_budget_available');
103
+ type = 'retry';
104
+ confidence = 0.7;
105
+ rationale += `Retrying (attempt ${input.retry_count + 1}/${input.max_retries + 1}).`;
106
+ } else {
107
+ if (input.evidence_count > 0) {
108
+ signals.push('evidence_sufficient');
109
+ confidence = Math.min(1.0, confidence + 0.1);
110
+ } else {
111
+ signals.push('evidence_missing');
112
+ confidence = Math.max(0.3, confidence - 0.2);
113
+ }
114
+
115
+ if (input.lesson_match) {
116
+ signals.push('lesson_match');
117
+ type = 'promote_lesson';
118
+ confidence = Math.min(1.0, confidence + 0.05);
119
+ }
120
+
121
+ if (!rationale) rationale = 'All signals green — proceeding with execution.';
122
+ }
123
+ }
124
+ }
125
+
126
+ return {
127
+ decision_id: `dec-${crypto.randomBytes(4).toString('hex')}`,
128
+ work_item_id: input.work_item_id ?? null,
129
+ run_id: input.run_id,
130
+ type,
131
+ seniority: computeSeniority(confidence),
132
+ confidence: Math.round(confidence * 100) / 100,
133
+ signals,
134
+ rationale: rationale.trim(),
135
+ timestamp: new Date().toISOString(),
136
+ ...(input.memo !== undefined ? { memo: input.memo } : {}),
137
+ };
138
+ }
139
+ }
140
+
141
+ export function appendDecision(projectRoot: string, runId: string, record: DecisionRecord): void {
142
+ const p = logPath(projectRoot, runId);
143
+ fs.mkdirSync(path.dirname(p), { recursive: true });
144
+
145
+ const log = loadDecisionLog(projectRoot, runId) ?? { run_id: runId, decisions: [] };
146
+ log.decisions.push(record);
147
+ fs.writeFileSync(p, JSON.stringify(log, null, 2), 'utf8');
148
+ }
149
+
150
+ export function loadDecisionLog(projectRoot: string, runId: string): DecisionLog | null {
151
+ const p = logPath(projectRoot, runId);
152
+ if (!fs.existsSync(p)) return null;
153
+ try {
154
+ return JSON.parse(fs.readFileSync(p, 'utf8')) as DecisionLog;
155
+ } catch {
156
+ return null;
157
+ }
158
+ }
159
+
160
+ export function queryDecisions(
161
+ log: DecisionLog,
162
+ filter: { type?: DecisionType; workItemId?: string; minConfidence?: number }
163
+ ): DecisionRecord[] {
164
+ return log.decisions.filter((d) => {
165
+ if (filter.type && d.type !== filter.type) return false;
166
+ if (filter.workItemId && d.work_item_id !== filter.workItemId) return false;
167
+ if (filter.minConfidence !== undefined && d.confidence < filter.minConfidence) return false;
168
+ return true;
169
+ });
170
+ }
171
+
172
+ function logPath(projectRoot: string, runId: string): string {
173
+ return path.join(projectRoot, '.oxe', 'runs', runId, 'decisions.json');
174
+ }
@@ -0,0 +1,211 @@
1
+ import crypto from 'crypto';
2
+ import path from 'path';
3
+ import fs from 'fs';
4
+
5
+ export type ChangeStrategy =
6
+ | 'minimal_patch'
7
+ | 'isolated_refactor'
8
+ | 'expand_contract'
9
+ | 'feature_flag'
10
+ | 'no_op';
11
+
12
+ export interface BlastRadiusEstimate {
13
+ estimated_files: number;
14
+ subsystems: string[];
15
+ risk_score: number;
16
+ reversible: boolean;
17
+ }
18
+
19
+ export interface RollbackPlan {
20
+ strategy: 'revert_commit' | 'restore_workspace' | 'undo_patch' | 'no_rollback';
21
+ steps: string[];
22
+ estimated_cost: 'low' | 'medium' | 'high';
23
+ preconditions: string[];
24
+ }
25
+
26
+ export interface DecisionMemo {
27
+ memo_id: string;
28
+ work_item_id: string;
29
+ run_id: string;
30
+ problem_summary: string;
31
+ chosen_strategy: ChangeStrategy;
32
+ alternatives_rejected: Array<{ strategy: ChangeStrategy; reason: string }>;
33
+ blast_radius: BlastRadiusEstimate;
34
+ rollback_plan: RollbackPlan;
35
+ min_evidence_required: string[];
36
+ confidence: number;
37
+ created_at: string;
38
+ }
39
+
40
+ // ─── BlastRadius estimation ───────────────────────────────────────────────────
41
+
42
+ function deriveSubsystems(mutationScope: string[]): string[] {
43
+ const seen = new Set<string>();
44
+ for (const p of mutationScope) {
45
+ const parts = p.replace(/\\/g, '/').split('/');
46
+ if (parts.length >= 2) seen.add(parts[0]);
47
+ else seen.add(p);
48
+ }
49
+ return [...seen];
50
+ }
51
+
52
+ function estimateRiskScore(mutationScope: string[], retryCount: number, riskLevel: string): number {
53
+ let score = 0;
54
+ score += Math.min(0.4, mutationScope.length * 0.05);
55
+ score += retryCount > 0 ? Math.min(0.2, retryCount * 0.05) : 0;
56
+ switch (riskLevel) {
57
+ case 'critical': score += 0.4; break;
58
+ case 'high': score += 0.3; break;
59
+ case 'medium': score += 0.15; break;
60
+ case 'low': score += 0.05; break;
61
+ default: break;
62
+ }
63
+ return Math.min(1, score);
64
+ }
65
+
66
+ export function buildBlastRadius(
67
+ mutationScope: string[],
68
+ retryCount: number,
69
+ riskLevel: string
70
+ ): BlastRadiusEstimate {
71
+ const risk_score = estimateRiskScore(mutationScope, retryCount, riskLevel);
72
+ return {
73
+ estimated_files: mutationScope.length,
74
+ subsystems: deriveSubsystems(mutationScope),
75
+ risk_score: Math.round(risk_score * 100) / 100,
76
+ reversible: riskLevel !== 'critical' && mutationScope.length <= 10,
77
+ };
78
+ }
79
+
80
+ // ─── RollbackPlan ─────────────────────────────────────────────────────────────
81
+
82
+ export function buildRollbackPlan(
83
+ blastRadius: BlastRadiusEstimate,
84
+ retryCount: number
85
+ ): RollbackPlan {
86
+ if (blastRadius.risk_score >= 0.7 || !blastRadius.reversible) {
87
+ return {
88
+ strategy: 'restore_workspace',
89
+ steps: ['snapshot workspace before mutation', 'restore snapshot on failure', 'verify clean state'],
90
+ estimated_cost: 'high',
91
+ preconditions: ['workspace snapshot available', 'no shared-state mutations'],
92
+ };
93
+ }
94
+ if (retryCount > 1 || blastRadius.estimated_files > 5) {
95
+ return {
96
+ strategy: 'revert_commit',
97
+ steps: ['record commit SHA before change', 'git revert on failure'],
98
+ estimated_cost: 'medium',
99
+ preconditions: ['git repo initialized', 'changes committed atomically'],
100
+ };
101
+ }
102
+ return {
103
+ strategy: 'undo_patch',
104
+ steps: ['apply inverse patch'],
105
+ estimated_cost: 'low',
106
+ preconditions: ['original file state recorded'],
107
+ };
108
+ }
109
+
110
+ // ─── StrategySelector ────────────────────────────────────────────────────────
111
+
112
+ export class StrategySelector {
113
+ select(mutationScope: string[], retryCount: number, riskLevel: string): ChangeStrategy {
114
+ if (riskLevel === 'critical') return 'feature_flag';
115
+ if (retryCount > 1) return 'minimal_patch';
116
+ if (mutationScope.length > 8) return 'isolated_refactor';
117
+ if (riskLevel === 'high') return 'feature_flag';
118
+ if (mutationScope.length > 3) return 'expand_contract';
119
+ if (mutationScope.length === 0) return 'no_op';
120
+ return 'minimal_patch';
121
+ }
122
+
123
+ alternatives(chosen: ChangeStrategy, mutationScope: string[], riskLevel: string): Array<{ strategy: ChangeStrategy; reason: string }> {
124
+ const all: ChangeStrategy[] = ['minimal_patch', 'isolated_refactor', 'expand_contract', 'feature_flag', 'no_op'];
125
+ return all
126
+ .filter((s) => s !== chosen)
127
+ .map((s) => ({ strategy: s, reason: this.rejectionReason(s, chosen, mutationScope, riskLevel) }));
128
+ }
129
+
130
+ private rejectionReason(s: ChangeStrategy, chosen: ChangeStrategy, mutationScope: string[], riskLevel: string): string {
131
+ switch (s) {
132
+ case 'no_op': return 'mutation scope is non-empty; no-op would not satisfy requirements';
133
+ case 'feature_flag': return riskLevel !== 'critical' && riskLevel !== 'high' ? 'risk level does not warrant feature flag overhead' : `${chosen} preferred for scope size`;
134
+ case 'isolated_refactor': return mutationScope.length <= 8 ? 'scope is small enough for simpler strategy' : `${chosen} preferred`;
135
+ case 'expand_contract': return 'expand-contract requires coordinated deployment; overhead not justified here';
136
+ case 'minimal_patch': return `${chosen} preferred due to scope or risk level`;
137
+ default: return 'not applicable';
138
+ }
139
+ }
140
+ }
141
+
142
+ // ─── Memo builder & persistence ──────────────────────────────────────────────
143
+
144
+ export interface BuildMemoInput {
145
+ work_item_id: string;
146
+ run_id: string;
147
+ problem_summary: string;
148
+ mutation_scope: string[];
149
+ retry_count: number;
150
+ risk_level: string;
151
+ min_evidence_required?: string[];
152
+ confidence?: number;
153
+ }
154
+
155
+ export function buildMemo(input: BuildMemoInput): DecisionMemo {
156
+ const selector = new StrategySelector();
157
+ const chosen = selector.select(input.mutation_scope, input.retry_count, input.risk_level);
158
+ const blast_radius = buildBlastRadius(input.mutation_scope, input.retry_count, input.risk_level);
159
+ const rollback_plan = buildRollbackPlan(blast_radius, input.retry_count);
160
+ const alternatives_rejected = selector.alternatives(chosen, input.mutation_scope, input.risk_level);
161
+
162
+ return {
163
+ memo_id: `memo-${crypto.randomBytes(4).toString('hex')}`,
164
+ work_item_id: input.work_item_id,
165
+ run_id: input.run_id,
166
+ problem_summary: input.problem_summary,
167
+ chosen_strategy: chosen,
168
+ alternatives_rejected,
169
+ blast_radius,
170
+ rollback_plan,
171
+ min_evidence_required: input.min_evidence_required ?? [],
172
+ confidence: input.confidence ?? (1 - blast_radius.risk_score),
173
+ created_at: new Date().toISOString(),
174
+ };
175
+ }
176
+
177
+ function memoDir(projectRoot: string, runId: string): string {
178
+ return path.join(projectRoot, '.oxe', 'runs', runId, 'memos');
179
+ }
180
+
181
+ export function saveMemo(projectRoot: string, memo: DecisionMemo): void {
182
+ const dir = memoDir(projectRoot, memo.run_id);
183
+ fs.mkdirSync(dir, { recursive: true });
184
+ fs.writeFileSync(path.join(dir, `${memo.memo_id}.json`), JSON.stringify(memo, null, 2), 'utf8');
185
+ }
186
+
187
+ export function loadMemo(projectRoot: string, runId: string, memoId: string): DecisionMemo | null {
188
+ const p = path.join(memoDir(projectRoot, runId), `${memoId}.json`);
189
+ if (!fs.existsSync(p)) return null;
190
+ try {
191
+ return JSON.parse(fs.readFileSync(p, 'utf8')) as DecisionMemo;
192
+ } catch {
193
+ return null;
194
+ }
195
+ }
196
+
197
+ export function listMemos(projectRoot: string, runId: string): DecisionMemo[] {
198
+ const dir = memoDir(projectRoot, runId);
199
+ if (!fs.existsSync(dir)) return [];
200
+ return fs
201
+ .readdirSync(dir)
202
+ .filter((f) => f.endsWith('.json'))
203
+ .map((f) => {
204
+ try {
205
+ return JSON.parse(fs.readFileSync(path.join(dir, f), 'utf8')) as DecisionMemo;
206
+ } catch {
207
+ return null;
208
+ }
209
+ })
210
+ .filter((m): m is DecisionMemo => m !== null);
211
+ }
@@ -0,0 +1,2 @@
1
+ export * from './decision-engine';
2
+ export * from './decision-memo';
@@ -1,84 +1,91 @@
1
- import { execFileSync, spawnSync } from 'child_process';
2
-
3
- export interface BranchInfo {
4
- name: string;
5
- current: boolean;
6
- commit: string;
7
- }
8
-
9
- export class BranchManager {
10
- constructor(private readonly projectRoot: string) {}
11
-
12
- currentBranch(): string {
13
- return this.git(['rev-parse', '--abbrev-ref', 'HEAD']).trim();
14
- }
15
-
16
- currentCommit(): string {
17
- return this.git(['rev-parse', 'HEAD']).trim();
18
- }
19
-
20
- createSessionBranch(sessionId: string): string {
21
- const name = `oxe/${sessionId}`;
22
- this.git(['checkout', '-b', name]);
23
- return name;
24
- }
25
-
26
- createOxeBranch(name: string, base?: string): string {
27
- const fullName = name.startsWith('oxe/') ? name : `oxe/${name}`;
28
- if (base) {
29
- this.git(['checkout', '-b', fullName, base]);
30
- } else {
31
- this.git(['checkout', '-b', fullName]);
32
- }
33
- return fullName;
34
- }
35
-
36
- switchTo(branchName: string): void {
37
- this.git(['checkout', branchName]);
38
- }
39
-
40
- deleteBranch(name: string, force = false): void {
41
- const flag = force ? '-D' : '-d';
42
- this.git(['branch', flag, name]);
43
- }
44
-
45
- listOxeBranches(): BranchInfo[] {
46
- const raw = this.git(['branch', '--list', 'oxe/*', '--format=%(refname:short) %(objectname:short) %(HEAD)']);
47
- return raw
48
- .split('\n')
49
- .filter(Boolean)
50
- .map((line) => {
51
- const parts = line.trim().split(/\s+/);
52
- return {
53
- name: parts[0],
54
- commit: parts[1] ?? '',
55
- current: parts[2] === '*',
56
- };
57
- });
58
- }
59
-
60
- mergeWorktreeBranch(worktreeBranch: string, targetBranch: string): void {
61
- const saved = this.currentBranch();
62
- try {
63
- this.git(['checkout', targetBranch]);
64
- this.git(['merge', '--no-ff', worktreeBranch, '-m', `oxe: merge ${worktreeBranch}`]);
65
- } finally {
66
- try { this.git(['checkout', saved]); } catch { /* best effort */ }
67
- }
68
- }
69
-
70
- branchExists(name: string): boolean {
71
- const result = spawnSync('git', ['rev-parse', '--verify', name], {
72
- cwd: this.projectRoot,
73
- encoding: 'utf8',
74
- });
75
- return result.status === 0;
76
- }
77
-
78
- private git(args: string[]): string {
79
- return execFileSync('git', args, {
80
- cwd: this.projectRoot,
81
- encoding: 'utf8',
82
- });
83
- }
84
- }
1
+ import { execFileSync, spawnSync } from 'child_process';
2
+
3
+ export interface BranchInfo {
4
+ name: string;
5
+ current: boolean;
6
+ commit: string;
7
+ }
8
+
9
+ export class BranchManager {
10
+ constructor(private readonly projectRoot: string) {}
11
+
12
+ currentBranch(): string {
13
+ return this.git(['rev-parse', '--abbrev-ref', 'HEAD']).trim();
14
+ }
15
+
16
+ currentCommit(): string {
17
+ return this.git(['rev-parse', 'HEAD']).trim();
18
+ }
19
+
20
+ createSessionBranch(sessionId: string): string {
21
+ const name = `oxe/${sessionId}`;
22
+ this.git(['checkout', '-b', name]);
23
+ return name;
24
+ }
25
+
26
+ createOxeBranch(name: string, base?: string): string {
27
+ const fullName = name.startsWith('oxe/') ? name : `oxe/${name}`;
28
+ if (base) {
29
+ this.git(['checkout', '-b', fullName, base]);
30
+ } else {
31
+ this.git(['checkout', '-b', fullName]);
32
+ }
33
+ return fullName;
34
+ }
35
+
36
+ switchTo(branchName: string): void {
37
+ this.git(['checkout', branchName]);
38
+ }
39
+
40
+ deleteBranch(name: string, force = false): void {
41
+ const flag = force ? '-D' : '-d';
42
+ this.git(['branch', flag, name]);
43
+ }
44
+
45
+ listOxeBranches(): BranchInfo[] {
46
+ const raw = this.git(['branch', '--list', 'oxe/*', '--format=%(refname:short) %(objectname:short) %(HEAD)']);
47
+ return raw
48
+ .split('\n')
49
+ .filter(Boolean)
50
+ .map((line) => {
51
+ const parts = line.trim().split(/\s+/);
52
+ return {
53
+ name: parts[0],
54
+ commit: parts[1] ?? '',
55
+ current: parts[2] === '*',
56
+ };
57
+ });
58
+ }
59
+
60
+ mergeWorktreeBranch(worktreeBranch: string, targetBranch: string): void {
61
+ const saved = this.currentBranch();
62
+ try {
63
+ this.git(['checkout', targetBranch]);
64
+ this.git(['merge', '--no-ff', worktreeBranch, '-m', `oxe: merge ${worktreeBranch}`]);
65
+ } finally {
66
+ try { this.git(['checkout', saved]); } catch { /* best effort */ }
67
+ }
68
+ }
69
+
70
+ branchExists(name: string): boolean {
71
+ const result = spawnSync('git', ['rev-parse', '--verify', name], {
72
+ cwd: this.projectRoot,
73
+ encoding: 'utf8',
74
+ });
75
+ return result.status === 0;
76
+ }
77
+
78
+ push(remote: string, branchName: string, setUpstream = false): void {
79
+ const args = ['push'];
80
+ if (setUpstream) args.push('-u');
81
+ args.push(remote, branchName);
82
+ this.git(args);
83
+ }
84
+
85
+ private git(args: string[]): string {
86
+ return execFileSync('git', args, {
87
+ cwd: this.projectRoot,
88
+ encoding: 'utf8',
89
+ });
90
+ }
91
+ }