loki-mode 7.5.17 → 7.5.27

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 (47) hide show
  1. package/README.md +10 -9
  2. package/SKILL.md +14 -14
  3. package/VERSION +1 -1
  4. package/autonomy/completion-council.sh +26 -3
  5. package/autonomy/lib/claude-flags.sh +132 -0
  6. package/autonomy/lib/mcp-config.sh +160 -0
  7. package/autonomy/lib/project-graph.sh +675 -0
  8. package/autonomy/lib/voter-agents.sh +356 -0
  9. package/autonomy/loki +61 -96
  10. package/autonomy/run.sh +95 -186
  11. package/bin/loki +10 -0
  12. package/dashboard/__init__.py +1 -1
  13. package/dashboard/requirements.txt +13 -8
  14. package/dashboard/server.py +33 -15
  15. package/dashboard/static/index.html +298 -299
  16. package/docs/INSTALLATION.md +54 -21
  17. package/docs/retrospectives/v7.5.15-fleet-postmortem.md +325 -0
  18. package/docs/retrospectives/v7.5.15-honesty-audit.md +136 -0
  19. package/docs/retrospectives/v7.5.15-llm-failure-modes.md +49 -0
  20. package/loki-ts/data/finding-schema.json +74 -0
  21. package/loki-ts/data/model-pricing.json +12 -0
  22. package/loki-ts/dist/loki.js +109 -108
  23. package/mcp/__init__.py +1 -1
  24. package/mcp/lsp_proxy.py +713 -0
  25. package/mcp/requirements.txt +9 -3
  26. package/mcp/tests/__init__.py +0 -0
  27. package/mcp/tests/test_lsp_proxy.py +377 -0
  28. package/memory/app_graph.py +153 -0
  29. package/memory/storage.py +6 -1
  30. package/memory/tests/test_app_graph.py +134 -0
  31. package/package.json +4 -3
  32. package/providers/claude.sh +115 -4
  33. package/providers/codex.sh +2 -2
  34. package/providers/loader.sh +4 -4
  35. package/providers/model_catalog.json +0 -9
  36. package/providers/models.sh +1 -2
  37. package/references/multi-provider.md +26 -35
  38. package/references/prompt-repetition.md +1 -1
  39. package/references/quality-control.md +1 -1
  40. package/skills/00-index.md +3 -3
  41. package/skills/model-selection.md +11 -14
  42. package/skills/providers.md +17 -57
  43. package/skills/quality-gates.md +2 -2
  44. package/skills/troubleshooting.md +1 -1
  45. package/src/integrations/github/action-handler.js +3 -2
  46. package/src/protocols/tools/start-project.js +1 -1
  47. package/providers/gemini.sh +0 -343
@@ -1,57 +1,57 @@
1
1
  // @bun
2
- var X6=Object.defineProperty;var Z6=($)=>$;function K6($,Q){this[$]=Z6.bind(null,Q)}var y=($,Q)=>{for(var z in Q)X6($,z,{get:Q[z],enumerable:!0,configurable:!0,set:K6.bind(Q,z)})};var S=($,Q)=>()=>($&&(Q=$($=0)),Q);var G1=import.meta.require;var a1={};y(a1,{lokiDir:()=>x,homeLokiDir:()=>w1,findRepoRootForVersion:()=>E1,REPO_ROOT:()=>u});import{resolve as f,dirname as F1}from"path";import{fileURLToPath as W6}from"url";import{existsSync as V1}from"fs";import{homedir as U6}from"os";function q6(){let $=n1;for(let Q=0;Q<6;Q++){if(V1(f($,"VERSION"))&&V1(f($,"autonomy/run.sh")))return $;let z=F1($);if(z===$)break;$=z}return f(n1,"..","..","..")}function E1($){let Q=$;for(let z=0;z<6;z++){if(V1(f(Q,"VERSION"))&&V1(f(Q,"autonomy/run.sh")))return Q;let X=F1(Q);if(X===Q)break;Q=X}return f($,"..","..","..")}function x(){return process.env.LOKI_DIR??f(process.cwd(),".loki")}function w1(){return f(U6(),".loki")}var n1,u;var g=S(()=>{n1=F1(W6(import.meta.url));u=q6()});var t1={};y(t1,{runOrThrow:()=>B6,run:()=>E,commandVersion:()=>M6,commandExists:()=>b,ShellError:()=>S1});async function E($,Q={}){let z=Bun.spawn({cmd:[...$],stdout:"pipe",stderr:"pipe",env:Q.env?{...process.env,...Q.env}:process.env,cwd:Q.cwd}),X,Z;if(Q.timeoutMs&&Q.timeoutMs>0)X=setTimeout(()=>{try{z.kill("SIGTERM")}catch{}Z=setTimeout(()=>{try{z.kill("SIGKILL")}catch{}},2000)},Q.timeoutMs);try{let[K,W,q]=await Promise.all([new Response(z.stdout).text(),new Response(z.stderr).text(),z.exited]);return{stdout:K,stderr:W,exitCode:q}}finally{if(X)clearTimeout(X);if(Z)clearTimeout(Z)}}async function B6($,Q={}){let z=await E($,Q);if(z.exitCode!==0)throw new S1(`command failed (${z.exitCode}): ${$.join(" ")}`,z.exitCode,z.stdout,z.stderr);return z}async function b($){let Q=Y6($),z=await E(["sh","-c",`command -v ${Q}`],{timeoutMs:5000});if(z.exitCode===0)return z.stdout.trim()||null;return null}function Y6($){if(!/^[A-Za-z0-9._/-]+$/.test($))throw Error(`refused to shell-escape suspect token: ${$}`);return $}async function M6($,Q="--version"){if(!await b($))return null;let X=await E([$,Q],{timeoutMs:5000});if(X.exitCode!==0)return null;return((X.stdout||X.stderr).split(/\r?\n/)[0]?.trim()??"")||null}var S1;var p=S(()=>{S1=class S1 extends Error{message;exitCode;stdout;stderr;constructor($,Q,z,X){super($);this.message=$;this.exitCode=Q;this.stdout=z;this.stderr=X;this.name="ShellError"}}});function c($){return A6?"":$}var A6,L,k,R,y5,A,C,P,U;var n=S(()=>{A6=(process.env.NO_COLOR??"").length>0;L=c("\x1B[0;31m"),k=c("\x1B[0;32m"),R=c("\x1B[1;33m"),y5=c("\x1B[0;34m"),A=c("\x1B[0;36m"),C=c("\x1B[1m"),P=c("\x1B[2m"),U=c("\x1B[0m")});import{existsSync as F6}from"fs";async function s(){if(Q1!==void 0)return Q1;let $="/opt/homebrew/bin/python3.12";if(F6($))return Q1=$,$;let Q=await b("python3.12");if(Q)return Q1=Q,Q;let z=await b("python3");return Q1=z,z}async function a($,Q={}){let z=await s();if(!z)return{stdout:"",stderr:"python3 not found",exitCode:127};return E([z,"-c",$],Q)}var Q1;var X1=S(()=>{p()});var q0={};y(q0,{runStatus:()=>m6});import{existsSync as N,readFileSync as K1,readdirSync as Q0,statSync as X0}from"fs";import{resolve as F,basename as D6}from"path";async function k6(){if(await b("jq"))return!0;return process.stdout.write(`${L}Error: jq is required but not installed.${U}
2
+ var X6=Object.defineProperty;var Z6=($)=>$;function K6($,Q){this[$]=Z6.bind(null,Q)}var y=($,Q)=>{for(var z in Q)X6($,z,{get:Q[z],enumerable:!0,configurable:!0,set:K6.bind(Q,z)})};var w=($,Q)=>()=>($&&(Q=$($=0)),Q);var H1=import.meta.require;var a1={};y(a1,{lokiDir:()=>P,homeLokiDir:()=>E1,findRepoRootForVersion:()=>F1,REPO_ROOT:()=>u});import{resolve as f,dirname as R1}from"path";import{fileURLToPath as W6}from"url";import{existsSync as G1}from"fs";import{homedir as U6}from"os";function q6(){let $=n1;for(let Q=0;Q<6;Q++){if(G1(f($,"VERSION"))&&G1(f($,"autonomy/run.sh")))return $;let z=R1($);if(z===$)break;$=z}return f(n1,"..","..","..")}function F1($){let Q=$;for(let z=0;z<6;z++){if(G1(f(Q,"VERSION"))&&G1(f(Q,"autonomy/run.sh")))return Q;let X=R1(Q);if(X===Q)break;Q=X}return f($,"..","..","..")}function P(){return process.env.LOKI_DIR??f(process.cwd(),".loki")}function E1(){return f(U6(),".loki")}var n1,u;var v=w(()=>{n1=R1(W6(import.meta.url));u=q6()});var t1={};y(t1,{runOrThrow:()=>B6,run:()=>E,commandVersion:()=>M6,commandExists:()=>b,ShellError:()=>w1});async function E($,Q={}){let z=Bun.spawn({cmd:[...$],stdout:"pipe",stderr:"pipe",env:Q.env?{...process.env,...Q.env}:process.env,cwd:Q.cwd}),X,Z;if(Q.timeoutMs&&Q.timeoutMs>0)X=setTimeout(()=>{try{z.kill("SIGTERM")}catch{}Z=setTimeout(()=>{try{z.kill("SIGKILL")}catch{}},2000)},Q.timeoutMs);try{let[K,W,q]=await Promise.all([new Response(z.stdout).text(),new Response(z.stderr).text(),z.exited]);return{stdout:K,stderr:W,exitCode:q}}finally{if(X)clearTimeout(X);if(Z)clearTimeout(Z)}}async function B6($,Q={}){let z=await E($,Q);if(z.exitCode!==0)throw new w1(`command failed (${z.exitCode}): ${$.join(" ")}`,z.exitCode,z.stdout,z.stderr);return z}async function b($){let Q=Y6($),z=await E(["sh","-c",`command -v ${Q}`],{timeoutMs:5000});if(z.exitCode===0)return z.stdout.trim()||null;return null}function Y6($){if(!/^[A-Za-z0-9._/-]+$/.test($))throw Error(`refused to shell-escape suspect token: ${$}`);return $}async function M6($,Q="--version"){if(!await b($))return null;let X=await E([$,Q],{timeoutMs:5000});if(X.exitCode!==0)return null;return((X.stdout||X.stderr).split(/\r?\n/)[0]?.trim()??"")||null}var w1;var p=w(()=>{w1=class w1 extends Error{message;exitCode;stdout;stderr;constructor($,Q,z,X){super($);this.message=$;this.exitCode=Q;this.stdout=z;this.stderr=X;this.name="ShellError"}}});function c($){return A6?"":$}var A6,L,C,x,y5,A,D,R,U;var n=w(()=>{A6=(process.env.NO_COLOR??"").length>0;L=c("\x1B[0;31m"),C=c("\x1B[0;32m"),x=c("\x1B[1;33m"),y5=c("\x1B[0;34m"),A=c("\x1B[0;36m"),D=c("\x1B[1m"),R=c("\x1B[2m"),U=c("\x1B[0m")});import{existsSync as F6}from"fs";async function r(){if(Q1!==void 0)return Q1;let $="/opt/homebrew/bin/python3.12";if(F6($))return Q1=$,$;let Q=await b("python3.12");if(Q)return Q1=Q,Q;let z=await b("python3");return Q1=z,z}async function a($,Q={}){let z=await r();if(!z)return{stdout:"",stderr:"python3 not found",exitCode:127};return E([z,"-c",$],Q)}var Q1;var X1=w(()=>{p()});var q0={};y(q0,{runStatus:()=>g6});import{existsSync as S,readFileSync as K1,readdirSync as Q0,statSync as X0}from"fs";import{resolve as F,basename as D6}from"path";async function k6(){if(await b("jq"))return!0;return process.stdout.write(`${L}Error: jq is required but not installed.${U}
3
3
  `),process.stdout.write(`Install with:
4
4
  `),process.stdout.write(` brew install jq (macOS)
5
5
  `),process.stdout.write(` apt install jq (Debian/Ubuntu)
6
6
  `),process.stdout.write(` yum install jq (RHEL/CentOS)
7
- `),!1}function J1($){if(!Number.isFinite($)||$<=0)return!1;try{return process.kill($,0),!0}catch{return!1}}function B1($){if(!N($))return null;try{let Q=K1($,"utf-8").trim();if(!Q)return null;let z=Number.parseInt(Q,10);return Number.isFinite(z)?z:null}catch{return null}}function b6($){let Q=[],z=B1(F($,"loki.pid"));if(z!==null&&J1(z))Q.push(`global:${z}`);let X=F($,"sessions");if(N(X)){let Z=[];try{Z=Q0(X)}catch{Z=[]}for(let K of Z){let W=F(X,K);try{if(!X0(W).isDirectory())continue}catch{continue}let q=F(W,"loki.pid"),H=B1(q);if(H!==null&&J1(H))Q.push(`${K}:${H}`)}}if(N($)){let Z=[];try{Z=Q0($)}catch{Z=[]}for(let K of Z){if(!K.startsWith("run-")||!K.endsWith(".pid"))continue;let W=F($,K);try{if(!X0(W).isFile())continue}catch{continue}let q=D6(K,".pid").slice(4),H=B1(W);if(H!==null&&J1(H)){if(!Q.some((J)=>J.startsWith(`${q}:`)))Q.push(`${q}:${H}`)}}}return Q}async function Z0($,Q){let z=await E(["jq","-r",$,Q]);if(z.exitCode!==0)return null;return z.stdout.trim()}function K0($,Q){try{let z=K1($,"utf-8"),Z=JSON.parse(z)[Q];if(typeof Z==="number"){if(Q==="budget_used"){let K=Math.round(Z*100)/100;if(Number.isInteger(K))return String(K);return String(K)}return String(Z)}if(Z===void 0||Z===null)return"0";return String(Z)}catch{return"0"}}function W0($,Q,z){try{let X=K1($,"utf-8"),K=JSON.parse(X)[Q];if(typeof K==="number"&&Number.isFinite(K))return K;return z}catch{return z}}async function y6(){let $=x();if(!await k6())return 1;if(!N($))return process.stdout.write(`${C}Loki Mode Status${U}
7
+ `),!1}function V1($){if(!Number.isFinite($)||$<=0)return!1;try{return process.kill($,0),!0}catch{return!1}}function J1($){if(!S($))return null;try{let Q=K1($,"utf-8").trim();if(!Q)return null;let z=Number.parseInt(Q,10);return Number.isFinite(z)?z:null}catch{return null}}function b6($){let Q=[],z=J1(F($,"loki.pid"));if(z!==null&&V1(z))Q.push(`global:${z}`);let X=F($,"sessions");if(S(X)){let Z=[];try{Z=Q0(X)}catch{Z=[]}for(let K of Z){let W=F(X,K);try{if(!X0(W).isDirectory())continue}catch{continue}let q=F(W,"loki.pid"),H=J1(q);if(H!==null&&V1(H))Q.push(`${K}:${H}`)}}if(S($)){let Z=[];try{Z=Q0($)}catch{Z=[]}for(let K of Z){if(!K.startsWith("run-")||!K.endsWith(".pid"))continue;let W=F($,K);try{if(!X0(W).isFile())continue}catch{continue}let q=D6(K,".pid").slice(4),H=J1(W);if(H!==null&&V1(H)){if(!Q.some((J)=>J.startsWith(`${q}:`)))Q.push(`${q}:${H}`)}}}return Q}async function Z0($,Q){let z=await E(["jq","-r",$,Q]);if(z.exitCode!==0)return null;return z.stdout.trim()}function K0($,Q){try{let z=K1($,"utf-8"),Z=JSON.parse(z)[Q];if(typeof Z==="number"){if(Q==="budget_used"){let K=Math.round(Z*100)/100;if(Number.isInteger(K))return String(K);return String(K)}return String(Z)}if(Z===void 0||Z===null)return"0";return String(Z)}catch{return"0"}}function W0($,Q,z){try{let X=K1($,"utf-8"),K=JSON.parse(X)[Q];if(typeof K==="number"&&Number.isFinite(K))return K;return z}catch{return z}}async function y6(){let $=P();if(!await k6())return 1;if(!S($))return process.stdout.write(`${D}Loki Mode Status${U}
8
8
  `),process.stdout.write(`
9
- `),process.stdout.write(`${R}No active session found.${U}
9
+ `),process.stdout.write(`${x}No active session found.${U}
10
10
  `),process.stdout.write(`Loki Mode has not been initialized in this directory.
11
11
  `),process.stdout.write(`
12
12
  `),process.stdout.write(`To start a session:
13
13
  `),process.stdout.write(` loki start <prd> - Start with a PRD file
14
14
  `),process.stdout.write(` loki start - Start without a PRD
15
15
  `),process.stdout.write(`
16
- `),process.stdout.write(`${P}Current directory: ${process.cwd()}${U}
17
- `),0;process.stdout.write(`${C}Loki Mode Status${U}
16
+ `),process.stdout.write(`${R}Current directory: ${process.cwd()}${U}
17
+ `),0;process.stdout.write(`${D}Loki Mode Status${U}
18
18
  `),process.stdout.write(`
19
- `);let Q="",z=F($,"state","provider");if(N(z))try{Q=K1(z,"utf-8").trim()}catch{Q=""}let X=Q||process.env.LOKI_PROVIDER||"claude",Z="full features";switch(X){case"codex":case"gemini":case"aider":Z="degraded mode";break;case"cline":Z="near-full mode";break;default:Z="full features";break}process.stdout.write(`${A}Provider:${U} ${X} (${Z})
20
- `),process.stdout.write(`${P} Switch with: loki provider set <claude|codex|gemini|cline|aider>${U}
19
+ `);let Q="",z=F($,"state","provider");if(S(z))try{Q=K1(z,"utf-8").trim()}catch{Q=""}let X=Q||process.env.LOKI_PROVIDER||"claude",Z="full features";switch(X){case"codex":case"aider":Z="degraded mode";break;case"cline":Z="near-full mode";break;default:Z="full features";break}process.stdout.write(`${A}Provider:${U} ${X} (${Z})
20
+ `),process.stdout.write(`${R} Switch with: loki provider set <claude|codex|cline|aider>${U}
21
21
  `),process.stdout.write(`
22
- `);let K=b6($);if(K.length>0){process.stdout.write(`${k}Active Sessions: ${K.length}${U}
23
- `);for(let V of K){let M=V.indexOf(":"),O=M>=0?V.slice(0,M):V,j=M>=0?V.slice(M+1):"";if(O==="global")process.stdout.write(` ${A}[global]${U} PID ${j}
24
- `);else process.stdout.write(` ${A}[#${O}]${U} PID ${j}
22
+ `);let K=b6($);if(K.length>0){process.stdout.write(`${C}Active Sessions: ${K.length}${U}
23
+ `);for(let V of K){let M=V.indexOf(":"),j=M>=0?V.slice(0,M):V,_=M>=0?V.slice(M+1):"";if(j==="global")process.stdout.write(` ${A}[global]${U} PID ${_}
24
+ `);else process.stdout.write(` ${A}[#${j}]${U} PID ${_}
25
25
  `)}process.stdout.write(`
26
- `),process.stdout.write(`${P} Stop specific: loki stop <session-id>${U}
27
- `),process.stdout.write(`${P} Stop all: loki stop${U}
26
+ `),process.stdout.write(`${R} Stop specific: loki stop <session-id>${U}
27
+ `),process.stdout.write(`${R} Stop all: loki stop${U}
28
28
  `),process.stdout.write(`
29
- `)}if(N(F($,"PAUSE")))process.stdout.write(`${R}Status: PAUSED${U}
30
- `),process.stdout.write(`${P} Resume with: loki resume${U}
29
+ `)}if(S(F($,"PAUSE")))process.stdout.write(`${x}Status: PAUSED${U}
30
+ `),process.stdout.write(`${R} Resume with: loki resume${U}
31
31
  `),process.stdout.write(`
32
- `);else if(N(F($,"STOP")))process.stdout.write(`${L}Status: STOPPED${U}
33
- `),process.stdout.write(`${P} Clear with: loki resume${U}
32
+ `);else if(S(F($,"STOP")))process.stdout.write(`${L}Status: STOPPED${U}
33
+ `),process.stdout.write(`${R} Clear with: loki resume${U}
34
34
  `),process.stdout.write(`
35
- `);let W=F($,"STATUS.txt");if(N(W)){process.stdout.write(`${A}Session Info:${U}
35
+ `);let W=F($,"STATUS.txt");if(S(W)){process.stdout.write(`${A}Session Info:${U}
36
36
  `);try{process.stdout.write(K1(W,"utf-8"))}catch{}process.stdout.write(`
37
- `)}let q=F($,"state","orchestrator.json");if(N(q)){process.stdout.write(`${A}Orchestrator State:${U}
37
+ `)}let q=F($,"state","orchestrator.json");if(S(q)){process.stdout.write(`${A}Orchestrator State:${U}
38
38
  `);let V=await Z0('.currentPhase // "unknown"',q);process.stdout.write(`${V??"unknown"}
39
- `)}let H=F($,"queue","pending.json");if(N(H)){let V=await Z0('if type == "array" then length elif .tasks then .tasks | length else 0 end',H);process.stdout.write(`${A}Pending Tasks:${U} ${V??"0"}
40
- `)}let G=F($,"metrics","budget.json");if(N(G)){let V=K0(G,"budget_limit"),M=K0(G,"budget_used");if(V!=="0")process.stdout.write(`${A}Budget:${U} $${M} / $${V}
39
+ `)}let H=F($,"queue","pending.json");if(S(H)){let V=await Z0('if type == "array" then length elif .tasks then .tasks | length else 0 end',H);process.stdout.write(`${A}Pending Tasks:${U} ${V??"0"}
40
+ `)}let G=F($,"metrics","budget.json");if(S(G)){let V=K0(G,"budget_limit"),M=K0(G,"budget_used");if(V!=="0")process.stdout.write(`${A}Budget:${U} $${M} / $${V}
41
41
  `);else process.stdout.write(`${A}Cost:${U} $${M} (no limit)
42
- `)}let J=F($,"state","context-usage.json");if(N(J)){let V=W0(J,"window_size",200000),M=W0(J,"used_tokens",0),O=0;if(V>0)O=Math.floor(M*100/V);process.stdout.write(`${A}Context:${U} ${O}% (${M} / ${V} tokens)
43
- `)}let B=F($,"dashboard","dashboard.pid");if(N(B)){let V=B1(B);if(V!==null&&J1(V)){let M=process.env.LOKI_DASHBOARD_PORT||"57374";process.stdout.write(`${A}Dashboard:${U} http://127.0.0.1:${M}/
42
+ `)}let J=F($,"state","context-usage.json");if(S(J)){let V=W0(J,"window_size",200000),M=W0(J,"used_tokens",0),j=0;if(V>0)j=Math.floor(M*100/V);process.stdout.write(`${A}Context:${U} ${j}% (${M} / ${V} tokens)
43
+ `)}let B=F($,"dashboard","dashboard.pid");if(S(B)){let V=J1(B);if(V!==null&&V1(V)){let M=process.env.LOKI_DASHBOARD_PORT||"57374";process.stdout.write(`${A}Dashboard:${U} http://127.0.0.1:${M}/
44
44
  `)}}return await h6($),process.stdout.write(`
45
- `),process.stdout.write(`${P} Tip: loki context show - detailed token breakdown${U}
46
- `),process.stdout.write(`${P} Tip: loki code overview - codebase intelligence${U}
47
- `),0}async function h6($){let Q=F($,"state"),z=v6(Q),X=F(Q,"relevant-learnings.json"),Z=F($,"escalations"),K=z.length>0,W=N(X),q=N(Z);if(!K&&!W&&!q)return;if(process.stdout.write(`
45
+ `),process.stdout.write(`${R} Tip: loki context show - detailed token breakdown${U}
46
+ `),process.stdout.write(`${R} Tip: loki code overview - codebase intelligence${U}
47
+ `),0}async function h6($){let Q=F($,"state"),z=v6(Q),X=F(Q,"relevant-learnings.json"),Z=F($,"escalations"),K=z.length>0,W=S(X),q=S(Z);if(!K&&!W&&!q)return;if(process.stdout.write(`
48
48
  ${A}Phase 1 artifacts:${U}
49
49
  `),K){let H=z[z.length-1],G=U0(H);if(G&&Array.isArray(G.findings)){let J={Critical:0,High:0,Medium:0,Low:0};for(let V of G.findings){let M=String(V.severity??"");if(M in J)J[M]=(J[M]??0)+1}let B=Object.entries(J).filter(([,V])=>V>0).map(([V,M])=>`${M} ${V.toLowerCase()}`).join(", ");process.stdout.write(` Findings (iter ${G.iteration??"?"}): ${B||"none"} -- ${G.findings.length} total
50
50
  `)}}if(W){let H=U0(X);if(H&&Array.isArray(H.learnings)&&H.learnings.length>0){let G=new Map;for(let B of H.learnings){let V=String(B.trigger??"unknown");G.set(V,(G.get(V)??0)+1)}let J=[...G.entries()].sort((B,V)=>V[1]-B[1]).slice(0,3).map(([B,V])=>`${V} ${B}`).join(", ");process.stdout.write(` Learnings: ${H.learnings.length} total (${J})
51
51
  `)}}if(q){let H=0,G="";try{let B=(await import("fs")).readdirSync(Z).filter((V)=>V.endsWith(".md"));if(H=B.length,B.length>0)B.sort(),G=B[B.length-1]??""}catch{}if(H>0)process.stdout.write(` Escalations: ${H} handoff doc${H===1?"":"s"} (latest: ${G})
52
- `)}}function v6($){if(!N($))return[];try{return G1("fs").readdirSync($).filter((X)=>/^findings-\d+\.json$/.test(X)).sort((X,Z)=>{let K=Number.parseInt(X.replace(/[^0-9]/g,""),10)||0,W=Number.parseInt(Z.replace(/[^0-9]/g,""),10)||0;return K-W}).map((X)=>F($,X))}catch{return[]}}function U0($){try{let Q=G1("fs");return JSON.parse(Q.readFileSync($,"utf-8"))}catch{return null}}async function g6(){let $=await s();if(!$)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
53
- `),1;let Q=u,z=x(),X=process.env.LOKI_DASHBOARD_PORT||"57374",Z=process.env.LOKI_PROVIDER||"claude",K=await E([$,"-c",C6,Q,z,X,Z],{timeoutMs:30000});if(K.exitCode!==0)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
54
- `),1;return process.stdout.write(K.stdout),0}async function m6($){let Q=[...$];while(Q.length>0){let z=Q[0];if(z==="--json")return g6();if(z==="--help"||z==="-h")return process.stdout.write(`Usage: loki status [--json]
52
+ `)}}function v6($){if(!S($))return[];try{return H1("fs").readdirSync($).filter((X)=>/^findings-\d+\.json$/.test(X)).sort((X,Z)=>{let K=Number.parseInt(X.replace(/[^0-9]/g,""),10)||0,W=Number.parseInt(Z.replace(/[^0-9]/g,""),10)||0;return K-W}).map((X)=>F($,X))}catch{return[]}}function U0($){try{let Q=H1("fs");return JSON.parse(Q.readFileSync($,"utf-8"))}catch{return null}}async function m6(){let $=await r();if(!$)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
53
+ `),1;let Q=u,z=P(),X=process.env.LOKI_DASHBOARD_PORT||"57374",Z=process.env.LOKI_PROVIDER||"claude",K=await E([$,"-c",C6,Q,z,X,Z],{timeoutMs:30000});if(K.exitCode!==0)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
54
+ `),1;return process.stdout.write(K.stdout),0}async function g6($){let Q=[...$];while(Q.length>0){let z=Q[0];if(z==="--json")return m6();if(z==="--help"||z==="-h")return process.stdout.write(`Usage: loki status [--json]
55
55
  `),0;return process.stdout.write(`${L}Unknown flag: ${z}${U}
56
56
  `),process.stdout.write(`Usage: loki status [--json]
57
57
  `),1}return y6()}var C6=`
@@ -261,9 +261,9 @@ if os.path.isfile(gate_count_file):
261
261
  result['phase1'] = phase1
262
262
 
263
263
  print(json.dumps(result, indent=2))
264
- `;var H0=S(()=>{p();X1();n();g()});var B0={};y(B0,{runStats:()=>d6,computeStats:()=>J0});import{readdirSync as G0,readFileSync as f6,statSync as V0}from"fs";import{join as l}from"path";function i($){try{if(!V0($).isFile())return null;return JSON.parse(f6($,"utf-8"))}catch{return null}}function k1($){try{return V0($).isDirectory()}catch{return!1}}function u6($){if(!k1($))return[];try{let Q=G0($).filter((z)=>z.startsWith("iteration-")&&z.endsWith(".json"));return Q.sort(),Q.map((z)=>l($,z))}catch{return[]}}function e($){return Math.trunc($).toLocaleString("en-US")}function D1($){let Q=Math.trunc($);if(Q<60)return`${Q}s`;let z=Math.trunc(Q/3600),X=Math.trunc(Q%3600/60),Z=Q%60;if(z>0)return`${z}h ${String(X).padStart(2,"0")}m`;return`${X}m ${String(Z).padStart(2,"0")}s`}function d($,Q=0){let z=Math.pow(10,Q);return Math.round($*z)/z}function $1($,Q){return $.toFixed(Q)}function C1($,Q){return $.length>=Q?$:$+" ".repeat(Q-$.length)}function p6($){let Q="N/A",z=0,X=i(l($,"state","orchestrator.json"));if(X&&typeof X==="object"){if(typeof X.currentPhase==="string")Q=X.currentPhase;if(typeof X.currentIteration==="number")z=X.currentIteration}let Z=l($,"metrics","efficiency"),K=u6(Z),W=[];for(let T of K){let _=i(T);if(_&&typeof _==="object")W.push(_)}if(W.length>0)z=Math.max(z,W.length);let q=W.reduce((T,_)=>T+(_.input_tokens??0),0),H=W.reduce((T,_)=>T+(_.output_tokens??0),0),G=q+H,J=W.reduce((T,_)=>T+(_.cost_usd??0),0),B=W.reduce((T,_)=>T+(_.duration_seconds??0),0),V=0,M=0,O=i(l($,"metrics","budget.json"));if(O&&typeof O==="object"){if(typeof O.budget_limit==="number")V=O.budget_limit;if(typeof O.budget_used==="number")M=O.budget_used}let j=0,w=0,m=i(l($,"state","quality-gates.json"));if(m&&typeof m==="object"){if(Array.isArray(m)){for(let T of m)if(w+=1,T===!0)j+=1;else if(T&&typeof T==="object"){let _=T;if(_.passed===!0||_.status==="passed")j+=1}}else for(let T of Object.values(m))if(typeof T==="boolean"){if(w+=1,T)j+=1}else if(T&&typeof T==="object"){w+=1;let _=T;if(_.passed===!0||_.status==="passed")j+=1}}let H1={},h=i(l($,"quality","gate-failure-count.json"));if(h&&typeof h==="object"&&!Array.isArray(h)){let T={};for(let[_,v]of Object.entries(h))if(typeof v==="number")T[_]=v;H1=T}let I=0,D=0,z1=0,R1=l($,"quality");if(k1(R1)){let T=[];try{T=G0(R1)}catch{T=[]}for(let _ of T){if(!_.endsWith(".json")||_==="gate-failure-count.json")continue;let v=i(l(R1,_));if(!v||typeof v!=="object")continue;if(!(("verdict"in v)||("approved"in v)||("reviewers"in v)))continue;I+=1;let o1=(v.verdict??"").toString().toLowerCase();if(v.approved===!0||["approved","approve","pass"].includes(o1))D+=1;else if(["revision","revise","changes_requested","reject"].includes(o1))z1+=1}}return{phase:Q,iterationCount:z,iterations:W,totalInput:q,totalOutput:H,totalTokens:G,totalCost:J,totalDuration:B,budgetLimit:V,budgetUsed:M,gatesPassed:j,gatesTotal:w,gateFailures:H1,reviewsTotal:I,reviewsApproved:D,reviewsRevision:z1}}function c6($,Q){let z=$.iterationCount,X={session:{iterations:z,duration_seconds:$.totalDuration,phase:$.phase},tokens:{input:$.totalInput,output:$.totalOutput,total:$.totalTokens,cost_usd:d($.totalCost,2)},quality:{gates_passed:$.gatesPassed,gates_total:$.gatesTotal,reviews_total:$.reviewsTotal,reviews_approved:$.reviewsApproved,reviews_revision:$.reviewsRevision,gate_failures:$.gateFailures},efficiency:{avg_tokens_per_iteration:z>0?d($.totalTokens/z,0):0,avg_cost_per_iteration:z>0?d($.totalCost/z,2):0,avg_duration_per_iteration:z>0?d($.totalDuration/z,1):0},budget:{used:d($.budgetUsed,2),limit:$.budgetLimit,percent:$.budgetLimit>0?d($.budgetUsed/$.budgetLimit*100,1):0}};if(Q)X.iterations=$.iterations.map((W,q)=>({number:q+1,input_tokens:W.input_tokens??0,output_tokens:W.output_tokens??0,cost_usd:d(W.cost_usd??0,2),duration_seconds:W.duration_seconds??0}));let Z=JSON.stringify(X,null,2);function K(W,q){if(!q)return;let H=new RegExp(`("${W}": )(-?\\d+)(,?)$`,"m");Z=Z.replace(H,(G,J,B,V)=>`${J}${B}.0${V}`)}if(K("avg_duration_per_iteration",z>0&&Number.isInteger(X.efficiency.avg_duration_per_iteration)),K("percent",$.budgetLimit>0&&Number.isInteger(X.budget.percent)),K("cost_usd",z>0&&Number.isInteger(X.tokens.cost_usd)),Q)Z=Z.replace(/("cost_usd": )(-?\d+)(,?)$/gm,(W,q,H,G)=>`${q}${H}.0${G}`);return Z}function l6($,Q){let z=[];if(z.push("Loki Mode Session Statistics"),z.push("============================"),z.push(""),z.push("Session"),z.push(` Iterations completed: ${$.iterationCount}`),z.push(` Duration: ${D1($.totalDuration)}`),z.push(` Current phase: ${$.phase}`),z.push(""),z.push("Token Usage"),$.iterations.length>0)z.push(` Input tokens: ${e($.totalInput)}`),z.push(` Output tokens: ${e($.totalOutput)}`),z.push(` Total tokens: ${e($.totalTokens)}`),z.push(` Estimated cost: $${$1($.totalCost,2)}`);else z.push(" N/A (no iteration metrics found)");if(z.push(""),z.push("Quality Gates"),$.gatesTotal>0){let X=Math.round($.gatesPassed/$.gatesTotal*100);z.push(` Gates passed: ${$.gatesPassed}/${$.gatesTotal} (${X}%)`)}else z.push(" Gates passed: N/A");if($.reviewsTotal>0){let X=[];if($.reviewsApproved>0)X.push(`${$.reviewsApproved} approved`);if($.reviewsRevision>0)X.push(`${$.reviewsRevision} revision requested`);let Z=X.length>0?X.join(", "):"N/A";z.push(` Code reviews: ${$.reviewsTotal} (${Z})`)}if(Object.keys($.gateFailures).length>0){let X=Object.entries($.gateFailures).filter(([,Z])=>Z>0).map(([Z,K])=>`${Z} (${K})`);if(X.length>0)z.push(` Gate failures: ${X.join(", ")}`)}if(z.push(""),z.push("Efficiency"),$.iterationCount>0&&$.iterations.length>0){let X=Math.round($.totalTokens/$.iterationCount),Z=$.totalCost/$.iterationCount,K=$.totalDuration/$.iterationCount;z.push(` Avg tokens/iteration: ${e(X)}`),z.push(` Avg cost/iteration: $${$1(Z,2)}`),z.push(` Avg duration/iteration: ${D1(K)}`)}else z.push(" N/A (no iteration metrics found)");if(z.push(""),z.push("Budget"),$.budgetLimit>0){let X=d($.budgetUsed/$.budgetLimit*100,1),Z=Number.isInteger(X)?`${X}.0`:`${X}`;z.push(` Used: $${$1($.budgetUsed,2)} / $${$1($.budgetLimit,2)} (${Z}%)`)}else if($.budgetUsed>0)z.push(` Used: $${$1($.budgetUsed,2)} (no limit set)`);else z.push(" N/A");if(Q&&$.iterations.length>0)z.push(""),z.push("Per-Iteration Breakdown"),$.iterations.forEach((X,Z)=>{let K=Z+1,W=C1(e(X.input_tokens??0),10),q=C1(e(X.output_tokens??0),10),H=X.cost_usd??0,G=D1(X.duration_seconds??0),J=C1(`${K}`,3);z.push(` #${J} input: ${W} output: ${q} cost: $${$1(H,2)} time: ${G}`)});return z.join(`
265
- `)}function J0($){let Q=!1,z=!1;for(let W of $)if(W==="--json")Q=!0;else if(W==="--efficiency")z=!0;let X=x();if(!k1(X)){if(Q)return{exitCode:0,stdout:'{"error": "No active session"}'};return{exitCode:0,stdout:`${R}No active session found.${U}
266
- Start a session with: loki start <prd>`}}let Z=p6(X);return{exitCode:0,stdout:Q?c6(Z,z):l6(Z,z)}}async function d6($){let Q=J0($);return console.log(Q.stdout),Q.exitCode}var Y0=S(()=>{g();n()});var P0={};y(P0,{runDoctor:()=>Z3,pythonImportOk:()=>h1,httpReachable:()=>b1,checkTool:()=>j0,checkSkills:()=>_0,checkDisk:()=>y1,buildDoctorJson:()=>L0,_setPythonImportOkForTest:()=>i6});import{existsSync as o6,lstatSync as n6,readlinkSync as a6,statfsSync as r6}from"fs";import{homedir as A0}from"os";import{resolve as M0}from"path";function t6($){let Q=$.match(s6);return Q?Q[1]:null}async function T0($){try{let Q=await E([$,"--version"],{timeoutMs:5000}),z=(Q.stdout||Q.stderr||"").trim();return t6(z)}catch{return null}}function O0($,Q){let z=$.split(".").map((Z)=>parseInt(Z,10)),X=Q.split(".").map((Z)=>parseInt(Z,10));while(z.length<2)z.push(0);while(X.length<2)X.push(0);for(let Z=0;Z<2;Z++){let K=z[Z]??0,W=X[Z]??0;if(Number.isNaN(K)||Number.isNaN(W))return 0;if(K!==W)return K-W}return 0}async function j0($,Q,z,X=null){let Z=await b(Q),K=Z!==null,W=K?await T0(Q):null,q="pass";if(!K)q=z==="required"?"fail":"warn";else if(X&&W){if(O0(W,X)<0)q=z==="required"?"fail":"warn"}return{name:$,command:Q,found:K,version:W,required:z,min_version:X,status:q,path:Z}}function y1(){let $=null;try{let z=r6(A0()),X=Number(z.bavail)*Number(z.bsize);$=Math.round(X/1073741824*10)/10}catch{$=null}let Q="pass";if($!==null){if($<1)Q="fail";else if($<5)Q="warn"}return{available_gb:$,status:Q}}async function b1($,Q=2000){try{return(await fetch($,{signal:AbortSignal.timeout(Q)})).ok}catch{return!1}}async function h1($,Q=!1){let z=`import ${$}`,X=Q?30000:5000;if(!Q)return(await a(z,{timeoutMs:X})).exitCode===0;let Z=await s();if(!Z)return!1;return(await E([Z,"-c",z],{timeoutMs:X})).exitCode===0}function i6($){A1.fn=$??h1}function _0(){let $=A0();return e6.map(({name:Q,dir:z})=>{let X=M0($,z),Z=X,K=M0(X,"SKILL.md");if(o6(K))return{name:Q,path:Z,status:"pass",detail:""};try{if(n6(X).isSymbolicLink()){let q="unknown";try{q=a6(X)}catch{}return{name:Q,path:Z,status:"fail",detail:`(broken symlink -> ${q})`}}}catch{}return{name:Q,path:Z,status:"warn",detail:"(not found - run 'loki setup-skill')"}})}async function I0(){return Promise.all($3.map(async($)=>{return{...await j0($.jsonName,$.cmd,$.required,$.min??null),displayName:$.displayName}}))}async function z3(){let Q=await b("sentrux")!==null,z=Q?await T0("sentrux"):null;return{found:Q,version:z,status:Q?"pass":"warn",required:"optional"}}async function L0(){let Q=(await I0()).map(({displayName:q,...H})=>H),z=y1(),X=await z3(),Z=0,K=0,W=0;for(let q of Q)if(q.status==="pass")Z++;else if(q.status==="fail")K++;else W++;if(z.status==="pass")Z++;else if(z.status==="fail")K++;else W++;return{checks:Q,disk:z,sentrux:X,summary:{passed:Z,failed:K,warnings:W,ok:K===0}}}function Y($){switch($){case"pass":return`${k}PASS${U}`;case"fail":return`${L}FAIL${U}`;case"warn":return`${R}WARN${U}`}}function Y1($){let Q=$.version?` (v${$.version})`:"",z=$.displayName;if(!$.found){let X=$.required==="required"?"not found":$.required==="recommended"?"not found (recommended)":"not found (optional)";return` ${Y($.status)} ${z} - ${X}`}if($.min_version&&$.version&&O0($.version,$.min_version)<0){let X=$.required==="required"?"requires":"recommended";return` ${Y($.status)} ${z}${Q} - ${X} >= ${$.min_version}`}return` ${Y($.status)} ${z}${Q}`}function M1($,Q){if(Q==="pass")$.pass++;else if(Q==="fail")$.fail++;else $.warn++}function Q3(){process.stdout.write(`${C}loki doctor${U} - Check system prerequisites
264
+ `;var H0=w(()=>{p();X1();n();v()});var B0={};y(B0,{runStats:()=>d6,computeStats:()=>J0});import{readdirSync as G0,readFileSync as f6,statSync as V0}from"fs";import{join as l}from"path";function i($){try{if(!V0($).isFile())return null;return JSON.parse(f6($,"utf-8"))}catch{return null}}function C1($){try{return V0($).isDirectory()}catch{return!1}}function u6($){if(!C1($))return[];try{let Q=G0($).filter((z)=>z.startsWith("iteration-")&&z.endsWith(".json"));return Q.sort(),Q.map((z)=>l($,z))}catch{return[]}}function e($){return Math.trunc($).toLocaleString("en-US")}function N1($){let Q=Math.trunc($);if(Q<60)return`${Q}s`;let z=Math.trunc(Q/3600),X=Math.trunc(Q%3600/60),Z=Q%60;if(z>0)return`${z}h ${String(X).padStart(2,"0")}m`;return`${X}m ${String(Z).padStart(2,"0")}s`}function d($,Q=0){let z=Math.pow(10,Q);return Math.round($*z)/z}function $1($,Q){return $.toFixed(Q)}function D1($,Q){return $.length>=Q?$:$+" ".repeat(Q-$.length)}function p6($){let Q="N/A",z=0,X=i(l($,"state","orchestrator.json"));if(X&&typeof X==="object"){if(typeof X.currentPhase==="string")Q=X.currentPhase;if(typeof X.currentIteration==="number")z=X.currentIteration}let Z=l($,"metrics","efficiency"),K=u6(Z),W=[];for(let T of K){let I=i(T);if(I&&typeof I==="object")W.push(I)}if(W.length>0)z=Math.max(z,W.length);let q=W.reduce((T,I)=>T+(I.input_tokens??0),0),H=W.reduce((T,I)=>T+(I.output_tokens??0),0),G=q+H,J=W.reduce((T,I)=>T+(I.cost_usd??0),0),B=W.reduce((T,I)=>T+(I.duration_seconds??0),0),V=0,M=0,j=i(l($,"metrics","budget.json"));if(j&&typeof j==="object"){if(typeof j.budget_limit==="number")V=j.budget_limit;if(typeof j.budget_used==="number")M=j.budget_used}let _=0,k=0,m=i(l($,"state","quality-gates.json"));if(m&&typeof m==="object"){if(Array.isArray(m)){for(let T of m)if(k+=1,T===!0)_+=1;else if(T&&typeof T==="object"){let I=T;if(I.passed===!0||I.status==="passed")_+=1}}else for(let T of Object.values(m))if(typeof T==="boolean"){if(k+=1,T)_+=1}else if(T&&typeof T==="object"){k+=1;let I=T;if(I.passed===!0||I.status==="passed")_+=1}}let g={},O=i(l($,"quality","gate-failure-count.json"));if(O&&typeof O==="object"&&!Array.isArray(O)){let T={};for(let[I,h]of Object.entries(O))if(typeof h==="number")T[I]=h;g=T}let N=0,z1=0,d1=0,x1=l($,"quality");if(C1(x1)){let T=[];try{T=G0(x1)}catch{T=[]}for(let I of T){if(!I.endsWith(".json")||I==="gate-failure-count.json")continue;let h=i(l(x1,I));if(!h||typeof h!=="object")continue;if(!(("verdict"in h)||("approved"in h)||("reviewers"in h)))continue;N+=1;let o1=(h.verdict??"").toString().toLowerCase();if(h.approved===!0||["approved","approve","pass"].includes(o1))z1+=1;else if(["revision","revise","changes_requested","reject"].includes(o1))d1+=1}}return{phase:Q,iterationCount:z,iterations:W,totalInput:q,totalOutput:H,totalTokens:G,totalCost:J,totalDuration:B,budgetLimit:V,budgetUsed:M,gatesPassed:_,gatesTotal:k,gateFailures:g,reviewsTotal:N,reviewsApproved:z1,reviewsRevision:d1}}function c6($,Q){let z=$.iterationCount,X={session:{iterations:z,duration_seconds:$.totalDuration,phase:$.phase},tokens:{input:$.totalInput,output:$.totalOutput,total:$.totalTokens,cost_usd:d($.totalCost,2)},quality:{gates_passed:$.gatesPassed,gates_total:$.gatesTotal,reviews_total:$.reviewsTotal,reviews_approved:$.reviewsApproved,reviews_revision:$.reviewsRevision,gate_failures:$.gateFailures},efficiency:{avg_tokens_per_iteration:z>0?d($.totalTokens/z,0):0,avg_cost_per_iteration:z>0?d($.totalCost/z,2):0,avg_duration_per_iteration:z>0?d($.totalDuration/z,1):0},budget:{used:d($.budgetUsed,2),limit:$.budgetLimit,percent:$.budgetLimit>0?d($.budgetUsed/$.budgetLimit*100,1):0}};if(Q)X.iterations=$.iterations.map((W,q)=>({number:q+1,input_tokens:W.input_tokens??0,output_tokens:W.output_tokens??0,cost_usd:d(W.cost_usd??0,2),duration_seconds:W.duration_seconds??0}));let Z=JSON.stringify(X,null,2);function K(W,q){if(!q)return;let H=new RegExp(`("${W}": )(-?\\d+)(,?)$`,"m");Z=Z.replace(H,(G,J,B,V)=>`${J}${B}.0${V}`)}if(K("avg_duration_per_iteration",z>0&&Number.isInteger(X.efficiency.avg_duration_per_iteration)),K("percent",$.budgetLimit>0&&Number.isInteger(X.budget.percent)),K("cost_usd",z>0&&Number.isInteger(X.tokens.cost_usd)),Q)Z=Z.replace(/("cost_usd": )(-?\d+)(,?)$/gm,(W,q,H,G)=>`${q}${H}.0${G}`);return Z}function l6($,Q){let z=[];if(z.push("Loki Mode Session Statistics"),z.push("============================"),z.push(""),z.push("Session"),z.push(` Iterations completed: ${$.iterationCount}`),z.push(` Duration: ${N1($.totalDuration)}`),z.push(` Current phase: ${$.phase}`),z.push(""),z.push("Token Usage"),$.iterations.length>0)z.push(` Input tokens: ${e($.totalInput)}`),z.push(` Output tokens: ${e($.totalOutput)}`),z.push(` Total tokens: ${e($.totalTokens)}`),z.push(` Estimated cost: $${$1($.totalCost,2)}`);else z.push(" N/A (no iteration metrics found)");if(z.push(""),z.push("Quality Gates"),$.gatesTotal>0){let X=Math.round($.gatesPassed/$.gatesTotal*100);z.push(` Gates passed: ${$.gatesPassed}/${$.gatesTotal} (${X}%)`)}else z.push(" Gates passed: N/A");if($.reviewsTotal>0){let X=[];if($.reviewsApproved>0)X.push(`${$.reviewsApproved} approved`);if($.reviewsRevision>0)X.push(`${$.reviewsRevision} revision requested`);let Z=X.length>0?X.join(", "):"N/A";z.push(` Code reviews: ${$.reviewsTotal} (${Z})`)}if(Object.keys($.gateFailures).length>0){let X=Object.entries($.gateFailures).filter(([,Z])=>Z>0).map(([Z,K])=>`${Z} (${K})`);if(X.length>0)z.push(` Gate failures: ${X.join(", ")}`)}if(z.push(""),z.push("Efficiency"),$.iterationCount>0&&$.iterations.length>0){let X=Math.round($.totalTokens/$.iterationCount),Z=$.totalCost/$.iterationCount,K=$.totalDuration/$.iterationCount;z.push(` Avg tokens/iteration: ${e(X)}`),z.push(` Avg cost/iteration: $${$1(Z,2)}`),z.push(` Avg duration/iteration: ${N1(K)}`)}else z.push(" N/A (no iteration metrics found)");if(z.push(""),z.push("Budget"),$.budgetLimit>0){let X=d($.budgetUsed/$.budgetLimit*100,1),Z=Number.isInteger(X)?`${X}.0`:`${X}`;z.push(` Used: $${$1($.budgetUsed,2)} / $${$1($.budgetLimit,2)} (${Z}%)`)}else if($.budgetUsed>0)z.push(` Used: $${$1($.budgetUsed,2)} (no limit set)`);else z.push(" N/A");if(Q&&$.iterations.length>0)z.push(""),z.push("Per-Iteration Breakdown"),$.iterations.forEach((X,Z)=>{let K=Z+1,W=D1(e(X.input_tokens??0),10),q=D1(e(X.output_tokens??0),10),H=X.cost_usd??0,G=N1(X.duration_seconds??0),J=D1(`${K}`,3);z.push(` #${J} input: ${W} output: ${q} cost: $${$1(H,2)} time: ${G}`)});return z.join(`
265
+ `)}function J0($){let Q=!1,z=!1;for(let W of $)if(W==="--json")Q=!0;else if(W==="--efficiency")z=!0;let X=P();if(!C1(X)){if(Q)return{exitCode:0,stdout:'{"error": "No active session"}'};return{exitCode:0,stdout:`${x}No active session found.${U}
266
+ Start a session with: loki start <prd>`}}let Z=p6(X);return{exitCode:0,stdout:Q?c6(Z,z):l6(Z,z)}}async function d6($){let Q=J0($);return console.log(Q.stdout),Q.exitCode}var Y0=w(()=>{v();n()});var P0={};y(P0,{runDoctor:()=>Z3,pythonImportOk:()=>y1,httpReachable:()=>k1,checkTool:()=>j0,checkSkills:()=>_0,checkDisk:()=>b1,buildDoctorJson:()=>L0,_setPythonImportOkForTest:()=>i6});import{existsSync as o6,lstatSync as n6,readlinkSync as a6,statfsSync as s6}from"fs";import{homedir as A0}from"os";import{resolve as M0}from"path";function t6($){let Q=$.match(r6);return Q?Q[1]:null}async function T0($){try{let Q=await E([$,"--version"],{timeoutMs:5000}),z=(Q.stdout||Q.stderr||"").trim();return t6(z)}catch{return null}}function O0($,Q){let z=$.split(".").map((Z)=>parseInt(Z,10)),X=Q.split(".").map((Z)=>parseInt(Z,10));while(z.length<2)z.push(0);while(X.length<2)X.push(0);for(let Z=0;Z<2;Z++){let K=z[Z]??0,W=X[Z]??0;if(Number.isNaN(K)||Number.isNaN(W))return 0;if(K!==W)return K-W}return 0}async function j0($,Q,z,X=null){let Z=await b(Q),K=Z!==null,W=K?await T0(Q):null,q="pass";if(!K)q=z==="required"?"fail":"warn";else if(X&&W){if(O0(W,X)<0)q=z==="required"?"fail":"warn"}return{name:$,command:Q,found:K,version:W,required:z,min_version:X,status:q,path:Z}}function b1(){let $=null;try{let z=s6(A0()),X=Number(z.bavail)*Number(z.bsize);$=Math.round(X/1073741824*10)/10}catch{$=null}let Q="pass";if($!==null){if($<1)Q="fail";else if($<5)Q="warn"}return{available_gb:$,status:Q}}async function k1($,Q=2000){try{return(await fetch($,{signal:AbortSignal.timeout(Q)})).ok}catch{return!1}}async function y1($,Q=!1){let z=`import ${$}`,X=Q?30000:5000;if(!Q)return(await a(z,{timeoutMs:X})).exitCode===0;let Z=await r();if(!Z)return!1;return(await E([Z,"-c",z],{timeoutMs:X})).exitCode===0}function i6($){M1.fn=$??y1}function _0(){let $=A0();return e6.map(({name:Q,dir:z})=>{let X=M0($,z),Z=X,K=M0(X,"SKILL.md");if(o6(K))return{name:Q,path:Z,status:"pass",detail:""};try{if(n6(X).isSymbolicLink()){let q="unknown";try{q=a6(X)}catch{}return{name:Q,path:Z,status:"fail",detail:`(broken symlink -> ${q})`}}}catch{}return{name:Q,path:Z,status:"warn",detail:"(not found - run 'loki setup-skill')"}})}async function I0(){return Promise.all($3.map(async($)=>{return{...await j0($.jsonName,$.cmd,$.required,$.min??null),displayName:$.displayName}}))}async function z3(){let Q=await b("sentrux")!==null,z=Q?await T0("sentrux"):null;return{found:Q,version:z,status:Q?"pass":"warn",required:"optional"}}async function L0(){let Q=(await I0()).map(({displayName:q,...H})=>H),z=b1(),X=await z3(),Z=0,K=0,W=0;for(let q of Q)if(q.status==="pass")Z++;else if(q.status==="fail")K++;else W++;if(z.status==="pass")Z++;else if(z.status==="fail")K++;else W++;return{checks:Q,disk:z,sentrux:X,summary:{passed:Z,failed:K,warnings:W,ok:K===0}}}function Y($){switch($){case"pass":return`${C}PASS${U}`;case"fail":return`${L}FAIL${U}`;case"warn":return`${x}WARN${U}`}}function B1($){let Q=$.version?` (v${$.version})`:"",z=$.displayName;if(!$.found){let X=$.required==="required"?"not found":$.required==="recommended"?"not found (recommended)":"not found (optional)";return` ${Y($.status)} ${z} - ${X}`}if($.min_version&&$.version&&O0($.version,$.min_version)<0){let X=$.required==="required"?"requires":"recommended";return` ${Y($.status)} ${z}${Q} - ${X} >= ${$.min_version}`}return` ${Y($.status)} ${z}${Q}`}function Y1($,Q){if(Q==="pass")$.pass++;else if(Q==="fail")$.fail++;else $.warn++}function Q3(){process.stdout.write(`${D}loki doctor${U} - Check system prerequisites
267
267
 
268
268
  `),process.stdout.write(`Usage: loki doctor [--json]
269
269
 
@@ -271,92 +271,93 @@ Start a session with: loki start <prd>`}}let Z=p6(X);return{exitCode:0,stdout:Q?
271
271
  `),process.stdout.write(` --json Output machine-readable JSON
272
272
 
273
273
  `),process.stdout.write(`Checks: node, python3, jq, git, curl, bash version,
274
- `),process.stdout.write(` claude/codex/gemini CLIs, and disk space.
275
- `)}async function X3(){process.stdout.write(`${C}Loki Mode Doctor${U}
274
+ `),process.stdout.write(` claude/codex CLIs, and disk space.
275
+ `)}async function X3(){process.stdout.write(`${D}Loki Mode Doctor${U}
276
276
 
277
277
  `),process.stdout.write(`Checking system prerequisites...
278
278
 
279
- `);let $={pass:0,fail:0,warn:0},Q=await I0(),z=new Map(Q.map((I)=>[I.command,I]));process.stdout.write(`${A}Required:${U}
280
- `);for(let I of["node","python3","jq","git","curl"]){let D=z.get(I);process.stdout.write(Y1(D)+`
281
- `),M1($,D.status)}process.stdout.write(`
279
+ `);let $={pass:0,fail:0,warn:0},Q=await I0(),z=new Map(Q.map((O)=>[O.command,O]));process.stdout.write(`${A}Required:${U}
280
+ `);for(let O of["node","python3","jq","git","curl"]){let N=z.get(O);process.stdout.write(B1(N)+`
281
+ `),Y1($,N.status)}process.stdout.write(`
282
282
  `),process.stdout.write(`${A}AI Providers:${U}
283
- `);let X=["claude","codex","gemini","cline","aider"],Z=!1;for(let I of X){let D=z.get(I);if(process.stdout.write(Y1(D)+`
284
- `),M1($,D.status),D.found)Z=!0}if(!Z)process.stdout.write(` ${Y("fail")} No AI provider CLI installed -- at least one is required
285
- `),process.stdout.write(` ${R}Install: npm install -g @anthropic-ai/claude-code${U}
283
+ `);let X=["claude","codex","cline","aider"],Z=!1;for(let O of X){let N=z.get(O);if(process.stdout.write(B1(N)+`
284
+ `),Y1($,N.status),N.found)Z=!0}if(!Z)process.stdout.write(` ${Y("fail")} No AI provider CLI installed -- at least one is required
285
+ `),process.stdout.write(` ${x}Install: npm install -g @anthropic-ai/claude-code${U}
286
286
  `),$.fail++;process.stdout.write(`
287
287
  `),process.stdout.write(`${A}API Keys:${U}
288
- `);let K=z.get("claude")?.found??!1,W=z.get("codex")?.found??!1,q=z.get("gemini")?.found??!1,H=process.env;if(H.ANTHROPIC_API_KEY)process.stdout.write(` ${Y("pass")} ANTHROPIC_API_KEY is set
289
- `),$.pass++;else if(K)process.stdout.write(` ${P} -- ${U} ANTHROPIC_API_KEY not set (Claude CLI uses its own login)
290
- `);if(H.OPENAI_API_KEY)process.stdout.write(` ${Y("pass")} OPENAI_API_KEY is set
291
- `),$.pass++;else if(W)process.stdout.write(` ${P} -- ${U} OPENAI_API_KEY not set (Codex CLI uses its own login)
292
- `);if(H.GOOGLE_API_KEY||H.GEMINI_API_KEY)process.stdout.write(` ${Y("pass")} GOOGLE_API_KEY is set
293
- `),$.pass++;else if(q)process.stdout.write(` ${P} -- ${U} GOOGLE_API_KEY not set (Gemini CLI uses its own login)
294
- `);process.stdout.write(`
288
+ `);let K=z.get("claude")?.found??!1,W=z.get("codex")?.found??!1,q=process.env;if(q.ANTHROPIC_API_KEY)process.stdout.write(` ${Y("pass")} ANTHROPIC_API_KEY is set
289
+ `),$.pass++;else if(K)process.stdout.write(` ${R} -- ${U} ANTHROPIC_API_KEY not set (Claude CLI uses its own login)
290
+ `);if(q.OPENAI_API_KEY)process.stdout.write(` ${Y("pass")} OPENAI_API_KEY is set
291
+ `),$.pass++;else if(W)process.stdout.write(` ${R} -- ${U} OPENAI_API_KEY not set (Codex CLI uses its own login)
292
+ `);if(q.ANTHROPIC_BASE_URL){let O=q.ANTHROPIC_BASE_URL;if(process.stdout.write(` ${Y("pass")} ANTHROPIC_BASE_URL: ${O}
293
+ `),$.pass++,!q.LOKI_MODEL_OVERRIDE)process.stdout.write(` ${Y("warn")} LOKI_MODEL_OVERRIDE not set -- opus/sonnet/haiku aliases may not resolve on alt-provider
294
+ `),$.warn++;else process.stdout.write(` ${Y("pass")} LOKI_MODEL_OVERRIDE: ${q.LOKI_MODEL_OVERRIDE}
295
+ `),$.pass++}process.stdout.write(`
295
296
  `),process.stdout.write(`${A}Skills:${U}
296
- `);for(let I of _0())if(I.status==="pass")process.stdout.write(` ${Y("pass")} ${I.name} ${P}${I.path}${U}
297
- `),$.pass++;else if(I.status==="fail")process.stdout.write(` ${Y("fail")} ${I.name} ${P}${I.detail}${U}
298
- `),process.stdout.write(` ${R}Fix: loki setup-skill${U}
299
- `),$.fail++;else process.stdout.write(` ${Y("warn")} ${I.name} ${P}${I.detail}${U}
297
+ `);for(let O of _0())if(O.status==="pass")process.stdout.write(` ${Y("pass")} ${O.name} ${R}${O.path}${U}
298
+ `),$.pass++;else if(O.status==="fail")process.stdout.write(` ${Y("fail")} ${O.name} ${R}${O.detail}${U}
299
+ `),process.stdout.write(` ${x}Fix: loki setup-skill${U}
300
+ `),$.fail++;else process.stdout.write(` ${Y("warn")} ${O.name} ${R}${O.detail}${U}
300
301
  `),$.warn++;process.stdout.write(`
301
302
  `),process.stdout.write(`${A}Integrations:${U}
302
- `);let[G,J,B]=await Promise.all([A1.fn("mcp"),A1.fn("numpy",!0),A1.fn("sentence_transformers",!0)]);if(G)process.stdout.write(` ${Y("pass")} MCP SDK (Python)
303
+ `);let[H,G,J]=await Promise.all([M1.fn("mcp"),M1.fn("numpy",!0),M1.fn("sentence_transformers",!0)]);if(H)process.stdout.write(` ${Y("pass")} MCP SDK (Python)
303
304
  `),$.pass++;else process.stdout.write(` ${Y("warn")} MCP SDK - not installed (pip3 install mcp)
304
- `),$.warn++;if(J)process.stdout.write(` ${Y("pass")} numpy (vector search)
305
+ `),$.warn++;if(G)process.stdout.write(` ${Y("pass")} numpy (vector search)
305
306
  `),$.pass++;else process.stdout.write(` ${Y("warn")} numpy - not installed (pip3 install numpy)
306
- `),$.warn++;if(B)process.stdout.write(` ${Y("pass")} sentence-transformers (embeddings)
307
+ `),$.warn++;if(J)process.stdout.write(` ${Y("pass")} sentence-transformers (embeddings)
307
308
  `),$.pass++;else process.stdout.write(` ${Y("warn")} sentence-transformers - not installed (loki memory vectors setup)
308
- `),$.warn++;if(await b1("http://localhost:8100/api/v2/heartbeat"))process.stdout.write(` ${Y("pass")} ChromaDB server (port 8100)
309
+ `),$.warn++;if(await k1("http://localhost:8100/api/v2/heartbeat"))process.stdout.write(` ${Y("pass")} ChromaDB server (port 8100)
309
310
  `),$.pass++;else process.stdout.write(` ${Y("warn")} ChromaDB - not running (docker start loki-chroma)
310
- `),$.warn++;let V=process.env.LOKI_MIROFISH_URL;if(V)if(await b1(`${V}/health`))process.stdout.write(` ${Y("pass")} MiroFish server (${V})
311
+ `),$.warn++;let B=process.env.LOKI_MIROFISH_URL;if(B)if(await k1(`${B}/health`))process.stdout.write(` ${Y("pass")} MiroFish server (${B})
311
312
  `),$.pass++;else process.stdout.write(` ${Y("warn")} MiroFish - not running (loki start --mirofish-docker <image>)
312
313
  `),$.warn++;if(process.env.LOKI_OTEL_ENDPOINT)process.stdout.write(` ${Y("pass")} OTEL endpoint: ${process.env.LOKI_OTEL_ENDPOINT}
313
314
  `),$.pass++;else process.stdout.write(` ${Y("warn")} OTEL - not configured (set LOKI_OTEL_ENDPOINT)
314
- `),$.warn++;if(await b("sentrux")){let I="unknown";try{let z1=(await E(["sentrux","--version"],{timeoutMs:2000})).stdout.split(/\s+/).filter(Boolean).pop();if(z1)I=z1.replace(/^v/,"")}catch{}process.stdout.write(` ${Y("pass")} sentrux ${I} (architectural drift gate: loki sentrux help)
315
+ `),$.warn++;if(await b("sentrux")){let O="unknown";try{let z1=(await E(["sentrux","--version"],{timeoutMs:2000})).stdout.split(/\s+/).filter(Boolean).pop();if(z1)O=z1.replace(/^v/,"")}catch{}process.stdout.write(` ${Y("pass")} sentrux ${O} (architectural drift gate: loki sentrux help)
315
316
  `),$.pass++}else process.stdout.write(` ${Y("warn")} sentrux - not installed (optional, brew install sentrux/tap/sentrux)
316
317
  `),$.warn++;process.stdout.write(`
317
318
  `),process.stdout.write(`${A}System:${U}
318
- `);let M=z.get("bash");process.stdout.write(Y1(M)+`
319
- `),M1($,M.status);let O=z.get("bun");if(O)process.stdout.write(Y1(O)+`
320
- `),M1($,O.status);let j=y1(),w=j.available_gb===null?null:Math.floor(j.available_gb);if(w===null)process.stdout.write(` ${Y("warn")} Disk space: unable to determine
321
- `),$.warn++;else if(j.status==="fail")process.stdout.write(` ${Y("fail")} Disk space: ${w}GB available (need >= 1GB)
322
- `),$.fail++;else if(j.status==="warn")process.stdout.write(` ${Y("warn")} Disk space: ${w}GB available (low)
323
- `),$.warn++;else process.stdout.write(` ${Y("pass")} Disk space: ${w}GB available
319
+ `);let V=z.get("bash");process.stdout.write(B1(V)+`
320
+ `),Y1($,V.status);let M=z.get("bun");if(M)process.stdout.write(B1(M)+`
321
+ `),Y1($,M.status);let j=b1(),_=j.available_gb===null?null:Math.floor(j.available_gb);if(_===null)process.stdout.write(` ${Y("warn")} Disk space: unable to determine
322
+ `),$.warn++;else if(j.status==="fail")process.stdout.write(` ${Y("fail")} Disk space: ${_}GB available (need >= 1GB)
323
+ `),$.fail++;else if(j.status==="warn")process.stdout.write(` ${Y("warn")} Disk space: ${_}GB available (low)
324
+ `),$.warn++;else process.stdout.write(` ${Y("pass")} Disk space: ${_}GB available
324
325
  `),$.pass++;process.stdout.write(`
325
326
  `),process.stdout.write(`${A}Runtime route:${U}
326
- `);let m=process.versions.bun!==void 0,H1=process.argv[0]??"(unknown)";if(process.stdout.write(` ${Y("pass")} Active runtime: ${m?"Bun":"Node"} (${H1})
327
+ `);let k=process.versions.bun!==void 0,m=process.argv[0]??"(unknown)";if(process.stdout.write(` ${Y("pass")} Active runtime: ${k?"Bun":"Node"} (${m})
327
328
  `),process.env.LOKI_LEGACY_BASH==="1"||process.env.LOKI_LEGACY_BASH==="true")process.stdout.write(` ${Y("warn")} LOKI_LEGACY_BASH set: shim routes every command to autonomy/loki (bash)
328
329
  `);if(process.env.LOKI_TS_ENTRY)process.stdout.write(` ${Y("pass")} LOKI_TS_ENTRY override: ${process.env.LOKI_TS_ENTRY}
329
330
  `);if(process.env.BUN_FROM_SOURCE==="1"||process.env.BUN_FROM_SOURCE==="true")process.stdout.write(` ${Y("pass")} BUN_FROM_SOURCE set: shim prefers loki-ts/src/ over dist/
330
- `);let h=await s();if(h!==null){let D=(await E([h,"-c","import sys; print(f'{sys.version_info.major}.{sys.version_info.minor}.{sys.version_info.micro}')"],{timeoutMs:5000})).stdout.trim();if(D.startsWith("3.12"))process.stdout.write(` ${Y("pass")} Python 3.12 (chromadb / sentence-transformers): ${D} at ${h}
331
- `);else if(D)process.stdout.write(` ${Y("warn")} Python 3.12 NOT found -- using ${D} at ${h}; chromadb / sentence-transformers may fail. Install python3.12 (brew install python@3.12 / apt install python3.12).
332
- `);else process.stdout.write(` ${Y("warn")} Python 3 found at ${h} but version probe failed; chromadb may not work.
331
+ `);let g=await r();if(g!==null){let N=(await E([g,"-c","import sys; print(f'{sys.version_info.major}.{sys.version_info.minor}.{sys.version_info.micro}')"],{timeoutMs:5000})).stdout.trim();if(N.startsWith("3.12"))process.stdout.write(` ${Y("pass")} Python 3.12 (chromadb / sentence-transformers): ${N} at ${g}
332
+ `);else if(N)process.stdout.write(` ${Y("warn")} Python 3.12 NOT found -- using ${N} at ${g}; chromadb / sentence-transformers may fail. Install python3.12 (brew install python@3.12 / apt install python3.12).
333
+ `);else process.stdout.write(` ${Y("warn")} Python 3 found at ${g} but version probe failed; chromadb may not work.
333
334
  `)}else process.stdout.write(` ${Y("warn")} Python 3 not on PATH -- memory + MCP integrations disabled.
334
335
  `);if(process.stdout.write(`
335
- `),process.stdout.write(`${C}Summary:${U} ${k}${$.pass} passed${U}, ${L}${$.fail} failed${U}, ${R}${$.warn} warnings${U}
336
+ `),process.stdout.write(`${D}Summary:${U} ${C}${$.pass} passed${U}, ${L}${$.fail} failed${U}, ${x}${$.warn} warnings${U}
336
337
 
337
338
  `),$.fail>0)return process.stdout.write(`${L}Some required prerequisites are missing.${U}
338
339
  `),process.stdout.write(`Install missing dependencies and run 'loki doctor' again.
339
- `),1;if($.warn>0)return process.stdout.write(`${R}All required checks passed with some warnings.${U}
340
- `),0;return process.stdout.write(`${k}All checks passed. System is ready for Loki Mode.${U}
340
+ `),1;if($.warn>0)return process.stdout.write(`${x}All required checks passed with some warnings.${U}
341
+ `),0;return process.stdout.write(`${C}All checks passed. System is ready for Loki Mode.${U}
341
342
  `),0}async function Z3($){let Q=!1;for(let z of $)if(z==="--json")Q=!0;else if(z==="--help"||z==="-h")return Q3(),0;else return process.stderr.write(`${L}Unknown option: ${z}${U}
342
343
  `),process.stderr.write(`Usage: loki doctor [--json]
343
344
  `),1;if(Q){let z=await L0();return process.stdout.write(JSON.stringify(z,null,2)+`
344
- `),0}return X3()}var s6,A1,e6,$3;var x0=S(()=>{p();X1();n();s6=/(\d+\.\d+(?:\.\d+)*)/;A1={fn:h1};e6=[{name:"Claude Code",dir:".claude/skills/loki-mode"},{name:"Codex CLI",dir:".codex/skills/loki-mode"},{name:"Gemini CLI",dir:".gemini/skills/loki-mode"},{name:"Cline CLI",dir:".cline/skills/loki-mode"},{name:"Aider CLI",dir:".aider/skills/loki-mode"}];$3=[{displayName:"Node.js (>= 18)",jsonName:"Node.js",cmd:"node",required:"required",min:"18.0"},{displayName:"Python 3 (>= 3.8)",jsonName:"Python 3",cmd:"python3",required:"required",min:"3.8"},{displayName:"jq",jsonName:"jq",cmd:"jq",required:"required"},{displayName:"git",jsonName:"git",cmd:"git",required:"required"},{displayName:"curl",jsonName:"curl",cmd:"curl",required:"required"},{displayName:"bash (>= 4.0)",jsonName:"bash",cmd:"bash",required:"recommended",min:"4.0"},{displayName:"Bun (>= 1.3)",jsonName:"Bun",cmd:"bun",required:"recommended",min:"1.3"},{displayName:"Claude CLI",jsonName:"Claude CLI",cmd:"claude",required:"optional"},{displayName:"Codex CLI",jsonName:"Codex CLI",cmd:"codex",required:"optional"},{displayName:"Gemini CLI",jsonName:"Gemini CLI",cmd:"gemini",required:"optional"},{displayName:"Cline CLI",jsonName:"Cline CLI",cmd:"cline",required:"optional"},{displayName:"Aider CLI",jsonName:"Aider CLI",cmd:"aider",required:"optional"}]});import{closeSync as T7,fstatSync as O7,lstatSync as j7,mkdirSync as K3,openSync as _7,readSync as I7,renameSync as W3,rmSync as L7,statSync as P7,unlinkSync as x7,writeFileSync as U3,writeSync as R7}from"fs";import{dirname as q3}from"path";function W1($,Q){K3(q3($),{recursive:!0});let z=`${$}.tmp.${process.pid}.${++H3}`;U3(z,`${JSON.stringify(Q,null,2)}
345
- `),W3(z,$)}async function R0($,Q){let z=T1.get($)??Promise.resolve(),X=()=>{},Z=new Promise((W)=>{X=W}),K=z.catch(()=>{}).then(()=>Z);T1.set($,K);try{return await z.catch(()=>{}),await Q()}finally{if(X(),T1.get($)===K)T1.delete($)}}var H3=0,T1;var O1=S(()=>{T1=new Map});import{existsSync as j1,mkdirSync as G3,copyFileSync as V3,readFileSync as J3,readdirSync as B3,statSync as Y3,writeFileSync as N7,renameSync as M3,appendFileSync as D7,rmSync as C7}from"fs";import{join as r,dirname as A3}from"path";function _1($){return r($,"state","checkpoints")}function O3($){let Q=_1($);if(!j1(Q))return[];return B3(Q).filter((z)=>z.startsWith("cp-")).filter((z)=>{try{return Y3(r(Q,z)).isDirectory()}catch{return!1}})}function j3($){return[...$].sort((Q,z)=>{let X=F0(Q),Z=F0(z);return X-Z})}function F0($){let Q=$.split("-");if(Q.length<3)return 0;let z=Q[Q.length-1],X=Number.parseInt(z??"0",10);return Number.isFinite(X)?X:0}function g1($){let Q=$??x(),z=j3(O3(Q)),X=[];for(let Z of z){let K=E0(Q,Z);if(K)X.push(K)}return X}function E0($,Q){let z=r(_1($),Q,"metadata.json");if(!j1(z))return null;try{let X=JSON.parse(J3(z,"utf-8"));return _3(X,z)}catch{return null}}function _3($,Q){let z=P3($,Q);return z.ok?z.value:null}function P3($,Q){if($===null||typeof $!=="object")return console.warn(`[checkpoint] invalid metadata at ${Q}: not an object`),{ok:!1,reason:"invalid_type",field:"<root>"};let z=$,X=["id","timestamp","task_id","task_description","git_sha","git_branch","provider","phase"];for(let Z of X){if(!(Z in z))return console.warn(`[checkpoint] invalid metadata at ${Q}: field "${Z}" missing`),{ok:!1,reason:"missing_field",field:Z};if(typeof z[Z]!=="string")return console.warn(`[checkpoint] invalid metadata at ${Q}: field "${Z}" not a string`),{ok:!1,reason:"invalid_type",field:Z}}if(!Object.prototype.hasOwnProperty.call(z,"iteration"))return console.warn(`[checkpoint] invalid metadata at ${Q}: field "iteration" missing`),{ok:!1,reason:"missing_field",field:"iteration"};if(typeof z.iteration!=="number"||!Number.isFinite(z.iteration))return console.warn(`[checkpoint] invalid metadata at ${Q}: field "iteration" not a finite number`),{ok:!1,reason:"invalid_type",field:"iteration"};for(let Z of L3){let K=z[Z];if(I3.test(K))return console.warn(`[checkpoint] invalid metadata at ${Q}: field "${Z}" contains control characters`),{ok:!1,reason:"control_chars",field:Z}}return{ok:!0,value:{id:z.id,timestamp:z.timestamp,iteration:z.iteration,task_id:z.task_id,task_description:z.task_description,git_sha:z.git_sha,git_branch:z.git_branch,provider:z.provider,phase:z.phase}}}function m1($,Q){if(!x3.test($))throw new w0($);let z=Q??x(),X=r(_1(z),$);if(!j1(X))throw new v1($);let Z=E0(z,$);if(!Z)throw new v1($);return Z}function S0($,Q){let z=m1($,Q),X=Q??x(),Z=r(_1(X),$),K=[];for(let W of R3){let q=r(Z,W);if(!j1(q))continue;K.push({from:q,to:r(X,W)})}return{id:$,metadata:z,restore:K}}function N0($){let Q=[],z=0;for(let X of $.restore)try{G3(A3(X.to),{recursive:!0});let Z=`${X.to}.tmp.${process.pid}.${++T3}`;V3(X.from,Z),M3(Z,X.to),z+=1}catch(Z){Q.push(`${X.from} -> ${X.to}: ${Z.message}`)}return{restored:z,errors:Q}}var v7,T3=0,I3,L3,x3,v1,w0,R3;var D0=S(()=>{g();p();O1();v7=Promise.resolve();I3=/[\x00-\x08\x0a-\x1f\x7f-\x9f]/,L3=["id","task_id","git_sha","git_branch","provider","phase"];x3=/^[a-zA-Z0-9_-]+$/;v1=class v1 extends Error{id;constructor($){super(`Checkpoint not found: ${$}`);this.id=$;this.name="CheckpointNotFoundError"}};w0=class w0 extends Error{id;constructor($){super(`Invalid checkpoint ID: must be alphanumeric, hyphens, underscores only (got: ${$})`);this.id=$;this.name="InvalidCheckpointIdError"}};R3=["state/orchestrator.json","queue/pending.json","queue/completed.json","queue/in-progress.json","queue/current-task.json"]});var b0={};y(b0,{runRollback:()=>F3});async function F3($){let Q=$[0],z=$.slice(1);if(Q===void 0||Q==="help"||Q==="--help"||Q==="-h")return process.stdout.write(C0),Q===void 0?1:0;switch(Q){case"list":{let X=[...g1()].reverse();if(X.length===0)return process.stdout.write(`${R}No checkpoints found.${U}
346
- `),0;process.stdout.write(`${C}Checkpoints${U} (${X.length}, newest first):
345
+ `),0}return X3()}var r6,M1,e6,$3;var x0=w(()=>{p();X1();n();r6=/(\d+\.\d+(?:\.\d+)*)/;M1={fn:y1};e6=[{name:"Claude Code",dir:".claude/skills/loki-mode"},{name:"Codex CLI",dir:".codex/skills/loki-mode"},{name:"Cline CLI",dir:".cline/skills/loki-mode"},{name:"Aider CLI",dir:".aider/skills/loki-mode"}];$3=[{displayName:"Node.js (>= 18)",jsonName:"Node.js",cmd:"node",required:"required",min:"18.0"},{displayName:"Python 3 (>= 3.8)",jsonName:"Python 3",cmd:"python3",required:"required",min:"3.8"},{displayName:"jq",jsonName:"jq",cmd:"jq",required:"required"},{displayName:"git",jsonName:"git",cmd:"git",required:"required"},{displayName:"curl",jsonName:"curl",cmd:"curl",required:"required"},{displayName:"bash (>= 4.0)",jsonName:"bash",cmd:"bash",required:"recommended",min:"4.0"},{displayName:"Bun (>= 1.3)",jsonName:"Bun",cmd:"bun",required:"recommended",min:"1.3"},{displayName:"Claude CLI",jsonName:"Claude CLI",cmd:"claude",required:"optional"},{displayName:"Codex CLI",jsonName:"Codex CLI",cmd:"codex",required:"optional"},{displayName:"Cline CLI",jsonName:"Cline CLI",cmd:"cline",required:"optional"},{displayName:"Aider CLI",jsonName:"Aider CLI",cmd:"aider",required:"optional"}]});import{closeSync as T7,fstatSync as O7,lstatSync as j7,mkdirSync as K3,openSync as _7,readSync as I7,renameSync as W3,rmSync as L7,statSync as P7,unlinkSync as x7,writeFileSync as U3,writeSync as R7}from"fs";import{dirname as q3}from"path";function W1($,Q){K3(q3($),{recursive:!0});let z=`${$}.tmp.${process.pid}.${++H3}`;U3(z,`${JSON.stringify(Q,null,2)}
346
+ `),W3(z,$)}async function R0($,Q){let z=A1.get($)??Promise.resolve(),X=()=>{},Z=new Promise((W)=>{X=W}),K=z.catch(()=>{}).then(()=>Z);A1.set($,K);try{return await z.catch(()=>{}),await Q()}finally{if(X(),A1.get($)===K)A1.delete($)}}var H3=0,A1;var T1=w(()=>{A1=new Map});import{existsSync as O1,mkdirSync as G3,copyFileSync as V3,readFileSync as J3,readdirSync as B3,statSync as Y3,writeFileSync as N7,renameSync as M3,appendFileSync as D7,rmSync as C7}from"fs";import{join as s,dirname as A3}from"path";function j1($){return s($,"state","checkpoints")}function O3($){let Q=j1($);if(!O1(Q))return[];return B3(Q).filter((z)=>z.startsWith("cp-")).filter((z)=>{try{return Y3(s(Q,z)).isDirectory()}catch{return!1}})}function j3($){return[...$].sort((Q,z)=>{let X=F0(Q),Z=F0(z);return X-Z})}function F0($){let Q=$.split("-");if(Q.length<3)return 0;let z=Q[Q.length-1],X=Number.parseInt(z??"0",10);return Number.isFinite(X)?X:0}function v1($){let Q=$??P(),z=j3(O3(Q)),X=[];for(let Z of z){let K=E0(Q,Z);if(K)X.push(K)}return X}function E0($,Q){let z=s(j1($),Q,"metadata.json");if(!O1(z))return null;try{let X=JSON.parse(J3(z,"utf-8"));return _3(X,z)}catch{return null}}function _3($,Q){let z=P3($,Q);return z.ok?z.value:null}function P3($,Q){if($===null||typeof $!=="object")return console.warn(`[checkpoint] invalid metadata at ${Q}: not an object`),{ok:!1,reason:"invalid_type",field:"<root>"};let z=$,X=["id","timestamp","task_id","task_description","git_sha","git_branch","provider","phase"];for(let Z of X){if(!(Z in z))return console.warn(`[checkpoint] invalid metadata at ${Q}: field "${Z}" missing`),{ok:!1,reason:"missing_field",field:Z};if(typeof z[Z]!=="string")return console.warn(`[checkpoint] invalid metadata at ${Q}: field "${Z}" not a string`),{ok:!1,reason:"invalid_type",field:Z}}if(!Object.prototype.hasOwnProperty.call(z,"iteration"))return console.warn(`[checkpoint] invalid metadata at ${Q}: field "iteration" missing`),{ok:!1,reason:"missing_field",field:"iteration"};if(typeof z.iteration!=="number"||!Number.isFinite(z.iteration))return console.warn(`[checkpoint] invalid metadata at ${Q}: field "iteration" not a finite number`),{ok:!1,reason:"invalid_type",field:"iteration"};for(let Z of L3){let K=z[Z];if(I3.test(K))return console.warn(`[checkpoint] invalid metadata at ${Q}: field "${Z}" contains control characters`),{ok:!1,reason:"control_chars",field:Z}}return{ok:!0,value:{id:z.id,timestamp:z.timestamp,iteration:z.iteration,task_id:z.task_id,task_description:z.task_description,git_sha:z.git_sha,git_branch:z.git_branch,provider:z.provider,phase:z.phase}}}function m1($,Q){if(!x3.test($))throw new w0($);let z=Q??P(),X=s(j1(z),$);if(!O1(X))throw new h1($);let Z=E0(z,$);if(!Z)throw new h1($);return Z}function S0($,Q){let z=m1($,Q),X=Q??P(),Z=s(j1(X),$),K=[];for(let W of R3){let q=s(Z,W);if(!O1(q))continue;K.push({from:q,to:s(X,W)})}return{id:$,metadata:z,restore:K}}function N0($){let Q=[],z=0;for(let X of $.restore)try{G3(A3(X.to),{recursive:!0});let Z=`${X.to}.tmp.${process.pid}.${++T3}`;V3(X.from,Z),M3(Z,X.to),z+=1}catch(Z){Q.push(`${X.from} -> ${X.to}: ${Z.message}`)}return{restored:z,errors:Q}}var v7,T3=0,I3,L3,x3,h1,w0,R3;var D0=w(()=>{v();p();T1();v7=Promise.resolve();I3=/[\x00-\x08\x0a-\x1f\x7f-\x9f]/,L3=["id","task_id","git_sha","git_branch","provider","phase"];x3=/^[a-zA-Z0-9_-]+$/;h1=class h1 extends Error{id;constructor($){super(`Checkpoint not found: ${$}`);this.id=$;this.name="CheckpointNotFoundError"}};w0=class w0 extends Error{id;constructor($){super(`Invalid checkpoint ID: must be alphanumeric, hyphens, underscores only (got: ${$})`);this.id=$;this.name="InvalidCheckpointIdError"}};R3=["state/orchestrator.json","queue/pending.json","queue/completed.json","queue/in-progress.json","queue/current-task.json"]});var b0={};y(b0,{runRollback:()=>F3});async function F3($){let Q=$[0],z=$.slice(1);if(Q===void 0||Q==="help"||Q==="--help"||Q==="-h")return process.stdout.write(C0),Q===void 0?1:0;switch(Q){case"list":{let X=[...v1()].reverse();if(X.length===0)return process.stdout.write(`${x}No checkpoints found.${U}
347
+ `),0;process.stdout.write(`${D}Checkpoints${U} (${X.length}, newest first):
347
348
  `);for(let Z of X)process.stdout.write(` ${A}${Z.id}${U} iter=${Z.iteration} ${Z.git_branch||"(no branch)"}@${(Z.git_sha||"").slice(0,7)} ${Z.timestamp}
348
349
  `);return 0}case"show":{let X=z[0];if(!X)return process.stderr.write(`${L}Missing checkpoint id.${U} Use \`loki rollback list\`.
349
350
  `),2;try{let Z=m1(X);return process.stdout.write(`${JSON.stringify(Z,null,2)}
350
351
  `),0}catch(Z){return process.stderr.write(`${L}Failed to read checkpoint:${U} ${Z.message}
351
352
  `),1}}case"to":{let X=z[0];if(!X)return process.stderr.write(`${L}Missing checkpoint id.${U} Use \`loki rollback list\`.
352
- `),2;return k0(X)}case"latest":{let X=g1(),Z=X[X.length-1];if(!Z)return process.stderr.write(`${L}No checkpoints found to roll back to.${U}
353
+ `),2;return k0(X)}case"latest":{let X=v1(),Z=X[X.length-1];if(!Z)return process.stderr.write(`${L}No checkpoints found to roll back to.${U}
353
354
  `),1;return process.stdout.write(`Rolling back to latest checkpoint: ${A}${Z.id}${U}
354
355
  `),k0(Z.id)}default:return process.stderr.write(`Unknown subcommand: ${Q}
355
356
  `),process.stderr.write(C0),2}}function k0($){let Q;try{Q=S0($)}catch(X){return process.stderr.write(`${L}Cannot plan rollback:${U} ${X.message}
356
- `),1}if(Q.restore.length===0)return process.stdout.write(`${R}Checkpoint ${$} has no restorable state files; nothing to do.${U}
357
+ `),1}if(Q.restore.length===0)return process.stdout.write(`${x}Checkpoint ${$} has no restorable state files; nothing to do.${U}
357
358
  `),0;let z=N0(Q);if(z.errors.length>0){for(let X of z.errors)process.stderr.write(`${L}restore error:${U} ${X}
358
359
  `);return process.stderr.write(`${L}Partial rollback: ${z.restored}/${Q.restore.length} files restored.${U}
359
- `),1}return process.stdout.write(`${k}Rolled back ${z.restored}/${Q.restore.length} state files from ${$}.${U}
360
+ `),1}return process.stdout.write(`${C}Rolled back ${z.restored}/${Q.restore.length} state files from ${$}.${U}
360
361
  `),process.stdout.write("Run `loki start` to resume from the restored state.\n"),0}var C0=`Usage: loki rollback <subcommand>
361
362
 
362
363
  Subcommands:
@@ -372,8 +373,8 @@ Restored files (matches autonomy/run.sh:7028 byte-for-byte):
372
373
  Note: only state files are restored. Source code, git history, and the
373
374
  session's autonomy-state.json are unchanged. Re-run \`loki start\` to
374
375
  resume from the restored state.
375
- `;var y0=S(()=>{D0();n()});var u1={};y(u1,{renderFindingsForPrompt:()=>D3,loadPreviousFindings:()=>f1,findLatestReviewDir:()=>f0,_parseReviewerOutputForTests:()=>C3});import{existsSync as v0,readFileSync as h0,readdirSync as g0,statSync as E3}from"fs";import{join as I1}from"path";function N3($){let Q=$.toLowerCase();if(Q==="critical")return"Critical";if(Q==="high")return"High";if(Q==="medium")return"Medium";return"Low"}function m0($,Q,z,X){let Z=[],K=$.split(/\r?\n/);for(let W of K){let q=W.trim();if(q.length===0)continue;let H=q.replace(/^[-*]\s*/,""),G=w3.exec(H);if(!G||!G[1]||!G[2])continue;let J=N3(G[1]),B=G[2].trim(),V=S3.exec(B),M=V&&V[1]?V[1]:null,O=V&&V[2]?Number.parseInt(V[2],10):null;Z.push({reviewId:z,iteration:X,reviewer:Q,severity:J,description:B,file:M,line:Number.isFinite(O)?O:null,raw:q})}return Z}function f0($,Q){let z=I1($,"quality","reviews");if(!v0(z))return null;let X;try{X=g0(z)}catch{return null}let Z=Q===void 0?X.filter((q)=>q.startsWith("review-")):X.filter((q)=>q.endsWith(`-${Q}`)&&q.startsWith("review-"));if(Z.length===0)return null;Z.sort();let K=Z[Z.length-1];if(!K)return null;let W=I1(z,K);try{if(!E3(W).isDirectory())return null}catch{return null}return W}function f1($,Q){let z=f0($,Q);if(z===null)return{reviewDir:null,reviewId:null,iteration:null,findings:[]};let X=null,Z=null,K=I1(z,"aggregate.json");if(v0(K))try{let G=h0(K,"utf-8"),J=JSON.parse(G);if(typeof J.review_id==="string")X=J.review_id;if(typeof J.iteration==="number")Z=J.iteration}catch{}let W;try{W=g0(z)}catch{return{reviewDir:z,reviewId:X,iteration:Z,findings:[]}}let q=new Set(["diff.txt","files.txt","anti-sycophancy.txt"]),H=[];for(let G of W){if(!G.endsWith(".txt"))continue;if(q.has(G))continue;if(G.endsWith("-prompt.txt"))continue;let J=G.replace(/\.txt$/,""),B;try{B=h0(I1(z,G),"utf-8")}catch{continue}H.push(...m0(B,J,X??"",Z??-1))}return{reviewDir:z,reviewId:X,iteration:Z,findings:H}}function D3($){if($.length===0)return"";let Q=["Critical","High","Medium","Low"],z=new Map;for(let Z of Q)z.set(Z,[]);for(let Z of $){let K=z.get(Z.severity);if(K)K.push(Z)}let X=[];X.push("PREVIOUS REVIEWER FINDINGS (must address each, or supply counter-evidence in .loki/state/counter-evidence-<iter>.json):");for(let Z of Q){let K=z.get(Z)??[];if(K.length===0)continue;X.push(` [${Z}] (${K.length}):`);for(let W of K){let q=W.file?` (${W.file}${W.line!==null?":"+W.line:""})`:"";X.push(` - ${W.description}${q} -- via ${W.reviewer}`)}}return X.join(`
376
- `)}function C3($,Q,z="review-test",X=0){return m0($,Q,z,X)}var w3,S3;var L1=S(()=>{w3=/\[(Critical|High|Medium|Low)\]\s*(.+)/i,S3=/([\w.\-/]+\.[a-zA-Z]+):(\d+)/});import{existsSync as k3}from"fs";import{join as b3}from"path";async function u0($,Q){let z=b3($,"memory");if(!k3(z))return{stored:!1,reason:"memory dir not initialized"};let X=Math.max(0,Math.floor(Q.durationSeconds??0)),Z={_LOKI_PROJECT_DIR:u,_LOKI_TARGET_DIR:process.cwd(),_LOKI_TASK_ID:Q.taskId,_LOKI_OUTCOME:Q.outcome,_LOKI_PHASE:Q.phase,_LOKI_GOAL:Q.goal,_LOKI_DURATION:String(X),_LOKI_LOKI_DIR:$},W=await a(`
376
+ `;var y0=w(()=>{D0();n()});var f1={};y(f1,{renderFindingsForPrompt:()=>D3,loadPreviousFindings:()=>g1,findLatestReviewDir:()=>f0,_parseReviewerOutputForTests:()=>C3});import{existsSync as v0,readFileSync as h0,readdirSync as m0,statSync as E3}from"fs";import{join as _1}from"path";function N3($){let Q=$.toLowerCase();if(Q==="critical")return"Critical";if(Q==="high")return"High";if(Q==="medium")return"Medium";return"Low"}function g0($,Q,z,X){let Z=[],K=$.split(/\r?\n/);for(let W of K){let q=W.trim();if(q.length===0)continue;let H=q.replace(/^[-*]\s*/,""),G=w3.exec(H);if(!G||!G[1]||!G[2])continue;let J=N3(G[1]),B=G[2].trim(),V=S3.exec(B),M=V&&V[1]?V[1]:null,j=V&&V[2]?Number.parseInt(V[2],10):null;Z.push({reviewId:z,iteration:X,reviewer:Q,severity:J,description:B,file:M,line:Number.isFinite(j)?j:null,raw:q})}return Z}function f0($,Q){let z=_1($,"quality","reviews");if(!v0(z))return null;let X;try{X=m0(z)}catch{return null}let Z=Q===void 0?X.filter((q)=>q.startsWith("review-")):X.filter((q)=>q.endsWith(`-${Q}`)&&q.startsWith("review-"));if(Z.length===0)return null;Z.sort();let K=Z[Z.length-1];if(!K)return null;let W=_1(z,K);try{if(!E3(W).isDirectory())return null}catch{return null}return W}function g1($,Q){let z=f0($,Q);if(z===null)return{reviewDir:null,reviewId:null,iteration:null,findings:[]};let X=null,Z=null,K=_1(z,"aggregate.json");if(v0(K))try{let G=h0(K,"utf-8"),J=JSON.parse(G);if(typeof J.review_id==="string")X=J.review_id;if(typeof J.iteration==="number")Z=J.iteration}catch{}let W;try{W=m0(z)}catch{return{reviewDir:z,reviewId:X,iteration:Z,findings:[]}}let q=new Set(["diff.txt","files.txt","anti-sycophancy.txt"]),H=[];for(let G of W){if(!G.endsWith(".txt"))continue;if(q.has(G))continue;if(G.endsWith("-prompt.txt"))continue;let J=G.replace(/\.txt$/,""),B;try{B=h0(_1(z,G),"utf-8")}catch{continue}H.push(...g0(B,J,X??"",Z??-1))}return{reviewDir:z,reviewId:X,iteration:Z,findings:H}}function D3($){if($.length===0)return"";let Q=["Critical","High","Medium","Low"],z=new Map;for(let Z of Q)z.set(Z,[]);for(let Z of $){let K=z.get(Z.severity);if(K)K.push(Z)}let X=[];X.push("PREVIOUS REVIEWER FINDINGS (must address each, or supply counter-evidence in .loki/state/counter-evidence-<iter>.json):");for(let Z of Q){let K=z.get(Z)??[];if(K.length===0)continue;X.push(` [${Z}] (${K.length}):`);for(let W of K){let q=W.file?` (${W.file}${W.line!==null?":"+W.line:""})`:"";X.push(` - ${W.description}${q} -- via ${W.reviewer}`)}}return X.join(`
377
+ `)}function C3($,Q,z="review-test",X=0){return g0($,Q,z,X)}var w3,S3;var I1=w(()=>{w3=/\[(Critical|High|Medium|Low)\]\s*(.+)/i,S3=/([\w.\-/]+\.[a-zA-Z]+):(\d+)/});import{existsSync as k3}from"fs";import{join as b3}from"path";async function u0($,Q){let z=b3($,"memory");if(!k3(z))return{stored:!1,reason:"memory dir not initialized"};let X=Math.max(0,Math.floor(Q.durationSeconds??0)),Z={_LOKI_PROJECT_DIR:u,_LOKI_TARGET_DIR:process.cwd(),_LOKI_TASK_ID:Q.taskId,_LOKI_OUTCOME:Q.outcome,_LOKI_PHASE:Q.phase,_LOKI_GOAL:Q.goal,_LOKI_DURATION:String(X),_LOKI_LOKI_DIR:$},W=await a(`
377
378
  import os, sys
378
379
  project = os.environ.get('_LOKI_PROJECT_DIR', '')
379
380
  loki = os.environ.get('_LOKI_LOKI_DIR', '.loki')
@@ -400,23 +401,23 @@ try:
400
401
  print('OK')
401
402
  except Exception as e:
402
403
  print('ERR:' + str(e))
403
- `,{env:Z,timeoutMs:15000});if(W.exitCode===127)return{stored:!1,reason:"python3 not found"};let q=W.stdout.trim();if(q==="OK")return{stored:!0,reason:"stored"};if(q.startsWith("ERR:"))return{stored:!1,reason:q.replace(/^ERR:/,"")};return{stored:!1,reason:W.stderr.trim()||"unknown"}}var p0=S(()=>{X1();g()});var o0={};y(o0,{loadLearnings:()=>p1,appendLearning:()=>U1,appendFromGateFailure:()=>p3});import{existsSync as y3,readFileSync as h3}from"fs";import{join as c0}from"path";import{createHash as v3}from"crypto";function l0($){return c0($,g3)}function m3($){if($===null||typeof $!=="object")return!1;let Q=$;return typeof Q.id==="string"&&typeof Q.timestamp==="string"&&typeof Q.iteration==="number"&&typeof Q.trigger==="string"&&typeof Q.rootCause==="string"&&typeof Q.fix==="string"&&typeof Q.preventInFuture==="string"&&typeof Q.evidence==="object"&&Q.evidence!==null}function d0($){if(!y3($))return{version:1,learnings:[]};try{let Q=h3($,"utf-8"),z=JSON.parse(Q);if(z.version===1&&Array.isArray(z.learnings))return{version:1,learnings:z.learnings.filter(m3)}}catch{}return{version:1,learnings:[]}}function f3($,Q){return v3("sha256").update(`${$}\x00${Q}`).digest("hex").slice(0,16)}async function U1($,Q,z={}){let X=f3(Q.trigger,Q.rootCause),Z=new Date().toISOString(),K={id:X,timestamp:Z,...Q},W=l0($);if(await R0(W,()=>{let H=d0(W),G=H.learnings.findIndex((J)=>J.id===X);if(G>=0){let J=H.learnings[G];H.learnings[G]={...J,timestamp:Z,iteration:K.iteration}}else H.learnings.push(K);W1(W,H)}),z.episodeBridge!==null&&(z.episodeBridge!==void 0||process.env.LOKI_AUTO_LEARNINGS_EPISODE==="1")){let H=z.episodeBridge??u0,G=z.bridgeFailureLog??u3;try{let J=await H($,{taskId:`learning-${X}`,outcome:"failure",phase:"VERIFY",goal:`${Q.trigger}: ${Q.rootCause}`});if(J&&!J.stored){if(!new Set(["memory dir not initialized","stub"]).has(J.reason))G(`episode_bridge skipped: ${J.reason}`)}}catch(J){G(`episode_bridge threw: ${J.message}`)}}return K}function u3($){process.stderr.write(`[learnings_writer] ${$}
404
- `)}async function p3($,Q,z,X={}){let Z=`[${z.severity}] ${z.description}`;return U1($,{iteration:Q,trigger:"gate_failure",rootCause:Z,fix:"pending: dev agent must address in next iteration or supply counter-evidence",preventInFuture:"if this finding recurs, lower its severity threshold or add a regression test",evidence:{reviewId:z.reviewId,file:z.file??void 0,line:z.line??void 0,severity:z.severity,reviewer:z.reviewer}},X)}function p1($){return d0(l0($))}var g3;var P1=S(()=>{O1();p0();g3=c0("state","relevant-learnings.json")});var a0={};y(a0,{runOverrideCouncil:()=>a3,recordOverrideOutcome:()=>r3,loadCounterEvidence:()=>n3,canonicalFindingId:()=>c1,DEFAULT_OVERRIDE_JUDGES:()=>n0});import{existsSync as c3,readFileSync as l3}from"fs";import{join as d3}from"path";function n3($,Q){let z=d3($,"state",`counter-evidence-${Q}.json`);if(!c3(z))return null;try{let X=l3(z,"utf-8"),Z=JSON.parse(X);if(typeof Z.iteration!=="number")return null;let K=Array.isArray(Z.evidence)?Z.evidence:[],W=[];for(let q of K){if(typeof q!=="object"||q===null)continue;let H=q;if(typeof H.findingId!=="string")continue;if(typeof H.claim!=="string")continue;let G=H.proofType;if(typeof G!=="string"||!o3.has(G))continue;let J=G,B=Array.isArray(H.artifacts)?H.artifacts:[];W.push({findingId:H.findingId,claim:H.claim,proofType:J,artifacts:B.filter((V)=>typeof V==="string")})}return{iteration:Z.iteration,evidence:W}}catch{return null}}async function a3($,Q,z,X={}){let Z=X.judges??n0,K=new Set,W=new Set,q={},H=new Map;for(let G of Q.evidence)H.set(G.findingId,G);for(let G of $){let J=c1(G),B=H.get(J);if(!B){W.add(J);continue}let V=await Promise.all(Z.map((O)=>z({finding:G,evidence:B,judge:O})));if(q[J]=V,V.filter((O)=>O.verdict==="APPROVE_OVERRIDE").length>=2)K.add(J);else W.add(J)}return{approvedFindingIds:K,rejectedFindingIds:W,votes:q}}function c1($){let Q=$.raw.slice(0,80).replace(/\s+/g," ").trim();return`${$.reviewer}::${Q}`}async function r3($,Q,z,X,Z={}){let K={episodeBridge:Z.episodeBridge===void 0?null:Z.episodeBridge};for(let W of X){let q=c1(W);if(z.approvedFindingIds.has(q))await U1($,{iteration:Q,trigger:"override_approved",rootCause:`[${W.severity}] ${W.description}`,fix:"override council approved counter-evidence; finding lifted",preventInFuture:"if this reviewer/file pair recurs, narrow the reviewer's selector OR add a baseline doc",evidence:{findingId:q,reviewId:W.reviewId,file:W.file??void 0,line:W.line??void 0,severity:W.severity,reviewer:W.reviewer}},K);else if(z.rejectedFindingIds.has(q))await U1($,{iteration:Q,trigger:"override_rejected",rootCause:`[${W.severity}] ${W.description}`,fix:"override council rejected -- dev agent must fix the finding",preventInFuture:"address this finding in the next iteration",evidence:{findingId:q,reviewId:W.reviewId,file:W.file??void 0,line:W.line??void 0,severity:W.severity,reviewer:W.reviewer}},K)}}var o3,n0;var r0=S(()=>{P1();o3=new Set(["file-exists","test-passes","grep-miss","reviewer-misread","duplicate-code-path","out-of-scope"]);n0=["judge-primary","judge-secondary","judge-tertiary"]});var i0={};y(i0,{writeEscalationHandoff:()=>U5,renderHandoff:()=>s0,readLatestHandoff:()=>q5});import{existsSync as s3,mkdirSync as t3,readdirSync as i3,readFileSync as e3,renameSync as $5,writeFileSync as z5}from"fs";import{dirname as Q5,join as x1}from"path";function X5(){return new Date().toISOString()}function Z5($){let Q=$.file?` (${$.file}${$.line!==null?":"+$.line:""})`:"";return` - [${$.severity}] ${$.description}${Q} -- ${$.reviewer}`}function K5($){let Q=$.evidence,z=Q.file?` ${Q.file}${Q.line!==void 0?":"+Q.line:""}`:"";return` - **${$.trigger}** (iter ${$.iteration})${z}: ${$.rootCause}`}function s0($,Q,z){let X=[];if(X.push(`# Loki escalation handoff -- ${X5()}`),X.push(""),X.push(`Gate **${$.gateName}** has failed ${$.consecutiveFailures} consecutive times at iteration ${$.iteration}.`),X.push(""),X.push(`Reason: ${$.detail}`),X.push(""),Q.length>0){X.push(`## Outstanding findings (${Q.length})`),X.push("");for(let Z of Q)X.push(Z5(Z));X.push("")}else X.push("## Outstanding findings"),X.push(""),X.push("(no per-finding records captured -- gate failed without populating reviewer outputs)"),X.push("");if(z.length>0){X.push(`## Recent learnings (${Math.min(z.length,10)})`),X.push("");for(let Z of z.slice(-10))X.push(K5(Z));X.push("")}return X.push("## What the human must decide"),X.push(""),X.push("- Approve override? Write `.loki/state/counter-evidence-<iter>.json` with one entry per finding to dispute, then `rm .loki/PAUSE` to resume."),X.push("- Disable a gate? Set `LOKI_GATE_<NAME>=false` in env (see skills/quality-gates.md)."),X.push("- Tweak escalation? Set `LOKI_GATE_PAUSE_LIMIT` or `LOKI_GATE_ESCALATE_LIMIT`."),X.push("- Roll back? Switch to `LOKI_LEGACY_BASH=1` and re-run; the bash route does not consult this handoff doc."),X.push(""),X.push("To resume: address the findings (or supply counter-evidence) and `rm .loki/PAUSE`."),X.join(`
405
- `)}function W5($,Q){t3(Q5($),{recursive:!0});let z=`${$}.tmp.${process.pid}.${++t0}`;z5(z,Q),$5(z,$)}function U5($,Q,z={}){let X=z.findings??f1($,Q.iteration).findings,Z=z.learnings??p1($).learnings,K=s0(Q,X,Z),W=(z.now?.()??new Date).toISOString().replace(/[-:.]/g,""),q=x1($,"escalations"),H=++t0,G=x1(q,`handoff-${W}-${process.pid}-${H}-${Q.gateName}.md`);return W5(G,K),{path:G,bytes:K.length}}function q5($){let Q=x1($,"escalations");if(!s3(Q))return null;let z;try{z=i3(Q).filter((K)=>K.endsWith(".md"))}catch{return null}if(z.length===0)return null;z.sort();let X=z[z.length-1];if(!X)return null;let Z=x1(Q,X);try{return{path:Z,body:e3(Z,"utf-8")}}catch{return null}}var t0=0;var e0=S(()=>{L1();P1()});var $6={};y($6,{runInternalPhase1Hooks:()=>Y5,_resolveForTests:()=>B5,_internalPhase1HooksHelp:()=>O5});import{existsSync as H5,mkdirSync as G5,readdirSync as V5,statSync as J5}from"fs";import{join as q1,resolve as B5}from"path";async function Y5($){let[Q,...z]=$;switch(Q){case void 0:case"help":case"--help":case"-h":return process.stdout.write(l1),Q===void 0?1:0;case"reflect":return M5(z);case"override":return A5(z);case"handoff":return T5(z);default:return process.stderr.write(`Unknown subcommand: ${Q}
406
- `),process.stderr.write(l1),2}}async function M5($){let Q=d1($[0]);if(Q===null)return process.stderr.write(`reflect: missing or invalid <iter>
407
- `),2;let z=x();try{let Z=(await Promise.resolve().then(() => (L1(),u1))).loadPreviousFindings(z,Q);if(Z.findings.length===0)return process.stdout.write(`reflect: no findings for iter ${Q} (nothing to do)
408
- `),0;let K=q1(z,"state");G5(K,{recursive:!0}),W1(q1(K,`findings-${Q}.json`),{review_id:Z.reviewId,iteration:Q,findings:Z.findings});let W=await Promise.resolve().then(() => (P1(),o0)),q=0;if(process.env.LOKI_AUTO_LEARNINGS!=="0"){for(let H of Z.findings)if(H.severity==="Critical"||H.severity==="High")await W.appendFromGateFailure(z,Q,H,{episodeBridge:null}),q+=1}return process.stdout.write(`reflect: persisted ${Z.findings.length} findings + ${q} learnings (iter ${Q})
404
+ `,{env:Z,timeoutMs:15000});if(W.exitCode===127)return{stored:!1,reason:"python3 not found"};let q=W.stdout.trim();if(q==="OK")return{stored:!0,reason:"stored"};if(q.startsWith("ERR:"))return{stored:!1,reason:q.replace(/^ERR:/,"")};return{stored:!1,reason:W.stderr.trim()||"unknown"}}var p0=w(()=>{X1();v()});var o0={};y(o0,{loadLearnings:()=>u1,appendLearning:()=>U1,appendFromGateFailure:()=>p3});import{existsSync as y3,readFileSync as h3}from"fs";import{join as c0}from"path";import{createHash as v3}from"crypto";function l0($){return c0($,m3)}function g3($){if($===null||typeof $!=="object")return!1;let Q=$;return typeof Q.id==="string"&&typeof Q.timestamp==="string"&&typeof Q.iteration==="number"&&typeof Q.trigger==="string"&&typeof Q.rootCause==="string"&&typeof Q.fix==="string"&&typeof Q.preventInFuture==="string"&&typeof Q.evidence==="object"&&Q.evidence!==null}function d0($){if(!y3($))return{version:1,learnings:[]};try{let Q=h3($,"utf-8"),z=JSON.parse(Q);if(z.version===1&&Array.isArray(z.learnings))return{version:1,learnings:z.learnings.filter(g3)}}catch{}return{version:1,learnings:[]}}function f3($,Q){return v3("sha256").update(`${$}\x00${Q}`).digest("hex").slice(0,16)}async function U1($,Q,z={}){let X=f3(Q.trigger,Q.rootCause),Z=new Date().toISOString(),K={id:X,timestamp:Z,...Q},W=l0($);if(await R0(W,()=>{let H=d0(W),G=H.learnings.findIndex((J)=>J.id===X);if(G>=0){let J=H.learnings[G];H.learnings[G]={...J,timestamp:Z,iteration:K.iteration}}else H.learnings.push(K);W1(W,H)}),z.episodeBridge!==null&&(z.episodeBridge!==void 0||process.env.LOKI_AUTO_LEARNINGS_EPISODE==="1")){let H=z.episodeBridge??u0,G=z.bridgeFailureLog??u3;try{let J=await H($,{taskId:`learning-${X}`,outcome:"failure",phase:"VERIFY",goal:`${Q.trigger}: ${Q.rootCause}`});if(J&&!J.stored){if(!new Set(["memory dir not initialized","stub"]).has(J.reason))G(`episode_bridge skipped: ${J.reason}`)}}catch(J){G(`episode_bridge threw: ${J.message}`)}}return K}function u3($){process.stderr.write(`[learnings_writer] ${$}
405
+ `)}async function p3($,Q,z,X={}){let Z=`[${z.severity}] ${z.description}`;return U1($,{iteration:Q,trigger:"gate_failure",rootCause:Z,fix:"pending: dev agent must address in next iteration or supply counter-evidence",preventInFuture:"if this finding recurs, lower its severity threshold or add a regression test",evidence:{reviewId:z.reviewId,file:z.file??void 0,line:z.line??void 0,severity:z.severity,reviewer:z.reviewer}},X)}function u1($){return d0(l0($))}var m3;var L1=w(()=>{T1();p0();m3=c0("state","relevant-learnings.json")});var a0={};y(a0,{runOverrideCouncil:()=>a3,recordOverrideOutcome:()=>s3,loadCounterEvidence:()=>n3,canonicalFindingId:()=>p1,DEFAULT_OVERRIDE_JUDGES:()=>n0});import{existsSync as c3,readFileSync as l3}from"fs";import{join as d3}from"path";function n3($,Q){let z=d3($,"state",`counter-evidence-${Q}.json`);if(!c3(z))return null;try{let X=l3(z,"utf-8"),Z=JSON.parse(X);if(typeof Z.iteration!=="number")return null;let K=Array.isArray(Z.evidence)?Z.evidence:[],W=[];for(let q of K){if(typeof q!=="object"||q===null)continue;let H=q;if(typeof H.findingId!=="string")continue;if(typeof H.claim!=="string")continue;let G=H.proofType;if(typeof G!=="string"||!o3.has(G))continue;let J=G,B=Array.isArray(H.artifacts)?H.artifacts:[];W.push({findingId:H.findingId,claim:H.claim,proofType:J,artifacts:B.filter((V)=>typeof V==="string")})}return{iteration:Z.iteration,evidence:W}}catch{return null}}async function a3($,Q,z,X={}){let Z=X.judges??n0,K=new Set,W=new Set,q={},H=new Map;for(let G of Q.evidence)H.set(G.findingId,G);for(let G of $){let J=p1(G),B=H.get(J);if(!B){W.add(J);continue}let V=await Promise.all(Z.map((j)=>z({finding:G,evidence:B,judge:j})));if(q[J]=V,V.filter((j)=>j.verdict==="APPROVE_OVERRIDE").length>=2)K.add(J);else W.add(J)}return{approvedFindingIds:K,rejectedFindingIds:W,votes:q}}function p1($){let Q=$.raw.slice(0,80).replace(/\s+/g," ").trim();return`${$.reviewer}::${Q}`}async function s3($,Q,z,X,Z={}){let K={episodeBridge:Z.episodeBridge===void 0?null:Z.episodeBridge};for(let W of X){let q=p1(W);if(z.approvedFindingIds.has(q))await U1($,{iteration:Q,trigger:"override_approved",rootCause:`[${W.severity}] ${W.description}`,fix:"override council approved counter-evidence; finding lifted",preventInFuture:"if this reviewer/file pair recurs, narrow the reviewer's selector OR add a baseline doc",evidence:{findingId:q,reviewId:W.reviewId,file:W.file??void 0,line:W.line??void 0,severity:W.severity,reviewer:W.reviewer}},K);else if(z.rejectedFindingIds.has(q))await U1($,{iteration:Q,trigger:"override_rejected",rootCause:`[${W.severity}] ${W.description}`,fix:"override council rejected -- dev agent must fix the finding",preventInFuture:"address this finding in the next iteration",evidence:{findingId:q,reviewId:W.reviewId,file:W.file??void 0,line:W.line??void 0,severity:W.severity,reviewer:W.reviewer}},K)}}var o3,n0;var s0=w(()=>{L1();o3=new Set(["file-exists","test-passes","grep-miss","reviewer-misread","duplicate-code-path","out-of-scope"]);n0=["judge-primary","judge-secondary","judge-tertiary"]});var i0={};y(i0,{writeEscalationHandoff:()=>U5,renderHandoff:()=>r0,readLatestHandoff:()=>q5});import{existsSync as r3,mkdirSync as t3,readdirSync as i3,readFileSync as e3,renameSync as $5,writeFileSync as z5}from"fs";import{dirname as Q5,join as P1}from"path";function X5(){return new Date().toISOString()}function Z5($){let Q=$.file?` (${$.file}${$.line!==null?":"+$.line:""})`:"";return` - [${$.severity}] ${$.description}${Q} -- ${$.reviewer}`}function K5($){let Q=$.evidence,z=Q.file?` ${Q.file}${Q.line!==void 0?":"+Q.line:""}`:"";return` - **${$.trigger}** (iter ${$.iteration})${z}: ${$.rootCause}`}function r0($,Q,z){let X=[];if(X.push(`# Loki escalation handoff -- ${X5()}`),X.push(""),X.push(`Gate **${$.gateName}** has failed ${$.consecutiveFailures} consecutive times at iteration ${$.iteration}.`),X.push(""),X.push(`Reason: ${$.detail}`),X.push(""),Q.length>0){X.push(`## Outstanding findings (${Q.length})`),X.push("");for(let Z of Q)X.push(Z5(Z));X.push("")}else X.push("## Outstanding findings"),X.push(""),X.push("(no per-finding records captured -- gate failed without populating reviewer outputs)"),X.push("");if(z.length>0){X.push(`## Recent learnings (${Math.min(z.length,10)})`),X.push("");for(let Z of z.slice(-10))X.push(K5(Z));X.push("")}return X.push("## What the human must decide"),X.push(""),X.push("- Approve override? Write `.loki/state/counter-evidence-<iter>.json` with one entry per finding to dispute, then `rm .loki/PAUSE` to resume."),X.push("- Disable a gate? Set `LOKI_GATE_<NAME>=false` in env (see skills/quality-gates.md)."),X.push("- Tweak escalation? Set `LOKI_GATE_PAUSE_LIMIT` or `LOKI_GATE_ESCALATE_LIMIT`."),X.push("- Roll back? Switch to `LOKI_LEGACY_BASH=1` and re-run; the bash route does not consult this handoff doc."),X.push(""),X.push("To resume: address the findings (or supply counter-evidence) and `rm .loki/PAUSE`."),X.join(`
406
+ `)}function W5($,Q){t3(Q5($),{recursive:!0});let z=`${$}.tmp.${process.pid}.${++t0}`;z5(z,Q),$5(z,$)}function U5($,Q,z={}){let X=z.findings??g1($,Q.iteration).findings,Z=z.learnings??u1($).learnings,K=r0(Q,X,Z),W=(z.now?.()??new Date).toISOString().replace(/[-:.]/g,""),q=P1($,"escalations"),H=++t0,G=P1(q,`handoff-${W}-${process.pid}-${H}-${Q.gateName}.md`);return W5(G,K),{path:G,bytes:K.length}}function q5($){let Q=P1($,"escalations");if(!r3(Q))return null;let z;try{z=i3(Q).filter((K)=>K.endsWith(".md"))}catch{return null}if(z.length===0)return null;z.sort();let X=z[z.length-1];if(!X)return null;let Z=P1(Q,X);try{return{path:Z,body:e3(Z,"utf-8")}}catch{return null}}var t0=0;var e0=w(()=>{I1();L1()});var $6={};y($6,{runInternalPhase1Hooks:()=>Y5,_resolveForTests:()=>B5,_internalPhase1HooksHelp:()=>O5});import{existsSync as H5,mkdirSync as G5,readdirSync as V5,statSync as J5}from"fs";import{join as q1,resolve as B5}from"path";async function Y5($){let[Q,...z]=$;switch(Q){case void 0:case"help":case"--help":case"-h":return process.stdout.write(c1),Q===void 0?1:0;case"reflect":return M5(z);case"override":return A5(z);case"handoff":return T5(z);default:return process.stderr.write(`Unknown subcommand: ${Q}
407
+ `),process.stderr.write(c1),2}}async function M5($){let Q=l1($[0]);if(Q===null)return process.stderr.write(`reflect: missing or invalid <iter>
408
+ `),2;let z=P();try{let Z=(await Promise.resolve().then(() => (I1(),f1))).loadPreviousFindings(z,Q);if(Z.findings.length===0)return process.stdout.write(`reflect: no findings for iter ${Q} (nothing to do)
409
+ `),0;let K=q1(z,"state");G5(K,{recursive:!0}),W1(q1(K,`findings-${Q}.json`),{review_id:Z.reviewId,iteration:Q,findings:Z.findings});let W=await Promise.resolve().then(() => (L1(),o0)),q=0;if(process.env.LOKI_AUTO_LEARNINGS!=="0"){for(let H of Z.findings)if(H.severity==="Critical"||H.severity==="High")await W.appendFromGateFailure(z,Q,H,{episodeBridge:null}),q+=1}return process.stdout.write(`reflect: persisted ${Z.findings.length} findings + ${q} learnings (iter ${Q})
409
410
  `),0}catch(X){return process.stderr.write(`reflect: ${X.message}
410
- `),1}}async function A5($){let Q=d1($[0]);if(Q===null)return process.stderr.write(`override: missing or invalid <iter>
411
- `),2;let z=x();try{let X=await Promise.resolve().then(() => (r0(),a0)),Z=X.loadCounterEvidence(z,Q);if(Z===null||Z.evidence.length===0)return process.stdout.write(`override: no counter-evidence for iter ${Q} (skip)
412
- `),0;let W=(await Promise.resolve().then(() => (L1(),u1))).loadPreviousFindings(z,Q),q=W.findings.filter((j)=>j.severity==="Critical"||j.severity==="High");if(q.length===0)return process.stdout.write(`override: no blocking findings for iter ${Q} (skip)
413
- `),0;let H=new Set(["duplicate-code-path","file-exists","test-passes","grep-miss","out-of-scope"]),G=async(j)=>{let w=H.has(j.evidence.proofType);return{judge:j.judge,verdict:w?"APPROVE_OVERRIDE":"REJECT_OVERRIDE",reasoning:w?`[stub] proofType=${j.evidence.proofType} trusted`:`[stub] proofType=${j.evidence.proofType} requires manual review`}},J=await X.runOverrideCouncil(q,Z,G);await X.recordOverrideOutcome(z,Q,J,q);let B=q1(z,"quality","reviews");if(H5(B))try{let j=V5(B).filter((m)=>m.startsWith("review-")).sort(),w=j[j.length-1];if(w&&J5(q1(B,w)).isDirectory())W1(q1(B,w,`override-${Q}.json`),{review_id:W.reviewId,iteration:Q,approved_finding_ids:Array.from(J.approvedFindingIds),rejected_finding_ids:Array.from(J.rejectedFindingIds),votes:J.votes})}catch{}let V=J.approvedFindingIds.size,M=J.rejectedFindingIds.size;if(M===0&&V>0)process.stdout.write(`override: LIFTED -- ${V} approved, ${M} rejected
411
+ `),1}}async function A5($){let Q=l1($[0]);if(Q===null)return process.stderr.write(`override: missing or invalid <iter>
412
+ `),2;let z=P();try{let X=await Promise.resolve().then(() => (s0(),a0)),Z=X.loadCounterEvidence(z,Q);if(Z===null||Z.evidence.length===0)return process.stdout.write(`override: no counter-evidence for iter ${Q} (skip)
413
+ `),0;let W=(await Promise.resolve().then(() => (I1(),f1))).loadPreviousFindings(z,Q),q=W.findings.filter((_)=>_.severity==="Critical"||_.severity==="High");if(q.length===0)return process.stdout.write(`override: no blocking findings for iter ${Q} (skip)
414
+ `),0;let H=new Set(["duplicate-code-path","file-exists","test-passes","grep-miss","out-of-scope"]),G=async(_)=>{let k=H.has(_.evidence.proofType);return{judge:_.judge,verdict:k?"APPROVE_OVERRIDE":"REJECT_OVERRIDE",reasoning:k?`[stub] proofType=${_.evidence.proofType} trusted`:`[stub] proofType=${_.evidence.proofType} requires manual review`}},J=await X.runOverrideCouncil(q,Z,G);await X.recordOverrideOutcome(z,Q,J,q);let B=q1(z,"quality","reviews");if(H5(B))try{let _=V5(B).filter((m)=>m.startsWith("review-")).sort(),k=_[_.length-1];if(k&&J5(q1(B,k)).isDirectory())W1(q1(B,k,`override-${Q}.json`),{review_id:W.reviewId,iteration:Q,approved_finding_ids:Array.from(J.approvedFindingIds),rejected_finding_ids:Array.from(J.rejectedFindingIds),votes:J.votes})}catch{}let V=J.approvedFindingIds.size,M=J.rejectedFindingIds.size;if(M===0&&V>0)process.stdout.write(`override: LIFTED -- ${V} approved, ${M} rejected
414
415
  `);else process.stdout.write(`override: BLOCKED -- ${V} approved, ${M} rejected
415
416
  `);return 0}catch(X){return process.stderr.write(`override: ${X.message}
416
- `),1}}async function T5($){let Q=$[0],z=Number.parseInt($[1]??"0",10),X=d1($[2]);if(!Q||!Number.isFinite(z)||X===null)return process.stderr.write(`handoff: usage: handoff <gate> <consecutive-failures> <iter>
417
- `),2;let Z=x();try{let W=(await Promise.resolve().then(() => (e0(),i0))).writeEscalationHandoff(Z,{gateName:Q,iteration:X,consecutiveFailures:z,detail:`${Q} hit PAUSE_LIMIT (${z} consecutive failures)`});return process.stdout.write(`handoff: wrote ${W.path} (${W.bytes}B)
417
+ `),1}}async function T5($){let Q=$[0],z=Number.parseInt($[1]??"0",10),X=l1($[2]);if(!Q||!Number.isFinite(z)||X===null)return process.stderr.write(`handoff: usage: handoff <gate> <consecutive-failures> <iter>
418
+ `),2;let Z=P();try{let W=(await Promise.resolve().then(() => (e0(),i0))).writeEscalationHandoff(Z,{gateName:Q,iteration:X,consecutiveFailures:z,detail:`${Q} hit PAUSE_LIMIT (${z} consecutive failures)`});return process.stdout.write(`handoff: wrote ${W.path} (${W.bytes}B)
418
419
  `),0}catch(K){return process.stderr.write(`handoff: ${K.message}
419
- `),1}}function d1($){if($===void 0)return null;let Q=Number.parseInt($,10);return Number.isFinite(Q)&&Q>=0?Q:null}var l1=`loki internal phase1-hooks <subcommand>
420
+ `),1}}function l1($){if($===void 0)return null;let Q=Number.parseInt($,10);return Number.isFinite(Q)&&Q>=0?Q:null}var c1=`loki internal phase1-hooks <subcommand>
420
421
 
421
422
  Subcommands:
422
423
  reflect <iter> Persist structured findings + auto-learnings.
@@ -425,24 +426,24 @@ Subcommands:
425
426
 
426
427
  This command is invoked by autonomy/run.sh between iterations. Users
427
428
  should not run it directly -- run \`loki start\` instead.
428
- `,O5;var z6=S(()=>{g();O1();O5=l1});g();import{readFileSync as H6}from"fs";import{resolve as G6,dirname as V6}from"path";import{fileURLToPath as J6}from"url";var o=null;function r1(){if(o!==null)return o;let $="7.5.17";if(typeof $==="string"&&$.length>0)return o=$,o;try{let Q=V6(J6(import.meta.url)),z=E1(Q);o=H6(G6(z,"VERSION"),"utf-8").trim()}catch{o="unknown"}return o}function s1(){return process.stdout.write(`Loki Mode v${r1()}
429
- `),0}p();n();g();import{readFileSync as T6,existsSync as O6}from"fs";import{resolve as j6}from"path";var _6=["claude","codex","gemini","cline","aider"];function i1(){let $=j6(x(),"state","provider");if(!O6($))return"";try{return T6($,"utf-8").trim()}catch{return""}}function I6($,Q){return $||Q||process.env.LOKI_PROVIDER||"claude"}function L6($){let Q=i1(),z=I6($,Q);switch(process.stdout.write(`${C}Current Provider${U}
429
+ `,O5;var z6=w(()=>{v();T1();O5=c1});v();import{readFileSync as H6}from"fs";import{resolve as G6,dirname as V6}from"path";import{fileURLToPath as J6}from"url";var o=null;function s1(){if(o!==null)return o;let $="7.5.27";if(typeof $==="string"&&$.length>0)return o=$,o;try{let Q=V6(J6(import.meta.url)),z=F1(Q);o=H6(G6(z,"VERSION"),"utf-8").trim()}catch{o="unknown"}return o}function r1(){return process.stdout.write(`Loki Mode v${s1()}
430
+ `),0}p();n();v();import{readFileSync as T6,existsSync as O6}from"fs";import{resolve as j6}from"path";var _6=["claude","codex","cline","aider"];function i1(){let $=j6(P(),"state","provider");if(!O6($))return"";try{return T6($,"utf-8").trim()}catch{return""}}function I6($,Q){return $||Q||process.env.LOKI_PROVIDER||"claude"}function L6($){let Q=i1(),z=I6($,Q);switch(process.stdout.write(`${D}Current Provider${U}
430
431
  `),process.stdout.write(`
431
432
  `),process.stdout.write(`${A}Provider:${U} ${z}
432
- `),z){case"claude":process.stdout.write(`${k}Status:${U} Full features (subagents, parallel, MCP)
433
- `);break;case"cline":process.stdout.write(`${k}Status:${U} Near-full mode (subagents, MCP, 12+ providers)
434
- `);break;case"codex":case"gemini":case"aider":process.stdout.write(`${R}Status:${U} Degraded mode (sequential only)
435
- `);break;default:break}if(Q)process.stdout.write(`${P}(saved in .loki/state/provider)${U}
436
- `);else process.stdout.write(`${P}(default - not explicitly set)${U}
433
+ `),z){case"claude":process.stdout.write(`${C}Status:${U} Full features (subagents, parallel, MCP)
434
+ `);break;case"cline":process.stdout.write(`${C}Status:${U} Near-full mode (subagents, MCP, 12+ providers)
435
+ `);break;case"codex":case"aider":process.stdout.write(`${x}Status:${U} Degraded mode (sequential only)
436
+ `);break;default:break}if(Q)process.stdout.write(`${R}(saved in .loki/state/provider)${U}
437
+ `);else process.stdout.write(`${R}(default - not explicitly set)${U}
437
438
  `);return process.stdout.write(`
438
439
  `),process.stdout.write(`Switch provider: ${A}loki provider set <name>${U}
439
440
  `),process.stdout.write(`Available: ${A}loki provider list${U}
440
- `),0}async function P6(){let Q=i1()||process.env.LOKI_PROVIDER||"claude";process.stdout.write(`${C}Available Providers${U}
441
+ `),0}async function P6(){let Q=i1()||process.env.LOKI_PROVIDER||"claude";process.stdout.write(`${D}Available Providers${U}
441
442
  `),process.stdout.write(`
442
- `);let z=await Promise.all(_6.map(async(K)=>[K,await b(K)!==null])),X=new Map;for(let[K,W]of z)X.set(K,W?`${k}installed${U}`:`${L}not installed${U}`);let Z=[["claude","claude - Claude Code (Anthropic) "],["codex","codex - Codex CLI (OpenAI) "],["gemini","gemini - Gemini CLI (Google) "],["cline","cline - Cline (multi-provider) "],["aider","aider - Aider (terminal pair prog) "]];for(let[K,W]of Z){let q=Q===K?` ${A}(current)${U}`:"";process.stdout.write(` ${W} ${X.get(K)}${q}
443
+ `);let z=await Promise.all(_6.map(async(K)=>[K,await b(K)!==null])),X=new Map;for(let[K,W]of z)X.set(K,W?`${C}installed${U}`:`${L}not installed${U}`);let Z=[["claude","claude - Claude Code (Anthropic) "],["codex","codex - Codex CLI (OpenAI) "],["cline","cline - Cline (multi-provider) "],["aider","aider - Aider (terminal pair prog) "]];for(let[K,W]of Z){let q=Q===K?` ${A}(current)${U}`:"";process.stdout.write(` ${W} ${X.get(K)}${q}
443
444
  `)}return process.stdout.write(`
444
445
  `),process.stdout.write(`Set provider: ${A}loki provider set <name>${U}
445
- `),0}function x6(){return process.stdout.write(`${C}Loki Mode Provider Management${U}
446
+ `),0}function x6(){return process.stdout.write(`${D}Loki Mode Provider Management${U}
446
447
  `),process.stdout.write(`
447
448
  `),process.stdout.write(`Usage: loki provider <command>
448
449
  `),process.stdout.write(`
@@ -458,13 +459,13 @@ should not run it directly -- run \`loki start\` instead.
458
459
  `),process.stdout.write(` loki provider set claude
459
460
  `),process.stdout.write(` loki provider set codex
460
461
  `),process.stdout.write(` loki provider list
461
- `),process.stdout.write(` loki provider info gemini
462
+ `),process.stdout.write(` loki provider info codex
462
463
  `),process.stdout.write(` loki provider models
463
- `),0}async function e1($){let Q=$[0]??"show",z=$.slice(1);switch(Q){case"show":case"current":return L6(z[0]);case"list":return P6();case"set":case"info":case"models":return R6(["provider",Q,...z]);default:return x6()}}async function R6($){let{run:Q}=await Promise.resolve().then(() => (p(),t1)),{resolve:z}=await import("path"),{REPO_ROOT:X}=await Promise.resolve().then(() => (g(),a1)),Z=z(X,"autonomy","loki"),K=await Q([Z,...$],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(K.stdout),process.stderr.write(K.stderr),K.exitCode}n();g();X1();p();import{existsSync as $0,readFileSync as E6}from"fs";import{resolve as t}from"path";import{mkdir as w6}from"fs/promises";var Z1=t(w1(),"learnings");function N1($){if(!$0($))return 0;try{let Q=E6($,"utf-8"),z=0;for(let X of Q.split(`
464
- `))if(X.includes('"description"'))z++;return z}catch{return 0}}async function S6(){await w6(Z1,{recursive:!0});let $=N1(t(Z1,"patterns.jsonl")),Q=N1(t(Z1,"mistakes.jsonl")),z=N1(t(Z1,"successes.jsonl"));return process.stdout.write(`${C}Cross-Project Learnings${U}
464
+ `),0}async function e1($){let Q=$[0]??"show",z=$.slice(1);switch(Q){case"show":case"current":return L6(z[0]);case"list":return P6();case"set":case"info":case"models":return R6(["provider",Q,...z]);default:return x6()}}async function R6($){let{run:Q}=await Promise.resolve().then(() => (p(),t1)),{resolve:z}=await import("path"),{REPO_ROOT:X}=await Promise.resolve().then(() => (v(),a1)),Z=z(X,"autonomy","loki"),K=await Q([Z,...$],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(K.stdout),process.stderr.write(K.stderr),K.exitCode}n();v();X1();p();import{existsSync as $0,readFileSync as E6}from"fs";import{resolve as t}from"path";import{mkdir as w6}from"fs/promises";var Z1=t(E1(),"learnings");function S1($){if(!$0($))return 0;try{let Q=E6($,"utf-8"),z=0;for(let X of Q.split(`
465
+ `))if(X.includes('"description"'))z++;return z}catch{return 0}}async function S6(){await w6(Z1,{recursive:!0});let $=S1(t(Z1,"patterns.jsonl")),Q=S1(t(Z1,"mistakes.jsonl")),z=S1(t(Z1,"successes.jsonl"));return process.stdout.write(`${D}Cross-Project Learnings${U}
465
466
  `),process.stdout.write(`
466
- `),process.stdout.write(` Patterns: ${k}${$}${U}
467
- `),process.stdout.write(` Mistakes: ${R}${Q}${U}
467
+ `),process.stdout.write(` Patterns: ${C}${$}${U}
468
+ `),process.stdout.write(` Mistakes: ${x}${Q}${U}
468
469
  `),process.stdout.write(` Successes: ${A}${z}${U}
469
470
  `),process.stdout.write(`
470
471
  `),process.stdout.write(`Location: ${Z1}
@@ -480,7 +481,7 @@ except ImportError:
480
481
  print('Error: memory.layers module not found')
481
482
  except Exception as e:
482
483
  print(f'Error: {e}')
483
- `.trim(),Z=await a(X,{cwd:u});return process.stdout.write(Z.stdout),0}let Q=t(x(),"memory","index.json");if(!$0(Q))return process.stdout.write(`No index found
484
+ `.trim(),Z=await a(X,{cwd:u});return process.stdout.write(Z.stdout),0}let Q=t(P(),"memory","index.json");if(!$0(Q))return process.stdout.write(`No index found
484
485
  `),0;let z=await a(`import json, sys; sys.stdout.write(json.dumps(json.load(open(${JSON.stringify(Q)})), indent=4) + "\\n")`);if(z.exitCode!==0)return process.stdout.write(`No index found
485
486
  `),0;return process.stdout.write(z.stdout),0}async function z0($){switch($[0]??"list"){case"list":case"ls":return S6();case"index":return N6($[1]==="rebuild");default:{let z=t(u,"autonomy","loki"),X=await E([z,"memory",...$],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(X.stdout),process.stderr.write(X.stderr),X.exitCode}}}var Q6=`Loki Mode (TypeScript port, Phase 2 of bash->Bun migration)
486
487
 
@@ -501,11 +502,11 @@ Phase 2 ported (Bun-native, fast):
501
502
  All other commands fall through to the bash CLI (autonomy/loki).
502
503
  Set LOKI_LEGACY_BASH=1 to force the bash CLI for every command.
503
504
  `;function j5(){let $=process.env.LOKI_LEGACY_BASH;if($===void 0)return;let Q=$.trim().toLowerCase();if(Q!=="1"&&Q!=="true"&&Q!=="yes"&&Q!=="on")return;if(process.env.LOKI_SUPPRESS_BUN_DIRECT_WARN==="1")return;process.stderr.write(`warning: LOKI_LEGACY_BASH is set, but you are running the Bun runtime directly (src/cli.ts). The env var only takes effect via the bin/loki shim, which dispatches between Bun and bash. Behavior is unchanged; this message is informational.
504
- `)}async function _5($){j5();let Q=$[0],z=$.slice(1);switch(Q){case void 0:case"help":case"--help":case"-h":return process.stdout.write(Q6),0;case"version":case"--version":case"-v":return s1();case"provider":return e1(z);case"memory":return z0(z);case"status":{let{runStatus:X}=await Promise.resolve().then(() => (H0(),q0));return X(z)}case"stats":{let{runStats:X}=await Promise.resolve().then(() => (Y0(),B0));return X(z)}case"doctor":{let{runDoctor:X}=await Promise.resolve().then(() => (x0(),P0));return X(z)}case"rollback":{let{runRollback:X}=await Promise.resolve().then(() => (y0(),b0));return X(z)}case"internal":{let X=z[0];if(!X||X==="--help"||X==="-h"||X==="help"){let K=["loki internal -- runtime hooks driven by autonomy/run.sh","","Subcommands:"," phase1-hooks Persist structured findings, run override council,"," append learnings, and write the escalation handoff"," doc once per iteration. Driven by run.sh; not"," intended for direct invocation.","","Phase 1 (RARV-C closure) env vars:"," LOKI_INJECT_FINDINGS=1 Persist structured reviewer findings to"," .loki/state/findings-<iter>.json so the"," next iteration can address them."," LOKI_OVERRIDE_COUNCIL=1 Allow a 3-LLM override panel to lift a"," BLOCK when counter-evidence is presented."," See LOKI_OVERRIDE_JUDGES (csv),"," LOKI_OVERRIDE_PANEL_SIZE,"," LOKI_OVERRIDE_REAL_JUDGE."," LOKI_AUTO_LEARNINGS=1 Append failure rootcauses to"," .loki/state/relevant-learnings.json via"," the episodic memory bridge."," LOKI_HANDOFF_MD=1 Write a structured human handoff doc to"," .loki/escalations/<ts>.md before PAUSE.","","All four are default-on as of v7.5.3. Set to 0 to disable.","Reference: CHANGELOG.md (search 'Phase 1') and skills/healing.md.","","These commands are wired into the autonomous loop and may change","without notice. Do not script against them.",""].join(`
505
+ `)}async function _5($){j5();let Q=$[0],z=$.slice(1);switch(Q){case void 0:case"help":case"--help":case"-h":return process.stdout.write(Q6),0;case"version":case"--version":case"-v":return r1();case"provider":return e1(z);case"memory":return z0(z);case"status":{let{runStatus:X}=await Promise.resolve().then(() => (H0(),q0));return X(z)}case"stats":{let{runStats:X}=await Promise.resolve().then(() => (Y0(),B0));return X(z)}case"doctor":{let{runDoctor:X}=await Promise.resolve().then(() => (x0(),P0));return X(z)}case"rollback":{let{runRollback:X}=await Promise.resolve().then(() => (y0(),b0));return X(z)}case"internal":{let X=z[0];if(!X||X==="--help"||X==="-h"||X==="help"){let K=["loki internal -- runtime hooks driven by autonomy/run.sh","","Subcommands:"," phase1-hooks Persist structured findings, run override council,"," append learnings, and write the escalation handoff"," doc once per iteration. Driven by run.sh; not"," intended for direct invocation.","","Phase 1 (RARV-C closure) env vars:"," LOKI_INJECT_FINDINGS=1 Persist structured reviewer findings to"," .loki/state/findings-<iter>.json so the"," next iteration can address them."," LOKI_OVERRIDE_COUNCIL=1 Allow a 3-LLM override panel to lift a"," BLOCK when counter-evidence is presented."," See LOKI_OVERRIDE_JUDGES (csv),"," LOKI_OVERRIDE_PANEL_SIZE,"," LOKI_OVERRIDE_REAL_JUDGE."," LOKI_AUTO_LEARNINGS=1 Append failure rootcauses to"," .loki/state/relevant-learnings.json via"," the episodic memory bridge."," LOKI_HANDOFF_MD=1 Write a structured human handoff doc to"," .loki/escalations/<ts>.md before PAUSE.","","All four are default-on as of v7.5.3. Set to 0 to disable.","Reference: CHANGELOG.md (search 'Phase 1') and skills/healing.md.","","These commands are wired into the autonomous loop and may change","without notice. Do not script against them.",""].join(`
505
506
  `);return process.stdout.write(`${K}
506
507
  `),0}if(X==="phase1-hooks"){let{runInternalPhase1Hooks:K}=await Promise.resolve().then(() => (z6(),$6));return K(z.slice(1))}return process.stderr.write(`Unknown internal subcommand: ${X}
507
508
  `),process.stderr.write(`Run 'loki internal --help' for the supported list.
508
509
  `),2}default:return process.stderr.write(`Unknown command: ${Q}
509
510
  `),process.stderr.write(Q6),2}}process.on("SIGINT",()=>process.exit(130));process.on("SIGTERM",()=>process.exit(143));var I5=await _5(Bun.argv.slice(2));process.exit(I5);
510
511
 
511
- //# debugId=9C22E648FE5F7C2364756E2164756E21
512
+ //# debugId=B969F5C49B8FCB5864756E2164756E21