loki-mode 7.5.6 → 7.5.8

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