loki-mode 7.7.15 → 7.7.17

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.
package/SKILL.md CHANGED
@@ -3,7 +3,7 @@ name: loki-mode
3
3
  description: Multi-agent autonomous startup system. Triggers on "Loki Mode". Takes a spec (PRD, GitHub issue, OpenAPI doc, etc.) to deployed product with minimal human intervention. Requires --dangerously-skip-permissions flag.
4
4
  ---
5
5
 
6
- # Loki Mode v7.7.15
6
+ # Loki Mode v7.7.17
7
7
 
8
8
  **You are an autonomous agent. You make decisions. You do not ask questions. You do not stop.**
9
9
 
@@ -381,4 +381,4 @@ See `CHANGELOG.md` entries [7.5.7], [7.5.8], [7.5.13] for the per-fix list and r
381
381
 
382
382
  ---
383
383
 
384
- **v7.7.15 | [Autonomi](https://www.autonomi.dev/) flagship product | ~260 lines core**
384
+ **v7.7.17 | [Autonomi](https://www.autonomi.dev/) flagship product | ~260 lines core**
package/VERSION CHANGED
@@ -1 +1 @@
1
- 7.7.15
1
+ 7.7.17
package/autonomy/loki CHANGED
@@ -7255,6 +7255,43 @@ sentrux = {
7255
7255
  'required': 'optional'
7256
7256
  }
7257
7257
 
7258
+ # v7.7.17: memory subsystem health surface. Reports the latest entries
7259
+ # from .loki/memory/.errors.log (rotated by memory/error_log.py) so
7260
+ # developers see regressions in the previously-silent-fail call sites
7261
+ # at run.sh:8724, 9087, 9136. Sibling of checks/disk/sentrux; not
7262
+ # counted in the summary tally to keep backwards-compatible numbers.
7263
+ # Inline file read (rather than importing memory.error_log) so this
7264
+ # block works regardless of the doctor's invocation cwd / sys.path.
7265
+ # v7.7.17 council fix (Opus 2): tail-only read (last 64 KB) so a
7266
+ # pathological oversize log cannot OOM the doctor command. Bash + Bun
7267
+ # emit the SAME relative path string for parity.
7268
+ memory_base = os.path.join(os.environ.get('LOKI_DIR', '.loki'), 'memory')
7269
+ _memory_log_path = os.path.join(memory_base, '.errors.log')
7270
+ _TAIL_BYTES = 64 * 1024
7271
+ memory_recent_errors = []
7272
+ if os.path.isfile(_memory_log_path):
7273
+ try:
7274
+ with open(_memory_log_path, 'rb') as _f:
7275
+ _f.seek(0, 2)
7276
+ _size = _f.tell()
7277
+ _offset = max(0, _size - _TAIL_BYTES)
7278
+ _f.seek(_offset)
7279
+ _chunk = _f.read()
7280
+ _text = _chunk.decode('utf-8', errors='replace')
7281
+ _lines = _text.split('\n')
7282
+ if _offset > 0 and _lines:
7283
+ _lines = _lines[1:] # drop possibly-partial first line
7284
+ _lines = [ln.strip() for ln in _lines if ln.strip()]
7285
+ memory_recent_errors = _lines[-5:]
7286
+ except OSError:
7287
+ memory_recent_errors = []
7288
+ memory = {
7289
+ 'errors_log_path': _memory_log_path if os.path.isfile(_memory_log_path) else None,
7290
+ 'recent_errors': memory_recent_errors,
7291
+ 'recent_error_count': len(memory_recent_errors),
7292
+ 'status': 'pass' if not memory_recent_errors else 'warn',
7293
+ }
7294
+
7258
7295
  pass_count = sum(1 for c in checks if c['status'] == 'pass')
7259
7296
  fail_count = sum(1 for c in checks if c['status'] == 'fail')
7260
7297
  warn_count = sum(1 for c in checks if c['status'] == 'warn')
@@ -7271,6 +7308,7 @@ result = {
7271
7308
  'status': disk_status
7272
7309
  },
7273
7310
  'sentrux': sentrux,
7311
+ 'memory': memory,
7274
7312
  'summary': {
7275
7313
  'passed': pass_count,
7276
7314
  'failed': fail_count,
package/autonomy/run.sh CHANGED
@@ -8722,7 +8722,13 @@ try:
8722
8722
  )
8723
8723
  engine.store_episode(trace)
8724
8724
  except Exception as e:
8725
- pass # Silently fail
8725
+ # v7.7.17: replace silent-fail with structured log to .errors.log.
8726
+ # log_memory_error itself never raises (it has its own try/except).
8727
+ try:
8728
+ from memory.error_log import log_memory_error
8729
+ log_memory_error(f'{target_dir}/.loki/memory', 'store_episode_trace', e)
8730
+ except Exception:
8731
+ pass
8726
8732
  PYEOF
8727
8733
  }
8728
8734
 
@@ -9072,8 +9078,15 @@ try:
9072
9078
  json.dump({'path': str(episode_file), 'importance': importance}, f)
9073
9079
  except OSError:
9074
9080
  pass
9075
- except Exception:
9076
- pass # Silently fail -- memory capture must never break the loop
9081
+ except Exception as e:
9082
+ # v7.7.17: replace silent-fail with structured log to .errors.log.
9083
+ # log_memory_error never raises; outer try/except guards even
9084
+ # against import failure of the logger itself.
9085
+ try:
9086
+ from memory.error_log import log_memory_error
9087
+ log_memory_error(f'{target_dir}/.loki/memory', 'auto_capture_episode', e)
9088
+ except Exception:
9089
+ pass
9077
9090
  PYEOF
9078
9091
 
9079
9092
  # v6.83.0 Phase 1: RARV-C REFLECT/VERIFY shadow-write. Only when both
@@ -9128,7 +9141,12 @@ try:
9128
9141
  if result.patterns_created > 0:
9129
9142
  print(f'Memory consolidation: {result.patterns_created} patterns created')
9130
9143
  except Exception as e:
9131
- pass # Silently fail
9144
+ # v7.7.17: replace silent-fail with structured log to .errors.log.
9145
+ try:
9146
+ from memory.error_log import log_memory_error
9147
+ log_memory_error(f'{target_dir}/.loki/memory', 'run_memory_consolidation', e)
9148
+ except Exception:
9149
+ pass
9132
9150
  PYEOF
9133
9151
  }
9134
9152
 
@@ -7,7 +7,7 @@ Modules:
7
7
  control: Session control API (start/stop/pause/resume)
8
8
  """
9
9
 
10
- __version__ = "7.7.15"
10
+ __version__ = "7.7.17"
11
11
 
12
12
  # Expose the control app for easy import
13
13
  try:
@@ -2,7 +2,7 @@
2
2
 
3
3
  The flagship product of [Autonomi](https://www.autonomi.dev/). Complete installation instructions for all platforms and use cases.
4
4
 
5
- **Version:** v7.7.15
5
+ **Version:** v7.7.17
6
6
 
7
7
  ---
8
8
 
@@ -1,58 +1,58 @@
1
1
  // @bun
2
- var _7=Object.defineProperty;var I7=(K)=>K;function P7(K,$){this[K]=I7.bind(null,$)}var b=(K,$)=>{for(var z in $)_7(K,z,{get:$[z],enumerable:!0,configurable:!0,set:P7.bind($,z)})};var R=(K,$)=>()=>(K&&($=K(K=0)),$);var V1=import.meta.require;var e1={};b(e1,{lokiDir:()=>P,homeLokiDir:()=>k1,findRepoRootForVersion:()=>N1,REPO_ROOT:()=>p});import{resolve as u,dirname as S1}from"path";import{fileURLToPath as L7}from"url";import{existsSync as J1}from"fs";import{homedir as R7}from"os";function E7(){let K=i1;for(let $=0;$<6;$++){if(J1(u(K,"VERSION"))&&J1(u(K,"autonomy/run.sh")))return K;let z=S1(K);if(z===K)break;K=z}return u(i1,"..","..","..")}function N1(K){let $=K;for(let z=0;z<6;z++){if(J1(u($,"VERSION"))&&J1(u($,"autonomy/run.sh")))return $;let Q=S1($);if(Q===$)break;$=Q}return u(K,"..","..","..")}function P(){return process.env.LOKI_DIR??u(process.cwd(),".loki")}function k1(){return u(R7(),".loki")}var i1,p;var y=R(()=>{i1=S1(L7(import.meta.url));p=E7()});import{readFileSync as x7}from"fs";import{resolve as F7,dirname as w7}from"path";import{fileURLToPath as S7}from"url";function G1(){if(n!==null)return n;let K="7.7.15";if(typeof K==="string"&&K.length>0)return n=K,n;try{let $=w7(S7(import.meta.url)),z=N1($);n=x7(F7(z,"VERSION"),"utf-8").trim()}catch{n="unknown"}return n}var n=null;var D1=R(()=>{y()});var $0={};b($0,{runOrThrow:()=>N7,run:()=>S,commandVersion:()=>D7,commandExists:()=>D,ShellError:()=>C1});async function S(K,$={}){let z=Bun.spawn({cmd:[...K],stdout:"pipe",stderr:"pipe",env:$.env?{...process.env,...$.env}:process.env,cwd:$.cwd}),Q,X;if($.timeoutMs&&$.timeoutMs>0)Q=setTimeout(()=>{try{z.kill("SIGTERM")}catch{}X=setTimeout(()=>{try{z.kill("SIGKILL")}catch{}},2000)},$.timeoutMs);try{let[H,Z,q]=await Promise.all([new Response(z.stdout).text(),new Response(z.stderr).text(),z.exited]);return{stdout:H,stderr:Z,exitCode:q}}finally{if(Q)clearTimeout(Q);if(X)clearTimeout(X)}}async function N7(K,$={}){let z=await S(K,$);if(z.exitCode!==0)throw new C1(`command failed (${z.exitCode}): ${K.join(" ")}`,z.exitCode,z.stdout,z.stderr);return z}async function D(K){let $=k7(K),z=await S(["sh","-c",`command -v ${$}`],{timeoutMs:5000});if(z.exitCode===0)return z.stdout.trim()||null;return null}function k7(K){if(!/^[A-Za-z0-9._/-]+$/.test(K))throw Error(`refused to shell-escape suspect token: ${K}`);return K}async function D7(K,$="--version"){if(!await D(K))return null;let Q=await S([K,$],{timeoutMs:5000});if(Q.exitCode!==0)return null;return((Q.stdout||Q.stderr).split(/\r?\n/)[0]?.trim()??"")||null}var C1;var c=R(()=>{C1=class C1 extends Error{message;exitCode;stdout;stderr;constructor(K,$,z,Q){super(K);this.message=K;this.exitCode=$;this.stdout=z;this.stderr=Q;this.name="ShellError"}}});function l(K){return C7?"":K}var C7,E,C,x,O6,O,k,F,W;var a=R(()=>{C7=(process.env.NO_COLOR??"").length>0;E=l("\x1B[0;31m"),C=l("\x1B[0;32m"),x=l("\x1B[1;33m"),O6=l("\x1B[0;34m"),O=l("\x1B[0;36m"),k=l("\x1B[1m"),F=l("\x1B[2m"),W=l("\x1B[0m")});import{existsSync as c7}from"fs";async function t(){if(z1!==void 0)return z1;let K="/opt/homebrew/bin/python3.12";if(c7(K))return z1=K,K;let $=await D("python3.12");if($)return z1=$,$;let z=await D("python3");return z1=z,z}async function s(K,$={}){let z=await t();if(!z)return{stdout:"",stderr:"python3 not found",exitCode:127};return S([z,"-c",K],$)}var z1;var Q1=R(()=>{c()});var G0={};b(G0,{runStatus:()=>z5});import{existsSync as N,readFileSync as Z1,readdirSync as H0,statSync as W0}from"fs";import{resolve as w,basename as a7}from"path";async function r7(){if(await D("jq"))return!0;return process.stdout.write(`${E}Error: jq is required but not installed.${W}
2
+ var _7=Object.defineProperty;var I7=(K)=>K;function P7(K,$){this[K]=I7.bind(null,$)}var v=(K,$)=>{for(var Q in $)_7(K,Q,{get:$[Q],enumerable:!0,configurable:!0,set:P7.bind($,Q)})};var R=(K,$)=>()=>(K&&($=K(K=0)),$);var t=import.meta.require;var e1={};v(e1,{lokiDir:()=>P,homeLokiDir:()=>k1,findRepoRootForVersion:()=>N1,REPO_ROOT:()=>p});import{resolve as u,dirname as S1}from"path";import{fileURLToPath as L7}from"url";import{existsSync as J1}from"fs";import{homedir as R7}from"os";function E7(){let K=i1;for(let $=0;$<6;$++){if(J1(u(K,"VERSION"))&&J1(u(K,"autonomy/run.sh")))return K;let Q=S1(K);if(Q===K)break;K=Q}return u(i1,"..","..","..")}function N1(K){let $=K;for(let Q=0;Q<6;Q++){if(J1(u($,"VERSION"))&&J1(u($,"autonomy/run.sh")))return $;let X=S1($);if(X===$)break;$=X}return u(K,"..","..","..")}function P(){return process.env.LOKI_DIR??u(process.cwd(),".loki")}function k1(){return u(R7(),".loki")}var i1,p;var g=R(()=>{i1=S1(L7(import.meta.url));p=E7()});import{readFileSync as F7}from"fs";import{resolve as w7,dirname as x7}from"path";import{fileURLToPath as S7}from"url";function G1(){if(n!==null)return n;let K="7.7.17";if(typeof K==="string"&&K.length>0)return n=K,n;try{let $=x7(S7(import.meta.url)),Q=N1($);n=F7(w7(Q,"VERSION"),"utf-8").trim()}catch{n="unknown"}return n}var n=null;var D1=R(()=>{g()});var $0={};v($0,{runOrThrow:()=>N7,run:()=>S,commandVersion:()=>D7,commandExists:()=>h,ShellError:()=>C1});async function S(K,$={}){let Q=Bun.spawn({cmd:[...K],stdout:"pipe",stderr:"pipe",env:$.env?{...process.env,...$.env}:process.env,cwd:$.cwd}),X,Z;if($.timeoutMs&&$.timeoutMs>0)X=setTimeout(()=>{try{Q.kill("SIGTERM")}catch{}Z=setTimeout(()=>{try{Q.kill("SIGKILL")}catch{}},2000)},$.timeoutMs);try{let[W,z,q]=await Promise.all([new Response(Q.stdout).text(),new Response(Q.stderr).text(),Q.exited]);return{stdout:W,stderr:z,exitCode:q}}finally{if(X)clearTimeout(X);if(Z)clearTimeout(Z)}}async function N7(K,$={}){let Q=await S(K,$);if(Q.exitCode!==0)throw new C1(`command failed (${Q.exitCode}): ${K.join(" ")}`,Q.exitCode,Q.stdout,Q.stderr);return Q}async function h(K){let $=k7(K),Q=await S(["sh","-c",`command -v ${$}`],{timeoutMs:5000});if(Q.exitCode===0)return Q.stdout.trim()||null;return null}function k7(K){if(!/^[A-Za-z0-9._/-]+$/.test(K))throw Error(`refused to shell-escape suspect token: ${K}`);return K}async function D7(K,$="--version"){if(!await h(K))return null;let X=await S([K,$],{timeoutMs:5000});if(X.exitCode!==0)return null;return((X.stdout||X.stderr).split(/\r?\n/)[0]?.trim()??"")||null}var C1;var c=R(()=>{C1=class C1 extends Error{message;exitCode;stdout;stderr;constructor(K,$,Q,X){super(K);this.message=K;this.exitCode=$;this.stdout=Q;this.stderr=X;this.name="ShellError"}}});function l(K){return C7?"":K}var C7,E,b,F,T6,O,D,w,H;var a=R(()=>{C7=(process.env.NO_COLOR??"").length>0;E=l("\x1B[0;31m"),b=l("\x1B[0;32m"),F=l("\x1B[1;33m"),T6=l("\x1B[0;34m"),O=l("\x1B[0;36m"),D=l("\x1B[1m"),w=l("\x1B[2m"),H=l("\x1B[0m")});import{existsSync as c7}from"fs";async function i(){if(X1!==void 0)return X1;let K="/opt/homebrew/bin/python3.12";if(c7(K))return X1=K,K;let $=await h("python3.12");if($)return X1=$,$;let Q=await h("python3");return X1=Q,Q}async function s(K,$={}){let Q=await i();if(!Q)return{stdout:"",stderr:"python3 not found",exitCode:127};return S([Q,"-c",K],$)}var X1;var Z1=R(()=>{c()});var G0={};v(G0,{runStatus:()=>Q5});import{existsSync as k,readFileSync as W1,readdirSync as W0,statSync as H0}from"fs";import{resolve as x,basename as a7}from"path";async function r7(){if(await h("jq"))return!0;return process.stdout.write(`${E}Error: jq is required but not installed.${H}
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 B1(K){if(!Number.isFinite(K)||K<=0)return!1;try{return process.kill(K,0),!0}catch{return!1}}function M1(K){if(!N(K))return null;try{let $=Z1(K,"utf-8").trim();if(!$)return null;let z=Number.parseInt($,10);return Number.isFinite(z)?z:null}catch{return null}}function t7(K){let $=[],z=M1(w(K,"loki.pid"));if(z!==null&&B1(z))$.push(`global:${z}`);let Q=w(K,"sessions");if(N(Q)){let X=[];try{X=H0(Q)}catch{X=[]}for(let H of X){let Z=w(Q,H);try{if(!W0(Z).isDirectory())continue}catch{continue}let q=w(Z,"loki.pid"),U=M1(q);if(U!==null&&B1(U))$.push(`${H}:${U}`)}}if(N(K)){let X=[];try{X=H0(K)}catch{X=[]}for(let H of X){if(!H.startsWith("run-")||!H.endsWith(".pid"))continue;let Z=w(K,H);try{if(!W0(Z).isFile())continue}catch{continue}let q=a7(H,".pid").slice(4),U=M1(Z);if(U!==null&&B1(U)){if(!$.some((G)=>G.startsWith(`${q}:`)))$.push(`${q}:${U}`)}}}return $}async function q0(K,$){let z=await S(["jq","-r",K,$]);if(z.exitCode!==0)return null;return z.stdout.trim()}function U0(K,$){try{let z=Z1(K,"utf-8"),X=JSON.parse(z)[$];if(typeof X==="number"){if($==="budget_used"){let H=Math.round(X*100)/100;if(Number.isInteger(H))return String(H);return String(H)}return String(X)}if(X===void 0||X===null)return"0";return String(X)}catch{return"0"}}function V0(K,$,z){try{let Q=Z1(K,"utf-8"),H=JSON.parse(Q)[$];if(typeof H==="number"&&Number.isFinite(H))return H;return z}catch{return z}}async function i7(){let K=P();if(!await r7())return 1;if(!N(K))return process.stdout.write(`${k}Loki Mode Status${W}
7
+ `),!1}function B1(K){if(!Number.isFinite(K)||K<=0)return!1;try{return process.kill(K,0),!0}catch{return!1}}function M1(K){if(!k(K))return null;try{let $=W1(K,"utf-8").trim();if(!$)return null;let Q=Number.parseInt($,10);return Number.isFinite(Q)?Q:null}catch{return null}}function t7(K){let $=[],Q=M1(x(K,"loki.pid"));if(Q!==null&&B1(Q))$.push(`global:${Q}`);let X=x(K,"sessions");if(k(X)){let Z=[];try{Z=W0(X)}catch{Z=[]}for(let W of Z){let z=x(X,W);try{if(!H0(z).isDirectory())continue}catch{continue}let q=x(z,"loki.pid"),U=M1(q);if(U!==null&&B1(U))$.push(`${W}:${U}`)}}if(k(K)){let Z=[];try{Z=W0(K)}catch{Z=[]}for(let W of Z){if(!W.startsWith("run-")||!W.endsWith(".pid"))continue;let z=x(K,W);try{if(!H0(z).isFile())continue}catch{continue}let q=a7(W,".pid").slice(4),U=M1(z);if(U!==null&&B1(U)){if(!$.some((G)=>G.startsWith(`${q}:`)))$.push(`${q}:${U}`)}}}return $}async function q0(K,$){let Q=await S(["jq","-r",K,$]);if(Q.exitCode!==0)return null;return Q.stdout.trim()}function U0(K,$){try{let Q=W1(K,"utf-8"),Z=JSON.parse(Q)[$];if(typeof Z==="number"){if($==="budget_used"){let W=Math.round(Z*100)/100;if(Number.isInteger(W))return String(W);return String(W)}return String(Z)}if(Z===void 0||Z===null)return"0";return String(Z)}catch{return"0"}}function V0(K,$,Q){try{let X=W1(K,"utf-8"),W=JSON.parse(X)[$];if(typeof W==="number"&&Number.isFinite(W))return W;return Q}catch{return Q}}async function i7(){let K=P();if(!await r7())return 1;if(!k(K))return process.stdout.write(`${D}Loki Mode Status${H}
8
8
  `),process.stdout.write(`
9
- `),process.stdout.write(`${x}No active session found.${W}
9
+ `),process.stdout.write(`${F}No active session found.${H}
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(`${F}Current directory: ${process.cwd()}${W}
17
- `),0;process.stdout.write(`${k}Loki Mode Status${W}
16
+ `),process.stdout.write(`${w}Current directory: ${process.cwd()}${H}
17
+ `),0;process.stdout.write(`${D}Loki Mode Status${H}
18
18
  `),process.stdout.write(`
19
- `);let $="",z=w(K,"state","provider");if(N(z))try{$=Z1(z,"utf-8").trim()}catch{$=""}let Q=$||process.env.LOKI_PROVIDER||"claude",X="full features";switch(Q){case"codex":case"aider":X="degraded mode";break;case"cline":X="near-full mode";break;default:X="full features";break}process.stdout.write(`${O}Provider:${W} ${Q} (${X})
20
- `),process.stdout.write(`${F} Switch with: loki provider set <claude|codex|cline|aider>${W}
19
+ `);let $="",Q=x(K,"state","provider");if(k(Q))try{$=W1(Q,"utf-8").trim()}catch{$=""}let X=$||process.env.LOKI_PROVIDER||"claude",Z="full features";switch(X){case"codex":case"aider":Z="degraded mode";break;case"cline":Z="near-full mode";break;default:Z="full features";break}process.stdout.write(`${O}Provider:${H} ${X} (${Z})
20
+ `),process.stdout.write(`${w} Switch with: loki provider set <claude|codex|cline|aider>${H}
21
21
  `),process.stdout.write(`
22
- `);let H=t7(K);if(H.length>0){process.stdout.write(`${C}Active Sessions: ${H.length}${W}
23
- `);for(let J of H){let Y=J.indexOf(":"),j=Y>=0?J.slice(0,Y):J,_=Y>=0?J.slice(Y+1):"";if(j==="global")process.stdout.write(` ${O}[global]${W} PID ${_}
24
- `);else process.stdout.write(` ${O}[#${j}]${W} PID ${_}
22
+ `);let W=t7(K);if(W.length>0){process.stdout.write(`${b}Active Sessions: ${W.length}${H}
23
+ `);for(let J of W){let Y=J.indexOf(":"),T=Y>=0?J.slice(0,Y):J,j=Y>=0?J.slice(Y+1):"";if(T==="global")process.stdout.write(` ${O}[global]${H} PID ${j}
24
+ `);else process.stdout.write(` ${O}[#${T}]${H} PID ${j}
25
25
  `)}process.stdout.write(`
26
- `),process.stdout.write(`${F} Stop specific: loki stop <session-id>${W}
27
- `),process.stdout.write(`${F} Stop all: loki stop${W}
26
+ `),process.stdout.write(`${w} Stop specific: loki stop <session-id>${H}
27
+ `),process.stdout.write(`${w} Stop all: loki stop${H}
28
28
  `),process.stdout.write(`
29
- `)}if(N(w(K,"PAUSE")))process.stdout.write(`${x}Status: PAUSED${W}
30
- `),process.stdout.write(`${F} Resume with: loki resume${W}
29
+ `)}if(k(x(K,"PAUSE")))process.stdout.write(`${F}Status: PAUSED${H}
30
+ `),process.stdout.write(`${w} Resume with: loki resume${H}
31
31
  `),process.stdout.write(`
32
- `);else if(N(w(K,"STOP")))process.stdout.write(`${E}Status: STOPPED${W}
33
- `),process.stdout.write(`${F} Clear with: loki resume${W}
32
+ `);else if(k(x(K,"STOP")))process.stdout.write(`${E}Status: STOPPED${H}
33
+ `),process.stdout.write(`${w} Clear with: loki resume${H}
34
34
  `),process.stdout.write(`
35
- `);let Z=w(K,"STATUS.txt");if(N(Z)){process.stdout.write(`${O}Session Info:${W}
36
- `);try{process.stdout.write(Z1(Z,"utf-8"))}catch{}process.stdout.write(`
37
- `)}let q=w(K,"state","orchestrator.json");if(N(q)){process.stdout.write(`${O}Orchestrator State:${W}
35
+ `);let z=x(K,"STATUS.txt");if(k(z)){process.stdout.write(`${O}Session Info:${H}
36
+ `);try{process.stdout.write(W1(z,"utf-8"))}catch{}process.stdout.write(`
37
+ `)}let q=x(K,"state","orchestrator.json");if(k(q)){process.stdout.write(`${O}Orchestrator State:${H}
38
38
  `);let J=await q0('.currentPhase // "unknown"',q);process.stdout.write(`${J??"unknown"}
39
- `)}let U=w(K,"queue","pending.json");if(N(U)){let J=await q0('if type == "array" then length elif .tasks then .tasks | length else 0 end',U);process.stdout.write(`${O}Pending Tasks:${W} ${J??"0"}
40
- `)}let V=w(K,"metrics","budget.json");if(N(V)){let J=U0(V,"budget_limit"),Y=U0(V,"budget_used");if(J!=="0")process.stdout.write(`${O}Budget:${W} $${Y} / $${J}
41
- `);else process.stdout.write(`${O}Cost:${W} $${Y} (no limit)
42
- `)}let G=w(K,"state","context-usage.json");if(N(G)){let J=V0(G,"window_size",200000),Y=V0(G,"used_tokens",0),j=0;if(J>0)j=Math.floor(Y*100/J);process.stdout.write(`${O}Context:${W} ${j}% (${Y} / ${J} tokens)
43
- `)}let B=w(K,"dashboard","dashboard.pid");if(N(B)){let J=M1(B);if(J!==null&&B1(J)){let Y=process.env.LOKI_DASHBOARD_PORT||"57374";process.stdout.write(`${O}Dashboard:${W} http://127.0.0.1:${Y}/
39
+ `)}let U=x(K,"queue","pending.json");if(k(U)){let J=await q0('if type == "array" then length elif .tasks then .tasks | length else 0 end',U);process.stdout.write(`${O}Pending Tasks:${H} ${J??"0"}
40
+ `)}let V=x(K,"metrics","budget.json");if(k(V)){let J=U0(V,"budget_limit"),Y=U0(V,"budget_used");if(J!=="0")process.stdout.write(`${O}Budget:${H} $${Y} / $${J}
41
+ `);else process.stdout.write(`${O}Cost:${H} $${Y} (no limit)
42
+ `)}let G=x(K,"state","context-usage.json");if(k(G)){let J=V0(G,"window_size",200000),Y=V0(G,"used_tokens",0),T=0;if(J>0)T=Math.floor(Y*100/J);process.stdout.write(`${O}Context:${H} ${T}% (${Y} / ${J} tokens)
43
+ `)}let B=x(K,"dashboard","dashboard.pid");if(k(B)){let J=M1(B);if(J!==null&&B1(J)){let Y=process.env.LOKI_DASHBOARD_PORT||"57374";process.stdout.write(`${O}Dashboard:${H} http://127.0.0.1:${Y}/
44
44
  `)}}return await e7(K),process.stdout.write(`
45
- `),process.stdout.write(`${F} Tip: loki context show - detailed token breakdown${W}
46
- `),process.stdout.write(`${F} Tip: loki code overview - codebase intelligence${W}
47
- `),0}async function e7(K){let $=w(K,"state"),z=K5($),Q=w($,"relevant-learnings.json"),X=w(K,"escalations"),H=z.length>0,Z=N(Q),q=N(X);if(!H&&!Z&&!q)return;if(process.stdout.write(`
48
- ${O}Phase 1 artifacts:${W}
49
- `),H){let U=z[z.length-1],V=J0(U);if(V&&Array.isArray(V.findings)){let G={Critical:0,High:0,Medium:0,Low:0};for(let J of V.findings){let Y=String(J.severity??"");if(Y in G)G[Y]=(G[Y]??0)+1}let B=Object.entries(G).filter(([,J])=>J>0).map(([J,Y])=>`${Y} ${J.toLowerCase()}`).join(", ");process.stdout.write(` Findings (iter ${V.iteration??"?"}): ${B||"none"} -- ${V.findings.length} total
50
- `)}}if(Z){let U=J0(Q);if(U&&Array.isArray(U.learnings)&&U.learnings.length>0){let V=new Map;for(let B of U.learnings){let J=String(B.trigger??"unknown");V.set(J,(V.get(J)??0)+1)}let G=[...V.entries()].sort((B,J)=>J[1]-B[1]).slice(0,3).map(([B,J])=>`${J} ${B}`).join(", ");process.stdout.write(` Learnings: ${U.learnings.length} total (${G})
51
- `)}}if(q){let U=0,V="";try{let B=(await import("fs")).readdirSync(X).filter((J)=>J.endsWith(".md"));if(U=B.length,B.length>0)B.sort(),V=B[B.length-1]??""}catch{}if(U>0)process.stdout.write(` Escalations: ${U} handoff doc${U===1?"":"s"} (latest: ${V})
52
- `)}}function K5(K){if(!N(K))return[];try{return V1("fs").readdirSync(K).filter((Q)=>/^findings-\d+\.json$/.test(Q)).sort((Q,X)=>{let H=Number.parseInt(Q.replace(/[^0-9]/g,""),10)||0,Z=Number.parseInt(X.replace(/[^0-9]/g,""),10)||0;return H-Z}).map((Q)=>w(K,Q))}catch{return[]}}function J0(K){try{let $=V1("fs");return JSON.parse($.readFileSync(K,"utf-8"))}catch{return null}}async function $5(){let K=await t();if(!K)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
53
- `),1;let $=p,z=P(),Q=process.env.LOKI_DASHBOARD_PORT||"57374",X=process.env.LOKI_PROVIDER||"claude",H=await S([K,"-c",s7,$,z,Q,X],{timeoutMs:30000});if(H.exitCode!==0)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
54
- `),1;return process.stdout.write(H.stdout),0}async function z5(K){let $=[...K];while($.length>0){let z=$[0];if(z==="--json")return $5();if(z==="--help"||z==="-h")return process.stdout.write(`Usage: loki status [--json]
55
- `),0;return process.stdout.write(`${E}Unknown flag: ${z}${W}
45
+ `),process.stdout.write(`${w} Tip: loki context show - detailed token breakdown${H}
46
+ `),process.stdout.write(`${w} Tip: loki code overview - codebase intelligence${H}
47
+ `),0}async function e7(K){let $=x(K,"state"),Q=K5($),X=x($,"relevant-learnings.json"),Z=x(K,"escalations"),W=Q.length>0,z=k(X),q=k(Z);if(!W&&!z&&!q)return;if(process.stdout.write(`
48
+ ${O}Phase 1 artifacts:${H}
49
+ `),W){let U=Q[Q.length-1],V=J0(U);if(V&&Array.isArray(V.findings)){let G={Critical:0,High:0,Medium:0,Low:0};for(let J of V.findings){let Y=String(J.severity??"");if(Y in G)G[Y]=(G[Y]??0)+1}let B=Object.entries(G).filter(([,J])=>J>0).map(([J,Y])=>`${Y} ${J.toLowerCase()}`).join(", ");process.stdout.write(` Findings (iter ${V.iteration??"?"}): ${B||"none"} -- ${V.findings.length} total
50
+ `)}}if(z){let U=J0(X);if(U&&Array.isArray(U.learnings)&&U.learnings.length>0){let V=new Map;for(let B of U.learnings){let J=String(B.trigger??"unknown");V.set(J,(V.get(J)??0)+1)}let G=[...V.entries()].sort((B,J)=>J[1]-B[1]).slice(0,3).map(([B,J])=>`${J} ${B}`).join(", ");process.stdout.write(` Learnings: ${U.learnings.length} total (${G})
51
+ `)}}if(q){let U=0,V="";try{let B=(await import("fs")).readdirSync(Z).filter((J)=>J.endsWith(".md"));if(U=B.length,B.length>0)B.sort(),V=B[B.length-1]??""}catch{}if(U>0)process.stdout.write(` Escalations: ${U} handoff doc${U===1?"":"s"} (latest: ${V})
52
+ `)}}function K5(K){if(!k(K))return[];try{return t("fs").readdirSync(K).filter((X)=>/^findings-\d+\.json$/.test(X)).sort((X,Z)=>{let W=Number.parseInt(X.replace(/[^0-9]/g,""),10)||0,z=Number.parseInt(Z.replace(/[^0-9]/g,""),10)||0;return W-z}).map((X)=>x(K,X))}catch{return[]}}function J0(K){try{let $=t("fs");return JSON.parse($.readFileSync(K,"utf-8"))}catch{return null}}async function $5(){let K=await i();if(!K)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
53
+ `),1;let $=p,Q=P(),X=process.env.LOKI_DASHBOARD_PORT||"57374",Z=process.env.LOKI_PROVIDER||"claude",W=await S([K,"-c",s7,$,Q,X,Z],{timeoutMs:30000});if(W.exitCode!==0)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
54
+ `),1;return process.stdout.write(W.stdout),0}async function Q5(K){let $=[...K];while($.length>0){let Q=$[0];if(Q==="--json")return $5();if(Q==="--help"||Q==="-h")return process.stdout.write(`Usage: loki status [--json]
55
+ `),0;return process.stdout.write(`${E}Unknown flag: ${Q}${H}
56
56
  `),process.stdout.write(`Usage: loki status [--json]
57
57
  `),1}return i7()}var s7=`
58
58
  import json, os, sys, time
@@ -309,9 +309,10 @@ if os.path.isfile(gate_count_file):
309
309
  result['phase1'] = phase1
310
310
 
311
311
  print(json.dumps(result, indent=2))
312
- `;var B0=R(()=>{c();Q1();a();y()});var T0={};b(T0,{runStats:()=>q5,computeStats:()=>O0});import{readdirSync as M0,readFileSync as Q5,statSync as Y0}from"fs";import{join as d}from"path";function e(K){try{if(!Y0(K).isFile())return null;return JSON.parse(Q5(K,"utf-8"))}catch{return null}}function v1(K){try{return Y0(K).isDirectory()}catch{return!1}}function X5(K){if(!v1(K))return[];try{let $=M0(K).filter((z)=>z.startsWith("iteration-")&&z.endsWith(".json"));return $.sort(),$.map((z)=>d(K,z))}catch{return[]}}function K1(K){return Math.trunc(K).toLocaleString("en-US")}function b1(K){let $=Math.trunc(K);if($<60)return`${$}s`;let z=Math.trunc($/3600),Q=Math.trunc($%3600/60),X=$%60;if(z>0)return`${z}h ${String(Q).padStart(2,"0")}m`;return`${Q}m ${String(X).padStart(2,"0")}s`}function o(K,$=0){let z=Math.pow(10,$);return Math.round(K*z)/z}function $1(K,$){return K.toFixed($)}function y1(K,$){return K.length>=$?K:K+" ".repeat($-K.length)}function Z5(K){let $="N/A",z=0,Q=e(d(K,"state","orchestrator.json"));if(Q&&typeof Q==="object"){if(typeof Q.currentPhase==="string")$=Q.currentPhase;if(typeof Q.currentIteration==="number")z=Q.currentIteration}let X=d(K,"metrics","efficiency"),H=X5(X),Z=[];for(let A of H){let I=e(A);if(I&&typeof I==="object")Z.push(I)}if(Z.length>0)z=Math.max(z,Z.length);let q=Z.reduce((A,I)=>A+(I.input_tokens??0),0),U=Z.reduce((A,I)=>A+(I.output_tokens??0),0),V=q+U,G=Z.reduce((A,I)=>A+(I.cost_usd??0),0),B=Z.reduce((A,I)=>A+(I.duration_seconds??0),0),J=0,Y=0,j=e(d(K,"metrics","budget.json"));if(j&&typeof j==="object"){if(typeof j.budget_limit==="number")J=j.budget_limit;if(typeof j.budget_used==="number")Y=j.budget_used}let _=0,h=0,g=e(d(K,"state","quality-gates.json"));if(g&&typeof g==="object"){if(Array.isArray(g)){for(let A of g)if(h+=1,A===!0)_+=1;else if(A&&typeof A==="object"){let I=A;if(I.passed===!0||I.status==="passed")_+=1}}else for(let A of Object.values(g))if(typeof A==="boolean"){if(h+=1,A)_+=1}else if(A&&typeof A==="object"){h+=1;let I=A;if(I.passed===!0||I.status==="passed")_+=1}}let m={},T=e(d(K,"quality","gate-failure-count.json"));if(T&&typeof T==="object"&&!Array.isArray(T)){let A={};for(let[I,v]of Object.entries(T))if(typeof v==="number")A[I]=v;m=A}let L=0,f=0,r1=0,w1=d(K,"quality");if(v1(w1)){let A=[];try{A=M0(w1)}catch{A=[]}for(let I of A){if(!I.endsWith(".json")||I==="gate-failure-count.json")continue;let v=e(d(w1,I));if(!v||typeof v!=="object")continue;if(!(("verdict"in v)||("approved"in v)||("reviewers"in v)))continue;L+=1;let t1=(v.verdict??"").toString().toLowerCase();if(v.approved===!0||["approved","approve","pass"].includes(t1))f+=1;else if(["revision","revise","changes_requested","reject"].includes(t1))r1+=1}}return{phase:$,iterationCount:z,iterations:Z,totalInput:q,totalOutput:U,totalTokens:V,totalCost:G,totalDuration:B,budgetLimit:J,budgetUsed:Y,gatesPassed:_,gatesTotal:h,gateFailures:m,reviewsTotal:L,reviewsApproved:f,reviewsRevision:r1}}function H5(K,$){let z=K.iterationCount,Q={session:{iterations:z,duration_seconds:K.totalDuration,phase:K.phase},tokens:{input:K.totalInput,output:K.totalOutput,total:K.totalTokens,cost_usd:o(K.totalCost,2)},quality:{gates_passed:K.gatesPassed,gates_total:K.gatesTotal,reviews_total:K.reviewsTotal,reviews_approved:K.reviewsApproved,reviews_revision:K.reviewsRevision,gate_failures:K.gateFailures},efficiency:{avg_tokens_per_iteration:z>0?o(K.totalTokens/z,0):0,avg_cost_per_iteration:z>0?o(K.totalCost/z,2):0,avg_duration_per_iteration:z>0?o(K.totalDuration/z,1):0},budget:{used:o(K.budgetUsed,2),limit:K.budgetLimit,percent:K.budgetLimit>0?o(K.budgetUsed/K.budgetLimit*100,1):0}};if($)Q.iterations=K.iterations.map((Z,q)=>({number:q+1,input_tokens:Z.input_tokens??0,output_tokens:Z.output_tokens??0,cost_usd:o(Z.cost_usd??0,2),duration_seconds:Z.duration_seconds??0}));let X=JSON.stringify(Q,null,2);function H(Z,q){if(!q)return;let U=new RegExp(`("${Z}": )(-?\\d+)(,?)$`,"m");X=X.replace(U,(V,G,B,J)=>`${G}${B}.0${J}`)}if(H("avg_duration_per_iteration",z>0&&Number.isInteger(Q.efficiency.avg_duration_per_iteration)),H("percent",K.budgetLimit>0&&Number.isInteger(Q.budget.percent)),H("cost_usd",z>0&&Number.isInteger(Q.tokens.cost_usd)),$)X=X.replace(/("cost_usd": )(-?\d+)(,?)$/gm,(Z,q,U,V)=>`${q}${U}.0${V}`);return X}function W5(K,$){let z=[];if(z.push("Loki Mode Session Statistics"),z.push("============================"),z.push(""),z.push("Session"),z.push(` Iterations completed: ${K.iterationCount}`),z.push(` Duration: ${b1(K.totalDuration)}`),z.push(` Current phase: ${K.phase}`),z.push(""),z.push("Token Usage"),K.iterations.length>0)z.push(` Input tokens: ${K1(K.totalInput)}`),z.push(` Output tokens: ${K1(K.totalOutput)}`),z.push(` Total tokens: ${K1(K.totalTokens)}`),z.push(` Estimated cost: $${$1(K.totalCost,2)}`);else z.push(" N/A (no iteration metrics found)");if(z.push(""),z.push("Quality Gates"),K.gatesTotal>0){let Q=Math.round(K.gatesPassed/K.gatesTotal*100);z.push(` Gates passed: ${K.gatesPassed}/${K.gatesTotal} (${Q}%)`)}else z.push(" Gates passed: N/A");if(K.reviewsTotal>0){let Q=[];if(K.reviewsApproved>0)Q.push(`${K.reviewsApproved} approved`);if(K.reviewsRevision>0)Q.push(`${K.reviewsRevision} revision requested`);let X=Q.length>0?Q.join(", "):"N/A";z.push(` Code reviews: ${K.reviewsTotal} (${X})`)}if(Object.keys(K.gateFailures).length>0){let Q=Object.entries(K.gateFailures).filter(([,X])=>X>0).map(([X,H])=>`${X} (${H})`);if(Q.length>0)z.push(` Gate failures: ${Q.join(", ")}`)}if(z.push(""),z.push("Efficiency"),K.iterationCount>0&&K.iterations.length>0){let Q=Math.round(K.totalTokens/K.iterationCount),X=K.totalCost/K.iterationCount,H=K.totalDuration/K.iterationCount;z.push(` Avg tokens/iteration: ${K1(Q)}`),z.push(` Avg cost/iteration: $${$1(X,2)}`),z.push(` Avg duration/iteration: ${b1(H)}`)}else z.push(" N/A (no iteration metrics found)");if(z.push(""),z.push("Budget"),K.budgetLimit>0){let Q=o(K.budgetUsed/K.budgetLimit*100,1),X=Number.isInteger(Q)?`${Q}.0`:`${Q}`;z.push(` Used: $${$1(K.budgetUsed,2)} / $${$1(K.budgetLimit,2)} (${X}%)`)}else if(K.budgetUsed>0)z.push(` Used: $${$1(K.budgetUsed,2)} (no limit set)`);else z.push(" N/A");if($&&K.iterations.length>0)z.push(""),z.push("Per-Iteration Breakdown"),K.iterations.forEach((Q,X)=>{let H=X+1,Z=y1(K1(Q.input_tokens??0),10),q=y1(K1(Q.output_tokens??0),10),U=Q.cost_usd??0,V=b1(Q.duration_seconds??0),G=y1(`${H}`,3);z.push(` #${G} input: ${Z} output: ${q} cost: $${$1(U,2)} time: ${V}`)});return z.join(`
313
- `)}function O0(K){let $=!1,z=!1;for(let Z of K)if(Z==="--json")$=!0;else if(Z==="--efficiency")z=!0;let Q=P();if(!v1(Q)){if($)return{exitCode:0,stdout:'{"error": "No active session"}'};return{exitCode:0,stdout:`${x}No active session found.${W}
314
- Start a session with: loki start <prd>`}}let X=Z5(Q);return{exitCode:0,stdout:$?H5(X,z):W5(X,z)}}async function q5(K){let $=O0(K);return console.log($.stdout),$.exitCode}var A0=R(()=>{y();a()});var F0={};b(F0,{runDoctor:()=>I5,pythonImportOk:()=>f1,httpReachable:()=>g1,checkTool:()=>L0,checkSkills:()=>R0,checkDisk:()=>m1,buildDoctorJson:()=>x0,_setPythonImportOkForTest:()=>Y5});import{existsSync as U5,lstatSync as V5,readlinkSync as J5,statfsSync as G5}from"fs";import{homedir as _0}from"os";import{resolve as j0}from"path";function M5(K){let $=K.match(B5);return $?$[1]:null}async function I0(K){try{let $=await S([K,"--version"],{timeoutMs:5000}),z=($.stdout||$.stderr||"").trim();return M5(z)}catch{return null}}function P0(K,$){let z=K.split(".").map((X)=>parseInt(X,10)),Q=$.split(".").map((X)=>parseInt(X,10));while(z.length<2)z.push(0);while(Q.length<2)Q.push(0);for(let X=0;X<2;X++){let H=z[X]??0,Z=Q[X]??0;if(Number.isNaN(H)||Number.isNaN(Z))return 0;if(H!==Z)return H-Z}return 0}async function L0(K,$,z,Q=null){let X=await D($),H=X!==null,Z=H?await I0($):null,q="pass";if(!H)q=z==="required"?"fail":"warn";else if(Q&&Z){if(P0(Z,Q)<0)q=z==="required"?"fail":"warn"}return{name:K,command:$,found:H,version:Z,required:z,min_version:Q,status:q,path:X}}function m1(){let K=null;try{let z=G5(_0()),Q=Number(z.bavail)*Number(z.bsize);K=Math.round(Q/1073741824*10)/10}catch{K=null}let $="pass";if(K!==null){if(K<1)$="fail";else if(K<5)$="warn"}return{available_gb:K,status:$}}async function g1(K,$=2000){try{return(await fetch(K,{signal:AbortSignal.timeout($)})).ok}catch{return!1}}async function f1(K,$=!1){let z=`import ${K}`,Q=$?30000:5000;if(!$)return(await s(z,{timeoutMs:Q})).exitCode===0;let X=await t();if(!X)return!1;return(await S([X,"-c",z],{timeoutMs:Q})).exitCode===0}function Y5(K){T1.fn=K??f1}function R0(){let K=_0();return O5.map(({name:$,dir:z})=>{let Q=j0(K,z),X=Q,H=j0(Q,"SKILL.md");if(U5(H))return{name:$,path:X,status:"pass",detail:""};try{if(V5(Q).isSymbolicLink()){let q="unknown";try{q=J5(Q)}catch{}return{name:$,path:X,status:"fail",detail:`(broken symlink -> ${q})`}}}catch{}return{name:$,path:X,status:"warn",detail:"(not found - run 'loki setup-skill')"}})}async function E0(){return Promise.all(T5.map(async(K)=>{return{...await L0(K.jsonName,K.cmd,K.required,K.min??null),displayName:K.displayName}}))}async function A5(){let $=await D("sentrux")!==null,z=$?await I0("sentrux"):null;return{found:$,version:z,status:$?"pass":"warn",required:"optional"}}async function x0(){let $=(await E0()).map(({displayName:q,...U})=>U),z=m1(),Q=await A5(),X=0,H=0,Z=0;for(let q of $)if(q.status==="pass")X++;else if(q.status==="fail")H++;else Z++;if(z.status==="pass")X++;else if(z.status==="fail")H++;else Z++;return{loki_mode_version:G1(),checks:$,disk:z,sentrux:Q,summary:{passed:X,failed:H,warnings:Z,ok:H===0}}}function M(K){switch(K){case"pass":return`${C}PASS${W}`;case"fail":return`${E}FAIL${W}`;case"warn":return`${x}WARN${W}`}}function Y1(K){let $=K.version?` (v${K.version})`:"",z=K.displayName;if(!K.found){let Q=K.required==="required"?"not found":K.required==="recommended"?"not found (recommended)":"not found (optional)";return` ${M(K.status)} ${z} - ${Q}`}if(K.min_version&&K.version&&P0(K.version,K.min_version)<0){let Q=K.required==="required"?"requires":"recommended";return` ${M(K.status)} ${z}${$} - ${Q} >= ${K.min_version}`}return` ${M(K.status)} ${z}${$}`}function O1(K,$){if($==="pass")K.pass++;else if($==="fail")K.fail++;else K.warn++}function j5(){process.stdout.write(`${k}loki doctor${W} - Check system prerequisites
312
+ `;var B0=R(()=>{c();Z1();a();g()});var T0={};v(T0,{runStats:()=>q5,computeStats:()=>O0});import{readdirSync as M0,readFileSync as X5,statSync as Y0}from"fs";import{join as d}from"path";function K1(K){try{if(!Y0(K).isFile())return null;return JSON.parse(X5(K,"utf-8"))}catch{return null}}function v1(K){try{return Y0(K).isDirectory()}catch{return!1}}function Z5(K){if(!v1(K))return[];try{let $=M0(K).filter((Q)=>Q.startsWith("iteration-")&&Q.endsWith(".json"));return $.sort(),$.map((Q)=>d(K,Q))}catch{return[]}}function $1(K){return Math.trunc(K).toLocaleString("en-US")}function b1(K){let $=Math.trunc(K);if($<60)return`${$}s`;let Q=Math.trunc($/3600),X=Math.trunc($%3600/60),Z=$%60;if(Q>0)return`${Q}h ${String(X).padStart(2,"0")}m`;return`${X}m ${String(Z).padStart(2,"0")}s`}function o(K,$=0){let Q=Math.pow(10,$);return Math.round(K*Q)/Q}function Q1(K,$){return K.toFixed($)}function y1(K,$){return K.length>=$?K:K+" ".repeat($-K.length)}function z5(K){let $="N/A",Q=0,X=K1(d(K,"state","orchestrator.json"));if(X&&typeof X==="object"){if(typeof X.currentPhase==="string")$=X.currentPhase;if(typeof X.currentIteration==="number")Q=X.currentIteration}let Z=d(K,"metrics","efficiency"),W=Z5(Z),z=[];for(let _ of W){let I=K1(_);if(I&&typeof I==="object")z.push(I)}if(z.length>0)Q=Math.max(Q,z.length);let q=z.reduce((_,I)=>_+(I.input_tokens??0),0),U=z.reduce((_,I)=>_+(I.output_tokens??0),0),V=q+U,G=z.reduce((_,I)=>_+(I.cost_usd??0),0),B=z.reduce((_,I)=>_+(I.duration_seconds??0),0),J=0,Y=0,T=K1(d(K,"metrics","budget.json"));if(T&&typeof T==="object"){if(typeof T.budget_limit==="number")J=T.budget_limit;if(typeof T.budget_used==="number")Y=T.budget_used}let j=0,C=0,N=K1(d(K,"state","quality-gates.json"));if(N&&typeof N==="object"){if(Array.isArray(N)){for(let _ of N)if(C+=1,_===!0)j+=1;else if(_&&typeof _==="object"){let I=_;if(I.passed===!0||I.status==="passed")j+=1}}else for(let _ of Object.values(N))if(typeof _==="boolean"){if(C+=1,_)j+=1}else if(_&&typeof _==="object"){C+=1;let I=_;if(I.passed===!0||I.status==="passed")j+=1}}let y={},A=K1(d(K,"quality","gate-failure-count.json"));if(A&&typeof A==="object"&&!Array.isArray(A)){let _={};for(let[I,m]of Object.entries(A))if(typeof m==="number")_[I]=m;y=_}let L=0,f=0,r1=0,x1=d(K,"quality");if(v1(x1)){let _=[];try{_=M0(x1)}catch{_=[]}for(let I of _){if(!I.endsWith(".json")||I==="gate-failure-count.json")continue;let m=K1(d(x1,I));if(!m||typeof m!=="object")continue;if(!(("verdict"in m)||("approved"in m)||("reviewers"in m)))continue;L+=1;let t1=(m.verdict??"").toString().toLowerCase();if(m.approved===!0||["approved","approve","pass"].includes(t1))f+=1;else if(["revision","revise","changes_requested","reject"].includes(t1))r1+=1}}return{phase:$,iterationCount:Q,iterations:z,totalInput:q,totalOutput:U,totalTokens:V,totalCost:G,totalDuration:B,budgetLimit:J,budgetUsed:Y,gatesPassed:j,gatesTotal:C,gateFailures:y,reviewsTotal:L,reviewsApproved:f,reviewsRevision:r1}}function W5(K,$){let Q=K.iterationCount,X={session:{iterations:Q,duration_seconds:K.totalDuration,phase:K.phase},tokens:{input:K.totalInput,output:K.totalOutput,total:K.totalTokens,cost_usd:o(K.totalCost,2)},quality:{gates_passed:K.gatesPassed,gates_total:K.gatesTotal,reviews_total:K.reviewsTotal,reviews_approved:K.reviewsApproved,reviews_revision:K.reviewsRevision,gate_failures:K.gateFailures},efficiency:{avg_tokens_per_iteration:Q>0?o(K.totalTokens/Q,0):0,avg_cost_per_iteration:Q>0?o(K.totalCost/Q,2):0,avg_duration_per_iteration:Q>0?o(K.totalDuration/Q,1):0},budget:{used:o(K.budgetUsed,2),limit:K.budgetLimit,percent:K.budgetLimit>0?o(K.budgetUsed/K.budgetLimit*100,1):0}};if($)X.iterations=K.iterations.map((z,q)=>({number:q+1,input_tokens:z.input_tokens??0,output_tokens:z.output_tokens??0,cost_usd:o(z.cost_usd??0,2),duration_seconds:z.duration_seconds??0}));let Z=JSON.stringify(X,null,2);function W(z,q){if(!q)return;let U=new RegExp(`("${z}": )(-?\\d+)(,?)$`,"m");Z=Z.replace(U,(V,G,B,J)=>`${G}${B}.0${J}`)}if(W("avg_duration_per_iteration",Q>0&&Number.isInteger(X.efficiency.avg_duration_per_iteration)),W("percent",K.budgetLimit>0&&Number.isInteger(X.budget.percent)),W("cost_usd",Q>0&&Number.isInteger(X.tokens.cost_usd)),$)Z=Z.replace(/("cost_usd": )(-?\d+)(,?)$/gm,(z,q,U,V)=>`${q}${U}.0${V}`);return Z}function H5(K,$){let Q=[];if(Q.push("Loki Mode Session Statistics"),Q.push("============================"),Q.push(""),Q.push("Session"),Q.push(` Iterations completed: ${K.iterationCount}`),Q.push(` Duration: ${b1(K.totalDuration)}`),Q.push(` Current phase: ${K.phase}`),Q.push(""),Q.push("Token Usage"),K.iterations.length>0)Q.push(` Input tokens: ${$1(K.totalInput)}`),Q.push(` Output tokens: ${$1(K.totalOutput)}`),Q.push(` Total tokens: ${$1(K.totalTokens)}`),Q.push(` Estimated cost: $${Q1(K.totalCost,2)}`);else Q.push(" N/A (no iteration metrics found)");if(Q.push(""),Q.push("Quality Gates"),K.gatesTotal>0){let X=Math.round(K.gatesPassed/K.gatesTotal*100);Q.push(` Gates passed: ${K.gatesPassed}/${K.gatesTotal} (${X}%)`)}else Q.push(" Gates passed: N/A");if(K.reviewsTotal>0){let X=[];if(K.reviewsApproved>0)X.push(`${K.reviewsApproved} approved`);if(K.reviewsRevision>0)X.push(`${K.reviewsRevision} revision requested`);let Z=X.length>0?X.join(", "):"N/A";Q.push(` Code reviews: ${K.reviewsTotal} (${Z})`)}if(Object.keys(K.gateFailures).length>0){let X=Object.entries(K.gateFailures).filter(([,Z])=>Z>0).map(([Z,W])=>`${Z} (${W})`);if(X.length>0)Q.push(` Gate failures: ${X.join(", ")}`)}if(Q.push(""),Q.push("Efficiency"),K.iterationCount>0&&K.iterations.length>0){let X=Math.round(K.totalTokens/K.iterationCount),Z=K.totalCost/K.iterationCount,W=K.totalDuration/K.iterationCount;Q.push(` Avg tokens/iteration: ${$1(X)}`),Q.push(` Avg cost/iteration: $${Q1(Z,2)}`),Q.push(` Avg duration/iteration: ${b1(W)}`)}else Q.push(" N/A (no iteration metrics found)");if(Q.push(""),Q.push("Budget"),K.budgetLimit>0){let X=o(K.budgetUsed/K.budgetLimit*100,1),Z=Number.isInteger(X)?`${X}.0`:`${X}`;Q.push(` Used: $${Q1(K.budgetUsed,2)} / $${Q1(K.budgetLimit,2)} (${Z}%)`)}else if(K.budgetUsed>0)Q.push(` Used: $${Q1(K.budgetUsed,2)} (no limit set)`);else Q.push(" N/A");if($&&K.iterations.length>0)Q.push(""),Q.push("Per-Iteration Breakdown"),K.iterations.forEach((X,Z)=>{let W=Z+1,z=y1($1(X.input_tokens??0),10),q=y1($1(X.output_tokens??0),10),U=X.cost_usd??0,V=b1(X.duration_seconds??0),G=y1(`${W}`,3);Q.push(` #${G} input: ${z} output: ${q} cost: $${Q1(U,2)} time: ${V}`)});return Q.join(`
313
+ `)}function O0(K){let $=!1,Q=!1;for(let z of K)if(z==="--json")$=!0;else if(z==="--efficiency")Q=!0;let X=P();if(!v1(X)){if($)return{exitCode:0,stdout:'{"error": "No active session"}'};return{exitCode:0,stdout:`${F}No active session found.${H}
314
+ Start a session with: loki start <prd>`}}let Z=z5(X);return{exitCode:0,stdout:$?W5(Z,Q):H5(Z,Q)}}async function q5(K){let $=O0(K);return console.log($.stdout),$.exitCode}var A0=R(()=>{g();a()});var w0={};v(w0,{runDoctor:()=>P5,pythonImportOk:()=>f1,httpReachable:()=>g1,checkTool:()=>L0,checkSkills:()=>R0,checkDisk:()=>m1,buildDoctorJson:()=>F0,_setPythonImportOkForTest:()=>Y5});import{existsSync as U5,lstatSync as V5,readlinkSync as J5,statfsSync as G5}from"fs";import{homedir as _0}from"os";import{resolve as j0}from"path";function M5(K){let $=K.match(B5);return $?$[1]:null}async function I0(K){try{let $=await S([K,"--version"],{timeoutMs:5000}),Q=($.stdout||$.stderr||"").trim();return M5(Q)}catch{return null}}function P0(K,$){let Q=K.split(".").map((Z)=>parseInt(Z,10)),X=$.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 W=Q[Z]??0,z=X[Z]??0;if(Number.isNaN(W)||Number.isNaN(z))return 0;if(W!==z)return W-z}return 0}async function L0(K,$,Q,X=null){let Z=await h($),W=Z!==null,z=W?await I0($):null,q="pass";if(!W)q=Q==="required"?"fail":"warn";else if(X&&z){if(P0(z,X)<0)q=Q==="required"?"fail":"warn"}return{name:K,command:$,found:W,version:z,required:Q,min_version:X,status:q,path:Z}}function m1(){let K=null;try{let Q=G5(_0()),X=Number(Q.bavail)*Number(Q.bsize);K=Math.round(X/1073741824*10)/10}catch{K=null}let $="pass";if(K!==null){if(K<1)$="fail";else if(K<5)$="warn"}return{available_gb:K,status:$}}async function g1(K,$=2000){try{return(await fetch(K,{signal:AbortSignal.timeout($)})).ok}catch{return!1}}async function f1(K,$=!1){let Q=`import ${K}`,X=$?30000:5000;if(!$)return(await s(Q,{timeoutMs:X})).exitCode===0;let Z=await i();if(!Z)return!1;return(await S([Z,"-c",Q],{timeoutMs:X})).exitCode===0}function Y5(K){T1.fn=K??f1}function R0(){let K=_0();return O5.map(({name:$,dir:Q})=>{let X=j0(K,Q),Z=X,W=j0(X,"SKILL.md");if(U5(W))return{name:$,path:Z,status:"pass",detail:""};try{if(V5(X).isSymbolicLink()){let q="unknown";try{q=J5(X)}catch{}return{name:$,path:Z,status:"fail",detail:`(broken symlink -> ${q})`}}}catch{}return{name:$,path:Z,status:"warn",detail:"(not found - run 'loki setup-skill')"}})}async function E0(){return Promise.all(T5.map(async(K)=>{return{...await L0(K.jsonName,K.cmd,K.required,K.min??null),displayName:K.displayName}}))}async function A5(){let $=await h("sentrux")!==null,Q=$?await I0("sentrux"):null;return{found:$,version:Q,status:$?"pass":"warn",required:"optional"}}async function j5(){let{openSync:K,statSync:$,readSync:Q,closeSync:X,existsSync:Z}=await import("fs"),{join:W}=await import("path"),z=65536,q=process.env.LOKI_DIR??".loki",U=W(q,"memory",".errors.log"),V=[],G=!1;try{if(Z(U)){G=!0;let B=$(U).size,J=Math.max(0,B-65536),Y=B-J,T=Buffer.alloc(Y),j=K(U,"r");try{Q(j,T,0,Y,J)}finally{X(j)}let N=T.toString("utf-8").split(`
315
+ `);if(J>0&&N.length>0)N=N.slice(1);N=N.map((y)=>y.trim()).filter((y)=>y.length>0),V=N.slice(-5)}}catch{V=[]}return{errors_log_path:G?U:null,recent_errors:V,recent_error_count:V.length,status:V.length===0?"pass":"warn"}}async function F0(){let $=(await E0()).map(({displayName:U,...V})=>V),Q=m1(),X=await A5(),Z=await j5(),W=0,z=0,q=0;for(let U of $)if(U.status==="pass")W++;else if(U.status==="fail")z++;else q++;if(Q.status==="pass")W++;else if(Q.status==="fail")z++;else q++;return{loki_mode_version:G1(),checks:$,disk:Q,sentrux:X,memory:Z,summary:{passed:W,failed:z,warnings:q,ok:z===0}}}function M(K){switch(K){case"pass":return`${b}PASS${H}`;case"fail":return`${E}FAIL${H}`;case"warn":return`${F}WARN${H}`}}function Y1(K){let $=K.version?` (v${K.version})`:"",Q=K.displayName;if(!K.found){let X=K.required==="required"?"not found":K.required==="recommended"?"not found (recommended)":"not found (optional)";return` ${M(K.status)} ${Q} - ${X}`}if(K.min_version&&K.version&&P0(K.version,K.min_version)<0){let X=K.required==="required"?"requires":"recommended";return` ${M(K.status)} ${Q}${$} - ${X} >= ${K.min_version}`}return` ${M(K.status)} ${Q}${$}`}function O1(K,$){if($==="pass")K.pass++;else if($==="fail")K.fail++;else K.warn++}function _5(){process.stdout.write(`${D}loki doctor${H} - Check system prerequisites
315
316
 
316
317
  `),process.stdout.write(`Usage: loki doctor [--json]
317
318
 
@@ -320,34 +321,34 @@ Start a session with: loki start <prd>`}}let X=Z5(Q);return{exitCode:0,stdout:$?
320
321
 
321
322
  `),process.stdout.write(`Checks: node, python3, jq, git, curl, bash version,
322
323
  `),process.stdout.write(` claude/codex CLIs, and disk space.
323
- `)}async function _5(){process.stdout.write(`${k}Loki Mode Doctor${W}
324
+ `)}async function I5(){process.stdout.write(`${D}Loki Mode Doctor${H}
324
325
 
325
326
  `),process.stdout.write(`Checking system prerequisites...
326
327
 
327
- `);let K={pass:0,fail:0,warn:0},$=await E0(),z=new Map($.map((T)=>[T.command,T]));process.stdout.write(`${O}Required:${W}
328
- `);for(let T of["node","python3","jq","git","curl"]){let L=z.get(T);process.stdout.write(Y1(L)+`
328
+ `);let K={pass:0,fail:0,warn:0},$=await E0(),Q=new Map($.map((A)=>[A.command,A]));process.stdout.write(`${O}Required:${H}
329
+ `);for(let A of["node","python3","jq","git","curl"]){let L=Q.get(A);process.stdout.write(Y1(L)+`
329
330
  `),O1(K,L.status)}process.stdout.write(`
330
- `),process.stdout.write(`${O}AI Providers:${W}
331
- `);let Q=["claude","codex","cline","aider"],X=!1;for(let T of Q){let L=z.get(T);if(process.stdout.write(Y1(L)+`
332
- `),O1(K,L.status),L.found)X=!0}if(!X)process.stdout.write(` ${M("fail")} No AI provider CLI installed -- at least one is required
333
- `),process.stdout.write(` ${x}Install: npm install -g @anthropic-ai/claude-code${W}
331
+ `),process.stdout.write(`${O}AI Providers:${H}
332
+ `);let X=["claude","codex","cline","aider"],Z=!1;for(let A of X){let L=Q.get(A);if(process.stdout.write(Y1(L)+`
333
+ `),O1(K,L.status),L.found)Z=!0}if(!Z)process.stdout.write(` ${M("fail")} No AI provider CLI installed -- at least one is required
334
+ `),process.stdout.write(` ${F}Install: npm install -g @anthropic-ai/claude-code${H}
334
335
  `),K.fail++;process.stdout.write(`
335
- `),process.stdout.write(`${O}API Keys:${W}
336
- `);let H=z.get("claude")?.found??!1,Z=z.get("codex")?.found??!1,q=process.env;if(q.ANTHROPIC_API_KEY)process.stdout.write(` ${M("pass")} ANTHROPIC_API_KEY is set
337
- `),K.pass++;else if(H)process.stdout.write(` ${F} -- ${W} ANTHROPIC_API_KEY not set (Claude CLI uses its own login)
336
+ `),process.stdout.write(`${O}API Keys:${H}
337
+ `);let W=Q.get("claude")?.found??!1,z=Q.get("codex")?.found??!1,q=process.env;if(q.ANTHROPIC_API_KEY)process.stdout.write(` ${M("pass")} ANTHROPIC_API_KEY is set
338
+ `),K.pass++;else if(W)process.stdout.write(` ${w} -- ${H} ANTHROPIC_API_KEY not set (Claude CLI uses its own login)
338
339
  `);if(q.OPENAI_API_KEY)process.stdout.write(` ${M("pass")} OPENAI_API_KEY is set
339
- `),K.pass++;else if(Z)process.stdout.write(` ${F} -- ${W} OPENAI_API_KEY not set (Codex CLI uses its own login)
340
- `);if(q.ANTHROPIC_BASE_URL){let T=q.ANTHROPIC_BASE_URL;if(process.stdout.write(` ${M("pass")} ANTHROPIC_BASE_URL: ${T}
340
+ `),K.pass++;else if(z)process.stdout.write(` ${w} -- ${H} OPENAI_API_KEY not set (Codex CLI uses its own login)
341
+ `);if(q.ANTHROPIC_BASE_URL){let A=q.ANTHROPIC_BASE_URL;if(process.stdout.write(` ${M("pass")} ANTHROPIC_BASE_URL: ${A}
341
342
  `),K.pass++,!q.LOKI_MODEL_OVERRIDE)process.stdout.write(` ${M("warn")} LOKI_MODEL_OVERRIDE not set -- opus/sonnet/haiku aliases may not resolve on alt-provider
342
343
  `),K.warn++;else process.stdout.write(` ${M("pass")} LOKI_MODEL_OVERRIDE: ${q.LOKI_MODEL_OVERRIDE}
343
344
  `),K.pass++}process.stdout.write(`
344
- `),process.stdout.write(`${O}Skills:${W}
345
- `);for(let T of R0())if(T.status==="pass")process.stdout.write(` ${M("pass")} ${T.name} ${F}${T.path}${W}
346
- `),K.pass++;else if(T.status==="fail")process.stdout.write(` ${M("fail")} ${T.name} ${F}${T.detail}${W}
347
- `),process.stdout.write(` ${x}Fix: loki setup-skill${W}
348
- `),K.fail++;else process.stdout.write(` ${M("warn")} ${T.name} ${F}${T.detail}${W}
345
+ `),process.stdout.write(`${O}Skills:${H}
346
+ `);for(let A of R0())if(A.status==="pass")process.stdout.write(` ${M("pass")} ${A.name} ${w}${A.path}${H}
347
+ `),K.pass++;else if(A.status==="fail")process.stdout.write(` ${M("fail")} ${A.name} ${w}${A.detail}${H}
348
+ `),process.stdout.write(` ${F}Fix: loki setup-skill${H}
349
+ `),K.fail++;else process.stdout.write(` ${M("warn")} ${A.name} ${w}${A.detail}${H}
349
350
  `),K.warn++;process.stdout.write(`
350
- `),process.stdout.write(`${O}Integrations:${W}
351
+ `),process.stdout.write(`${O}Integrations:${H}
351
352
  `);let[U,V,G]=await Promise.all([T1.fn("mcp"),T1.fn("numpy",!0),T1.fn("sentence_transformers",!0)]);if(U)process.stdout.write(` ${M("pass")} MCP SDK (Python)
352
353
  `),K.pass++;else process.stdout.write(` ${M("warn")} MCP SDK - not installed (pip3 install mcp)
353
354
  `),K.warn++;if(V)process.stdout.write(` ${M("pass")} numpy (vector search)
@@ -356,48 +357,48 @@ Start a session with: loki start <prd>`}}let X=Z5(Q);return{exitCode:0,stdout:$?
356
357
  `),K.pass++;else process.stdout.write(` ${M("warn")} sentence-transformers - not installed (loki memory vectors setup)
357
358
  `),K.warn++;if(await g1("http://localhost:8100/api/v2/heartbeat"))process.stdout.write(` ${M("pass")} ChromaDB server (port 8100)
358
359
  `),K.pass++;else process.stdout.write(` ${M("warn")} ChromaDB - not running (docker start loki-chroma)
359
- `),K.warn++;{let T=["pyright-langserver","pylsp","typescript-language-server","gopls","rust-analyzer","jdtls"],L=[];for(let f of T)if(await D(f))L.push(f);if(L.length>0)process.stdout.write(` ${M("pass")} LSP servers detected (${L.length}): ${L.join(", ")}
360
+ `),K.warn++;{let A=["pyright-langserver","pylsp","typescript-language-server","gopls","rust-analyzer","jdtls"],L=[];for(let f of A)if(await h(f))L.push(f);if(L.length>0)process.stdout.write(` ${M("pass")} LSP servers detected (${L.length}): ${L.join(", ")}
360
361
  `),K.pass++;else process.stdout.write(` ${M("warn")} LSP servers - none on PATH (install for symbol grounding: npm i -g pyright typescript-language-server; brew install gopls)
361
362
  `),K.warn++}let B=process.env.LOKI_MIROFISH_URL;if(B)if(await g1(`${B}/health`))process.stdout.write(` ${M("pass")} MiroFish server (${B})
362
363
  `),K.pass++;else process.stdout.write(` ${M("warn")} MiroFish - not running (loki start --mirofish-docker <image>)
363
364
  `),K.warn++;if(process.env.LOKI_OTEL_ENDPOINT)process.stdout.write(` ${M("pass")} OTEL endpoint: ${process.env.LOKI_OTEL_ENDPOINT}
364
365
  `),K.pass++;else process.stdout.write(` ${M("warn")} OTEL - not configured (set LOKI_OTEL_ENDPOINT)
365
- `),K.warn++;if(await D("sentrux")){let T="unknown";try{let f=(await S(["sentrux","--version"],{timeoutMs:2000})).stdout.split(/\s+/).filter(Boolean).pop();if(f)T=f.replace(/^v/,"")}catch{}process.stdout.write(` ${M("pass")} sentrux ${T} (architectural drift gate: loki sentrux help)
366
+ `),K.warn++;if(await h("sentrux")){let A="unknown";try{let f=(await S(["sentrux","--version"],{timeoutMs:2000})).stdout.split(/\s+/).filter(Boolean).pop();if(f)A=f.replace(/^v/,"")}catch{}process.stdout.write(` ${M("pass")} sentrux ${A} (architectural drift gate: loki sentrux help)
366
367
  `),K.pass++}else process.stdout.write(` ${M("warn")} sentrux - not installed (optional, brew install sentrux/tap/sentrux)
367
368
  `),K.warn++;process.stdout.write(`
368
- `),process.stdout.write(`${O}System:${W}
369
- `);let J=z.get("bash");process.stdout.write(Y1(J)+`
370
- `),O1(K,J.status);let Y=z.get("bun");if(Y)process.stdout.write(Y1(Y)+`
371
- `),O1(K,Y.status);let j=m1(),_=j.available_gb===null?null:Math.floor(j.available_gb);if(_===null)process.stdout.write(` ${M("warn")} Disk space: unable to determine
372
- `),K.warn++;else if(j.status==="fail")process.stdout.write(` ${M("fail")} Disk space: ${_}GB available (need >= 1GB)
373
- `),K.fail++;else if(j.status==="warn")process.stdout.write(` ${M("warn")} Disk space: ${_}GB available (low)
374
- `),K.warn++;else process.stdout.write(` ${M("pass")} Disk space: ${_}GB available
369
+ `),process.stdout.write(`${O}System:${H}
370
+ `);let J=Q.get("bash");process.stdout.write(Y1(J)+`
371
+ `),O1(K,J.status);let Y=Q.get("bun");if(Y)process.stdout.write(Y1(Y)+`
372
+ `),O1(K,Y.status);let T=m1(),j=T.available_gb===null?null:Math.floor(T.available_gb);if(j===null)process.stdout.write(` ${M("warn")} Disk space: unable to determine
373
+ `),K.warn++;else if(T.status==="fail")process.stdout.write(` ${M("fail")} Disk space: ${j}GB available (need >= 1GB)
374
+ `),K.fail++;else if(T.status==="warn")process.stdout.write(` ${M("warn")} Disk space: ${j}GB available (low)
375
+ `),K.warn++;else process.stdout.write(` ${M("pass")} Disk space: ${j}GB available
375
376
  `),K.pass++;process.stdout.write(`
376
- `),process.stdout.write(`${O}Runtime route:${W}
377
- `);let h=process.versions.bun!==void 0,g=process.argv[0]??"(unknown)";if(process.stdout.write(` ${M("pass")} Active runtime: ${h?"Bun":"Node"} (${g})
377
+ `),process.stdout.write(`${O}Runtime route:${H}
378
+ `);let C=process.versions.bun!==void 0,N=process.argv[0]??"(unknown)";if(process.stdout.write(` ${M("pass")} Active runtime: ${C?"Bun":"Node"} (${N})
378
379
  `),process.env.LOKI_LEGACY_BASH==="1"||process.env.LOKI_LEGACY_BASH==="true")process.stdout.write(` ${M("warn")} LOKI_LEGACY_BASH set: shim routes every command to autonomy/loki (bash)
379
380
  `);if(process.env.LOKI_TS_ENTRY)process.stdout.write(` ${M("pass")} LOKI_TS_ENTRY override: ${process.env.LOKI_TS_ENTRY}
380
381
  `);if(process.env.BUN_FROM_SOURCE==="1"||process.env.BUN_FROM_SOURCE==="true")process.stdout.write(` ${M("pass")} BUN_FROM_SOURCE set: shim prefers loki-ts/src/ over dist/
381
- `);let m=await t();if(m!==null){let L=(await S([m,"-c","import sys; print(f'{sys.version_info.major}.{sys.version_info.minor}.{sys.version_info.micro}')"],{timeoutMs:5000})).stdout.trim();if(L.startsWith("3.12"))process.stdout.write(` ${M("pass")} Python 3.12 (chromadb / sentence-transformers): ${L} at ${m}
382
- `);else if(L)process.stdout.write(` ${M("warn")} Python 3.12 NOT found -- using ${L} at ${m}; chromadb / sentence-transformers may fail. Install python3.12 (brew install python@3.12 / apt install python3.12).
383
- `);else process.stdout.write(` ${M("warn")} Python 3 found at ${m} but version probe failed; chromadb may not work.
382
+ `);let y=await i();if(y!==null){let L=(await S([y,"-c","import sys; print(f'{sys.version_info.major}.{sys.version_info.minor}.{sys.version_info.micro}')"],{timeoutMs:5000})).stdout.trim();if(L.startsWith("3.12"))process.stdout.write(` ${M("pass")} Python 3.12 (chromadb / sentence-transformers): ${L} at ${y}
383
+ `);else if(L)process.stdout.write(` ${M("warn")} Python 3.12 NOT found -- using ${L} at ${y}; chromadb / sentence-transformers may fail. Install python3.12 (brew install python@3.12 / apt install python3.12).
384
+ `);else process.stdout.write(` ${M("warn")} Python 3 found at ${y} but version probe failed; chromadb may not work.
384
385
  `)}else process.stdout.write(` ${M("warn")} Python 3 not on PATH -- memory + MCP integrations disabled.
385
386
  `);if(process.stdout.write(`
386
- `),process.stdout.write(`${k}Summary:${W} ${C}${K.pass} passed${W}, ${E}${K.fail} failed${W}, ${x}${K.warn} warnings${W}
387
+ `),process.stdout.write(`${D}Summary:${H} ${b}${K.pass} passed${H}, ${E}${K.fail} failed${H}, ${F}${K.warn} warnings${H}
387
388
 
388
- `),K.fail>0)return process.stdout.write(`${E}Some required prerequisites are missing.${W}
389
+ `),K.fail>0)return process.stdout.write(`${E}Some required prerequisites are missing.${H}
389
390
  `),process.stdout.write(`Install missing dependencies and run 'loki doctor' again.
390
- `),1;if(K.warn>0)return process.stdout.write(`${x}All required checks passed with some warnings.${W}
391
- `),0;return process.stdout.write(`${C}All checks passed. System is ready for Loki Mode.${W}
392
- `),0}async function I5(K){let $=!1;for(let z of K)if(z==="--json")$=!0;else if(z==="--help"||z==="-h")return j5(),0;else return process.stderr.write(`${E}Unknown option: ${z}${W}
391
+ `),1;if(K.warn>0)return process.stdout.write(`${F}All required checks passed with some warnings.${H}
392
+ `),0;return process.stdout.write(`${b}All checks passed. System is ready for Loki Mode.${H}
393
+ `),0}async function P5(K){let $=!1;for(let Q of K)if(Q==="--json")$=!0;else if(Q==="--help"||Q==="-h")return _5(),0;else return process.stderr.write(`${E}Unknown option: ${Q}${H}
393
394
  `),process.stderr.write(`Usage: loki doctor [--json]
394
- `),1;if($){let z=await x0();return process.stdout.write(JSON.stringify(z,null,2)+`
395
- `),0}return _5()}var B5,T1,O5,T5;var w0=R(()=>{c();Q1();a();D1();B5=/(\d+\.\d+(?:\.\d+)*)/;T1={fn:f1};O5=[{name:"Claude Code",dir:".claude/skills/loki-mode"},{name:"Codex CLI",dir:".codex/skills/loki-mode"},{name:"Cline CLI",dir:".cline/skills/loki-mode"},{name:"Aider CLI",dir:".aider/skills/loki-mode"}];T5=[{displayName:"Node.js (>= 18)",jsonName:"Node.js",cmd:"node",required:"required",min:"18.0"},{displayName:"Python 3 (>= 3.8)",jsonName:"Python 3",cmd:"python3",required:"required",min:"3.8"},{displayName:"jq",jsonName:"jq",cmd:"jq",required:"required"},{displayName:"git",jsonName:"git",cmd:"git",required:"required"},{displayName:"curl",jsonName:"curl",cmd:"curl",required:"required"},{displayName:"bash (>= 4.0)",jsonName:"bash",cmd:"bash",required:"recommended",min:"4.0"},{displayName:"Bun (>= 1.3)",jsonName:"Bun",cmd:"bun",required:"recommended",min:"1.3"},{displayName:"Claude CLI",jsonName:"Claude CLI",cmd:"claude",required:"optional"},{displayName:"Codex CLI",jsonName:"Codex CLI",cmd:"codex",required:"optional"},{displayName:"Cline CLI",jsonName:"Cline CLI",cmd:"cline",required:"optional"},{displayName:"Aider CLI",jsonName:"Aider CLI",cmd:"aider",required:"optional"}]});import{existsSync as k0,mkdirSync as e6,readdirSync as P5,readFileSync as D0,renameSync as K8,writeFileSync as $8}from"fs";import{dirname as L5,join as R5,resolve as E5}from"path";import{fileURLToPath as x5}from"url";function F5(){try{let K=L5(x5(import.meta.url)),$=E5(K,"..","..","data","model-pricing.json");if(!k0($))return H1;let Q=JSON.parse(D0($,"utf8")).pricing;if(!Q||typeof Q!=="object")return H1;let X={};for(let[H,Z]of Object.entries(Q))if(Z!==null&&typeof Z==="object"&&typeof Z.input==="number"&&typeof Z.output==="number")X[H]={input:Z.input,output:Z.output};for(let H of Object.keys(H1))if(!(H in X))return H1;return X}catch{return H1}}function w5(K){return Math.round((K+Number.EPSILON)*1e4)/1e4}function S5(K){let $=(K??N0).toLowerCase();return S0[$]??S0[N0]}function C0(K){let $=0;for(let z of K){if(typeof z.cost_usd==="number"&&Number.isFinite(z.cost_usd)){$+=z.cost_usd;continue}let Q=S5(z.model),X=typeof z.input_tokens==="number"?z.input_tokens:0,H=typeof z.output_tokens==="number"?z.output_tokens:0;$+=X/1e6*Q.input+H/1e6*Q.output}return w5($)}function h0(K){if(!k0(K))return[];let $=[],z;try{z=P5(K)}catch{return[]}for(let Q of z){if(!Q.endsWith(".json"))continue;let X=R5(K,Q);try{let H=D0(X,"utf8"),Z=JSON.parse(H);if(Z&&typeof Z==="object")$.push(Z)}catch{}}return $}var H1,S0,N0="sonnet";var b0=R(()=>{y();H1={opus:{input:5,output:25},sonnet:{input:3,output:15},haiku:{input:1,output:5},"gpt-5.3-codex":{input:1.5,output:12}};S0=Object.freeze(F5())});import{existsSync as A1,readdirSync as N5,readFileSync as k5,statSync as D5}from"fs";import{join as j1}from"path";function C5(K){let $=[],z=j1(K,"votes");if(!A1(z))return $;let Q;try{Q=N5(z)}catch{return $}for(let X of Q){if(!X.startsWith("round-")||!X.endsWith(".json"))continue;try{let H=j1(z,X);if(!D5(H).isFile())continue;let Z=JSON.parse(k5(H,"utf8"));$.push({iteration:typeof Z.iteration==="number"?Z.iteration:void 0,verdict:typeof Z.verdict==="string"?Z.verdict:void 0,complete_votes:typeof Z.complete_votes==="number"?Z.complete_votes:void 0,total_members:typeof Z.total_members==="number"?Z.total_members:void 0,threshold:typeof Z.threshold==="number"?Z.threshold:void 0})}catch{}}return $}function h5(){return{iteration_count:0,total_cost_usd:0,avg_cost_per_iteration:null,total_input_tokens:0,total_output_tokens:0,total_duration_ms:0,avg_duration_ms_per_iteration:null,model_breakdown:{},phase_breakdown:{},status_breakdown:{}}}function b5(){return{council_rounds:0,unanimous_rate:null,approval_rate:null,iteration_success_rate:null}}function y5(K){let $=h5();if(K.length===0)return $;$.iteration_count=K.length,$.total_cost_usd=Math.round(C0(K)*1e4)/1e4;for(let z of K){if(typeof z.input_tokens==="number")$.total_input_tokens+=z.input_tokens;if(typeof z.output_tokens==="number")$.total_output_tokens+=z.output_tokens;let Q=z;if(typeof Q.duration_ms==="number")$.total_duration_ms+=Q.duration_ms;if(typeof z.model==="string")$.model_breakdown[z.model]=($.model_breakdown[z.model]??0)+1;if(typeof Q.phase==="string")$.phase_breakdown[Q.phase]=($.phase_breakdown[Q.phase]??0)+1;if(typeof Q.status==="string")$.status_breakdown[Q.status]=($.status_breakdown[Q.status]??0)+1}return $.avg_cost_per_iteration=Math.round($.total_cost_usd/$.iteration_count*1e4)/1e4,$.avg_duration_ms_per_iteration=Math.round($.total_duration_ms/$.iteration_count),$}function v5(K,$,z){let Q=b5();if(Q.council_rounds=K.length,K.length>0){let X=0,H=0;for(let Z of K){if(typeof Z.complete_votes==="number"&&typeof Z.total_members==="number"&&Z.total_members>0&&Z.complete_votes===Z.total_members)X+=1;if(Z.verdict==="COMPLETE")H+=1}Q.unanimous_rate=Math.round(X/K.length*1e4)/1e4,Q.approval_rate=Math.round(H/K.length*1e4)/1e4}if(z>0)Q.iteration_success_rate=Math.round($/z*1e4)/1e4;return Q}function y0(K){let $=[],z=j1(K,"metrics","efficiency"),Q=j1(K,"council"),X=A1(z)?h0(z):[];if(!A1(z))$.push("no .loki/metrics/efficiency/ dir (efficiency KPIs zeroed)");else if(X.length===0)$.push(".loki/metrics/efficiency/ exists but no iteration files found");let H=C5(Q);if(!A1(Q))$.push("no .loki/council/ dir (accuracy KPIs zeroed)");else if(H.length===0)$.push(".loki/council/ exists but no round-N.json files found");let Z=y5(X),q=Z.status_breakdown.success??0,U=v5(H,q,Z.iteration_count);return{schema_version:1,generated_at:new Date().toISOString(),loki_dir:K,efficiency:Z,accuracy:U,notes:$}}function v0(K){return JSON.stringify(K,null,2)}function g0(K){let $=[];$.push(`Loki Mode KPIs (snapshot at ${K.generated_at})`),$.push(`Source: ${K.loki_dir}`),$.push(""),$.push("Efficiency"),$.push(` Iterations: ${K.efficiency.iteration_count}`),$.push(` Total cost USD: ${K.efficiency.total_cost_usd}`),$.push(` Avg cost per iter: ${K.efficiency.avg_cost_per_iteration??"n/a"}`),$.push(` Total input tokens: ${K.efficiency.total_input_tokens}`),$.push(` Total output tokens: ${K.efficiency.total_output_tokens}`),$.push(` Total duration (ms): ${K.efficiency.total_duration_ms}`),$.push(` Avg duration / iter: ${K.efficiency.avg_duration_ms_per_iteration??"n/a"}`);let z=Object.entries(K.efficiency.model_breakdown).sort((H,Z)=>H[0].localeCompare(Z[0]));if(z.length>0)$.push(` Model breakdown: ${z.map(([H,Z])=>`${H}=${Z}`).join(", ")}`);let Q=Object.entries(K.efficiency.phase_breakdown).sort((H,Z)=>H[0].localeCompare(Z[0]));if(Q.length>0)$.push(` Phase breakdown: ${Q.map(([H,Z])=>`${H}=${Z}`).join(", ")}`);let X=Object.entries(K.efficiency.status_breakdown).sort((H,Z)=>H[0].localeCompare(Z[0]));if(X.length>0)$.push(` Status breakdown: ${X.map(([H,Z])=>`${H}=${Z}`).join(", ")}`);if($.push(""),$.push("Accuracy"),$.push(` Council rounds: ${K.accuracy.council_rounds}`),$.push(` Unanimous rate: ${K.accuracy.unanimous_rate??"n/a"}`),$.push(` Approval rate: ${K.accuracy.approval_rate??"n/a"}`),$.push(` Iter success rate: ${K.accuracy.iteration_success_rate??"n/a"}`),K.notes.length>0){$.push(""),$.push("Notes");for(let H of K.notes)$.push(` - ${H}`)}return $.join(`
396
- `)}var m0=R(()=>{b0()});var f0={};b(f0,{runKpis:()=>m5});function m5(K){let $=!1;for(let Q of K){if(Q==="--help"||Q==="-h"||Q==="help")return process.stdout.write(g5),0;if(Q==="--json"){$=!0;continue}return process.stderr.write(`loki kpis: unknown arg: ${Q}
395
+ `),1;if($){let Q=await F0();return process.stdout.write(JSON.stringify(Q,null,2)+`
396
+ `),0}return I5()}var B5,T1,O5,T5;var x0=R(()=>{c();Z1();a();D1();B5=/(\d+\.\d+(?:\.\d+)*)/;T1={fn:f1};O5=[{name:"Claude Code",dir:".claude/skills/loki-mode"},{name:"Codex CLI",dir:".codex/skills/loki-mode"},{name:"Cline CLI",dir:".cline/skills/loki-mode"},{name:"Aider CLI",dir:".aider/skills/loki-mode"}];T5=[{displayName:"Node.js (>= 18)",jsonName:"Node.js",cmd:"node",required:"required",min:"18.0"},{displayName:"Python 3 (>= 3.8)",jsonName:"Python 3",cmd:"python3",required:"required",min:"3.8"},{displayName:"jq",jsonName:"jq",cmd:"jq",required:"required"},{displayName:"git",jsonName:"git",cmd:"git",required:"required"},{displayName:"curl",jsonName:"curl",cmd:"curl",required:"required"},{displayName:"bash (>= 4.0)",jsonName:"bash",cmd:"bash",required:"recommended",min:"4.0"},{displayName:"Bun (>= 1.3)",jsonName:"Bun",cmd:"bun",required:"recommended",min:"1.3"},{displayName:"Claude CLI",jsonName:"Claude CLI",cmd:"claude",required:"optional"},{displayName:"Codex CLI",jsonName:"Codex CLI",cmd:"codex",required:"optional"},{displayName:"Cline CLI",jsonName:"Cline CLI",cmd:"cline",required:"optional"},{displayName:"Aider CLI",jsonName:"Aider CLI",cmd:"aider",required:"optional"}]});import{existsSync as k0,mkdirSync as K8,readdirSync as L5,readFileSync as D0,renameSync as $8,writeFileSync as Q8}from"fs";import{dirname as R5,join as E5,resolve as F5}from"path";import{fileURLToPath as w5}from"url";function x5(){try{let K=R5(w5(import.meta.url)),$=F5(K,"..","..","data","model-pricing.json");if(!k0($))return H1;let X=JSON.parse(D0($,"utf8")).pricing;if(!X||typeof X!=="object")return H1;let Z={};for(let[W,z]of Object.entries(X))if(z!==null&&typeof z==="object"&&typeof z.input==="number"&&typeof z.output==="number")Z[W]={input:z.input,output:z.output};for(let W of Object.keys(H1))if(!(W in Z))return H1;return Z}catch{return H1}}function S5(K){return Math.round((K+Number.EPSILON)*1e4)/1e4}function N5(K){let $=(K??N0).toLowerCase();return S0[$]??S0[N0]}function C0(K){let $=0;for(let Q of K){if(typeof Q.cost_usd==="number"&&Number.isFinite(Q.cost_usd)){$+=Q.cost_usd;continue}let X=N5(Q.model),Z=typeof Q.input_tokens==="number"?Q.input_tokens:0,W=typeof Q.output_tokens==="number"?Q.output_tokens:0;$+=Z/1e6*X.input+W/1e6*X.output}return S5($)}function h0(K){if(!k0(K))return[];let $=[],Q;try{Q=L5(K)}catch{return[]}for(let X of Q){if(!X.endsWith(".json"))continue;let Z=E5(K,X);try{let W=D0(Z,"utf8"),z=JSON.parse(W);if(z&&typeof z==="object")$.push(z)}catch{}}return $}var H1,S0,N0="sonnet";var b0=R(()=>{g();H1={opus:{input:5,output:25},sonnet:{input:3,output:15},haiku:{input:1,output:5},"gpt-5.3-codex":{input:1.5,output:12}};S0=Object.freeze(x5())});import{existsSync as A1,readdirSync as k5,readFileSync as D5,statSync as C5}from"fs";import{join as j1}from"path";function h5(K){let $=[],Q=j1(K,"votes");if(!A1(Q))return $;let X;try{X=k5(Q)}catch{return $}for(let Z of X){if(!Z.startsWith("round-")||!Z.endsWith(".json"))continue;try{let W=j1(Q,Z);if(!C5(W).isFile())continue;let z=JSON.parse(D5(W,"utf8"));$.push({iteration:typeof z.iteration==="number"?z.iteration:void 0,verdict:typeof z.verdict==="string"?z.verdict:void 0,complete_votes:typeof z.complete_votes==="number"?z.complete_votes:void 0,total_members:typeof z.total_members==="number"?z.total_members:void 0,threshold:typeof z.threshold==="number"?z.threshold:void 0})}catch{}}return $}function b5(){return{iteration_count:0,total_cost_usd:0,avg_cost_per_iteration:null,total_input_tokens:0,total_output_tokens:0,total_duration_ms:0,avg_duration_ms_per_iteration:null,model_breakdown:{},phase_breakdown:{},status_breakdown:{}}}function y5(){return{council_rounds:0,unanimous_rate:null,approval_rate:null,iteration_success_rate:null}}function v5(K){let $=b5();if(K.length===0)return $;$.iteration_count=K.length,$.total_cost_usd=Math.round(C0(K)*1e4)/1e4;for(let Q of K){if(typeof Q.input_tokens==="number")$.total_input_tokens+=Q.input_tokens;if(typeof Q.output_tokens==="number")$.total_output_tokens+=Q.output_tokens;let X=Q;if(typeof X.duration_ms==="number")$.total_duration_ms+=X.duration_ms;if(typeof Q.model==="string")$.model_breakdown[Q.model]=($.model_breakdown[Q.model]??0)+1;if(typeof X.phase==="string")$.phase_breakdown[X.phase]=($.phase_breakdown[X.phase]??0)+1;if(typeof X.status==="string")$.status_breakdown[X.status]=($.status_breakdown[X.status]??0)+1}return $.avg_cost_per_iteration=Math.round($.total_cost_usd/$.iteration_count*1e4)/1e4,$.avg_duration_ms_per_iteration=Math.round($.total_duration_ms/$.iteration_count),$}function g5(K,$,Q){let X=y5();if(X.council_rounds=K.length,K.length>0){let Z=0,W=0;for(let z of K){if(typeof z.complete_votes==="number"&&typeof z.total_members==="number"&&z.total_members>0&&z.complete_votes===z.total_members)Z+=1;if(z.verdict==="COMPLETE")W+=1}X.unanimous_rate=Math.round(Z/K.length*1e4)/1e4,X.approval_rate=Math.round(W/K.length*1e4)/1e4}if(Q>0)X.iteration_success_rate=Math.round($/Q*1e4)/1e4;return X}function y0(K){let $=[],Q=j1(K,"metrics","efficiency"),X=j1(K,"council"),Z=A1(Q)?h0(Q):[];if(!A1(Q))$.push("no .loki/metrics/efficiency/ dir (efficiency KPIs zeroed)");else if(Z.length===0)$.push(".loki/metrics/efficiency/ exists but no iteration files found");let W=h5(X);if(!A1(X))$.push("no .loki/council/ dir (accuracy KPIs zeroed)");else if(W.length===0)$.push(".loki/council/ exists but no round-N.json files found");let z=v5(Z),q=z.status_breakdown.success??0,U=g5(W,q,z.iteration_count);return{schema_version:1,generated_at:new Date().toISOString(),loki_dir:K,efficiency:z,accuracy:U,notes:$}}function v0(K){return JSON.stringify(K,null,2)}function g0(K){let $=[];$.push(`Loki Mode KPIs (snapshot at ${K.generated_at})`),$.push(`Source: ${K.loki_dir}`),$.push(""),$.push("Efficiency"),$.push(` Iterations: ${K.efficiency.iteration_count}`),$.push(` Total cost USD: ${K.efficiency.total_cost_usd}`),$.push(` Avg cost per iter: ${K.efficiency.avg_cost_per_iteration??"n/a"}`),$.push(` Total input tokens: ${K.efficiency.total_input_tokens}`),$.push(` Total output tokens: ${K.efficiency.total_output_tokens}`),$.push(` Total duration (ms): ${K.efficiency.total_duration_ms}`),$.push(` Avg duration / iter: ${K.efficiency.avg_duration_ms_per_iteration??"n/a"}`);let Q=Object.entries(K.efficiency.model_breakdown).sort((W,z)=>W[0].localeCompare(z[0]));if(Q.length>0)$.push(` Model breakdown: ${Q.map(([W,z])=>`${W}=${z}`).join(", ")}`);let X=Object.entries(K.efficiency.phase_breakdown).sort((W,z)=>W[0].localeCompare(z[0]));if(X.length>0)$.push(` Phase breakdown: ${X.map(([W,z])=>`${W}=${z}`).join(", ")}`);let Z=Object.entries(K.efficiency.status_breakdown).sort((W,z)=>W[0].localeCompare(z[0]));if(Z.length>0)$.push(` Status breakdown: ${Z.map(([W,z])=>`${W}=${z}`).join(", ")}`);if($.push(""),$.push("Accuracy"),$.push(` Council rounds: ${K.accuracy.council_rounds}`),$.push(` Unanimous rate: ${K.accuracy.unanimous_rate??"n/a"}`),$.push(` Approval rate: ${K.accuracy.approval_rate??"n/a"}`),$.push(` Iter success rate: ${K.accuracy.iteration_success_rate??"n/a"}`),K.notes.length>0){$.push(""),$.push("Notes");for(let W of K.notes)$.push(` - ${W}`)}return $.join(`
397
+ `)}var m0=R(()=>{b0()});var f0={};v(f0,{runKpis:()=>f5});function f5(K){let $=!1;for(let X of K){if(X==="--help"||X==="-h"||X==="help")return process.stdout.write(m5),0;if(X==="--json"){$=!0;continue}return process.stderr.write(`loki kpis: unknown arg: ${X}
397
398
  Run 'loki kpis --help' for usage.
398
- `),1}let z=y0(P());return process.stdout.write($?v0(z)+`
399
- `:g0(z)+`
400
- `),0}var g5=`loki kpis -- accuracy + efficiency KPI snapshot (v7.5.28 MVP)
399
+ `),1}let Q=y0(P());return process.stdout.write($?v0(Q)+`
400
+ `:g0(Q)+`
401
+ `),0}var m5=`loki kpis -- accuracy + efficiency KPI snapshot (v7.5.28 MVP)
401
402
 
402
403
  Usage:
403
404
  loki kpis Pretty-print KPI snapshot
@@ -417,22 +418,22 @@ iteration success rate.
417
418
  This is the Phase K MVP -- read-only derivation. Per-iteration
418
419
  emission, dashboard panel, and the loki-bench harness are deferred
419
420
  follow-ups (see project_v7_5_18_arc_status.md).
420
- `;var u0=R(()=>{m0();y()});import{closeSync as B8,fstatSync as M8,lstatSync as Y8,mkdirSync as f5,openSync as O8,readSync as T8,renameSync as u5,rmSync as A8,statSync as j8,unlinkSync as _8,writeFileSync as p5,writeSync as I8}from"fs";import{dirname as c5}from"path";function W1(K,$){f5(c5(K),{recursive:!0});let z=`${K}.tmp.${process.pid}.${++l5}`;p5(z,`${JSON.stringify($,null,2)}
421
- `),u5(z,K)}async function p0(K,$){let z=_1.get(K)??Promise.resolve(),Q=()=>{},X=new Promise((Z)=>{Q=Z}),H=z.catch(()=>{}).then(()=>X);_1.set(K,H);try{return await z.catch(()=>{}),await $()}finally{if(Q(),_1.get(K)===H)_1.delete(K)}}var l5=0,_1;var I1=R(()=>{_1=new Map});import{existsSync as P1,mkdirSync as d5,copyFileSync as o5,readFileSync as n5,readdirSync as a5,statSync as s5,writeFileSync as x8,renameSync as r5,appendFileSync as F8,rmSync as w8}from"fs";import{join as r,dirname as t5}from"path";function L1(K){return r(K,"state","checkpoints")}function e5(K){let $=L1(K);if(!P1($))return[];return a5($).filter((z)=>z.startsWith("cp-")).filter((z)=>{try{return s5(r($,z)).isDirectory()}catch{return!1}})}function KK(K){return[...K].sort(($,z)=>{let Q=c0($),X=c0(z);return Q-X})}function c0(K){let $=K.split("-");if($.length<3)return 0;let z=$[$.length-1],Q=Number.parseInt(z??"0",10);return Number.isFinite(Q)?Q:0}function p1(K){let $=K??P(),z=KK(e5($)),Q=[];for(let X of z){let H=l0($,X);if(H)Q.push(H)}return Q}function l0(K,$){let z=r(L1(K),$,"metadata.json");if(!P1(z))return null;try{let Q=JSON.parse(n5(z,"utf-8"));return $K(Q,z)}catch{return null}}function $K(K,$){let z=XK(K,$);return z.ok?z.value:null}function XK(K,$){if(K===null||typeof K!=="object")return console.warn(`[checkpoint] invalid metadata at ${$}: not an object`),{ok:!1,reason:"invalid_type",field:"<root>"};let z=K,Q=["id","timestamp","task_id","task_description","git_sha","git_branch","provider","phase"];for(let X of Q){if(!(X in z))return console.warn(`[checkpoint] invalid metadata at ${$}: field "${X}" missing`),{ok:!1,reason:"missing_field",field:X};if(typeof z[X]!=="string")return console.warn(`[checkpoint] invalid metadata at ${$}: field "${X}" not a string`),{ok:!1,reason:"invalid_type",field:X}}if(!Object.prototype.hasOwnProperty.call(z,"iteration"))return console.warn(`[checkpoint] invalid metadata at ${$}: 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 ${$}: field "iteration" not a finite number`),{ok:!1,reason:"invalid_type",field:"iteration"};for(let X of QK){let H=z[X];if(zK.test(H))return console.warn(`[checkpoint] invalid metadata at ${$}: field "${X}" contains control characters`),{ok:!1,reason:"control_chars",field:X}}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 c1(K,$){if(!ZK.test(K))throw new d0(K);let z=$??P(),Q=r(L1(z),K);if(!P1(Q))throw new u1(K);let X=l0(z,K);if(!X)throw new u1(K);return X}function o0(K,$){let z=c1(K,$),Q=$??P(),X=r(L1(Q),K),H=[];for(let Z of HK){let q=r(X,Z);if(!P1(q))continue;H.push({from:q,to:r(Q,Z)})}return{id:K,metadata:z,restore:H}}function n0(K){let $=[],z=0;for(let Q of K.restore)try{d5(t5(Q.to),{recursive:!0});let X=`${Q.to}.tmp.${process.pid}.${++i5}`;o5(Q.from,X),r5(X,Q.to),z+=1}catch(X){$.push(`${Q.from} -> ${Q.to}: ${X.message}`)}return{restored:z,errors:$}}var C8,i5=0,zK,QK,ZK,u1,d0,HK;var a0=R(()=>{y();c();I1();C8=Promise.resolve();zK=/[\x00-\x08\x0a-\x1f\x7f-\x9f]/,QK=["id","task_id","git_sha","git_branch","provider","phase"];ZK=/^[a-zA-Z0-9_-]+$/;u1=class u1 extends Error{id;constructor(K){super(`Checkpoint not found: ${K}`);this.id=K;this.name="CheckpointNotFoundError"}};d0=class d0 extends Error{id;constructor(K){super(`Invalid checkpoint ID: must be alphanumeric, hyphens, underscores only (got: ${K})`);this.id=K;this.name="InvalidCheckpointIdError"}};HK=["state/orchestrator.json","queue/pending.json","queue/completed.json","queue/in-progress.json","queue/current-task.json"]});var t0={};b(t0,{runRollback:()=>WK});async function WK(K){let $=K[0],z=K.slice(1);if($===void 0||$==="help"||$==="--help"||$==="-h")return process.stdout.write(s0),$===void 0?1:0;switch($){case"list":{let Q=[...p1()].reverse();if(Q.length===0)return process.stdout.write(`${x}No checkpoints found.${W}
422
- `),0;process.stdout.write(`${k}Checkpoints${W} (${Q.length}, newest first):
423
- `);for(let X of Q)process.stdout.write(` ${O}${X.id}${W} iter=${X.iteration} ${X.git_branch||"(no branch)"}@${(X.git_sha||"").slice(0,7)} ${X.timestamp}
424
- `);return 0}case"show":{let Q=z[0];if(!Q)return process.stderr.write(`${E}Missing checkpoint id.${W} Use \`loki rollback list\`.
425
- `),2;try{let X=c1(Q);return process.stdout.write(`${JSON.stringify(X,null,2)}
426
- `),0}catch(X){return process.stderr.write(`${E}Failed to read checkpoint:${W} ${X.message}
427
- `),1}}case"to":{let Q=z[0];if(!Q)return process.stderr.write(`${E}Missing checkpoint id.${W} Use \`loki rollback list\`.
428
- `),2;return r0(Q)}case"latest":{let Q=p1(),X=Q[Q.length-1];if(!X)return process.stderr.write(`${E}No checkpoints found to roll back to.${W}
429
- `),1;return process.stdout.write(`Rolling back to latest checkpoint: ${O}${X.id}${W}
430
- `),r0(X.id)}default:return process.stderr.write(`Unknown subcommand: ${$}
431
- `),process.stderr.write(s0),2}}function r0(K){let $;try{$=o0(K)}catch(Q){return process.stderr.write(`${E}Cannot plan rollback:${W} ${Q.message}
432
- `),1}if($.restore.length===0)return process.stdout.write(`${x}Checkpoint ${K} has no restorable state files; nothing to do.${W}
433
- `),0;let z=n0($);if(z.errors.length>0){for(let Q of z.errors)process.stderr.write(`${E}restore error:${W} ${Q}
434
- `);return process.stderr.write(`${E}Partial rollback: ${z.restored}/${$.restore.length} files restored.${W}
435
- `),1}return process.stdout.write(`${C}Rolled back ${z.restored}/${$.restore.length} state files from ${K}.${W}
421
+ `;var u0=R(()=>{m0();g()});import{closeSync as M8,fstatSync as Y8,lstatSync as O8,mkdirSync as u5,openSync as T8,readSync as A8,renameSync as p5,rmSync as j8,statSync as _8,unlinkSync as I8,writeFileSync as c5,writeSync as P8}from"fs";import{dirname as l5}from"path";function q1(K,$){u5(l5(K),{recursive:!0});let Q=`${K}.tmp.${process.pid}.${++d5}`;c5(Q,`${JSON.stringify($,null,2)}
422
+ `),p5(Q,K)}async function p0(K,$){let Q=_1.get(K)??Promise.resolve(),X=()=>{},Z=new Promise((z)=>{X=z}),W=Q.catch(()=>{}).then(()=>Z);_1.set(K,W);try{return await Q.catch(()=>{}),await $()}finally{if(X(),_1.get(K)===W)_1.delete(K)}}var d5=0,_1;var I1=R(()=>{_1=new Map});import{existsSync as P1,mkdirSync as o5,copyFileSync as n5,readFileSync as a5,readdirSync as s5,statSync as r5,writeFileSync as w8,renameSync as t5,appendFileSync as x8,rmSync as S8}from"fs";import{join as r,dirname as i5}from"path";function L1(K){return r(K,"state","checkpoints")}function KK(K){let $=L1(K);if(!P1($))return[];return s5($).filter((Q)=>Q.startsWith("cp-")).filter((Q)=>{try{return r5(r($,Q)).isDirectory()}catch{return!1}})}function $K(K){return[...K].sort(($,Q)=>{let X=c0($),Z=c0(Q);return X-Z})}function c0(K){let $=K.split("-");if($.length<3)return 0;let Q=$[$.length-1],X=Number.parseInt(Q??"0",10);return Number.isFinite(X)?X:0}function p1(K){let $=K??P(),Q=$K(KK($)),X=[];for(let Z of Q){let W=l0($,Z);if(W)X.push(W)}return X}function l0(K,$){let Q=r(L1(K),$,"metadata.json");if(!P1(Q))return null;try{let X=JSON.parse(a5(Q,"utf-8"));return QK(X,Q)}catch{return null}}function QK(K,$){let Q=zK(K,$);return Q.ok?Q.value:null}function zK(K,$){if(K===null||typeof K!=="object")return console.warn(`[checkpoint] invalid metadata at ${$}: not an object`),{ok:!1,reason:"invalid_type",field:"<root>"};let Q=K,X=["id","timestamp","task_id","task_description","git_sha","git_branch","provider","phase"];for(let Z of X){if(!(Z in Q))return console.warn(`[checkpoint] invalid metadata at ${$}: field "${Z}" missing`),{ok:!1,reason:"missing_field",field:Z};if(typeof Q[Z]!=="string")return console.warn(`[checkpoint] invalid metadata at ${$}: field "${Z}" not a string`),{ok:!1,reason:"invalid_type",field:Z}}if(!Object.prototype.hasOwnProperty.call(Q,"iteration"))return console.warn(`[checkpoint] invalid metadata at ${$}: field "iteration" missing`),{ok:!1,reason:"missing_field",field:"iteration"};if(typeof Q.iteration!=="number"||!Number.isFinite(Q.iteration))return console.warn(`[checkpoint] invalid metadata at ${$}: field "iteration" not a finite number`),{ok:!1,reason:"invalid_type",field:"iteration"};for(let Z of ZK){let W=Q[Z];if(XK.test(W))return console.warn(`[checkpoint] invalid metadata at ${$}: field "${Z}" contains control characters`),{ok:!1,reason:"control_chars",field:Z}}return{ok:!0,value:{id:Q.id,timestamp:Q.timestamp,iteration:Q.iteration,task_id:Q.task_id,task_description:Q.task_description,git_sha:Q.git_sha,git_branch:Q.git_branch,provider:Q.provider,phase:Q.phase}}}function c1(K,$){if(!WK.test(K))throw new d0(K);let Q=$??P(),X=r(L1(Q),K);if(!P1(X))throw new u1(K);let Z=l0(Q,K);if(!Z)throw new u1(K);return Z}function o0(K,$){let Q=c1(K,$),X=$??P(),Z=r(L1(X),K),W=[];for(let z of HK){let q=r(Z,z);if(!P1(q))continue;W.push({from:q,to:r(X,z)})}return{id:K,metadata:Q,restore:W}}function n0(K){let $=[],Q=0;for(let X of K.restore)try{o5(i5(X.to),{recursive:!0});let Z=`${X.to}.tmp.${process.pid}.${++e5}`;n5(X.from,Z),t5(Z,X.to),Q+=1}catch(Z){$.push(`${X.from} -> ${X.to}: ${Z.message}`)}return{restored:Q,errors:$}}var h8,e5=0,XK,ZK,WK,u1,d0,HK;var a0=R(()=>{g();c();I1();h8=Promise.resolve();XK=/[\x00-\x08\x0a-\x1f\x7f-\x9f]/,ZK=["id","task_id","git_sha","git_branch","provider","phase"];WK=/^[a-zA-Z0-9_-]+$/;u1=class u1 extends Error{id;constructor(K){super(`Checkpoint not found: ${K}`);this.id=K;this.name="CheckpointNotFoundError"}};d0=class d0 extends Error{id;constructor(K){super(`Invalid checkpoint ID: must be alphanumeric, hyphens, underscores only (got: ${K})`);this.id=K;this.name="InvalidCheckpointIdError"}};HK=["state/orchestrator.json","queue/pending.json","queue/completed.json","queue/in-progress.json","queue/current-task.json"]});var t0={};v(t0,{runRollback:()=>qK});async function qK(K){let $=K[0],Q=K.slice(1);if($===void 0||$==="help"||$==="--help"||$==="-h")return process.stdout.write(s0),$===void 0?1:0;switch($){case"list":{let X=[...p1()].reverse();if(X.length===0)return process.stdout.write(`${F}No checkpoints found.${H}
423
+ `),0;process.stdout.write(`${D}Checkpoints${H} (${X.length}, newest first):
424
+ `);for(let Z of X)process.stdout.write(` ${O}${Z.id}${H} iter=${Z.iteration} ${Z.git_branch||"(no branch)"}@${(Z.git_sha||"").slice(0,7)} ${Z.timestamp}
425
+ `);return 0}case"show":{let X=Q[0];if(!X)return process.stderr.write(`${E}Missing checkpoint id.${H} Use \`loki rollback list\`.
426
+ `),2;try{let Z=c1(X);return process.stdout.write(`${JSON.stringify(Z,null,2)}
427
+ `),0}catch(Z){return process.stderr.write(`${E}Failed to read checkpoint:${H} ${Z.message}
428
+ `),1}}case"to":{let X=Q[0];if(!X)return process.stderr.write(`${E}Missing checkpoint id.${H} Use \`loki rollback list\`.
429
+ `),2;return r0(X)}case"latest":{let X=p1(),Z=X[X.length-1];if(!Z)return process.stderr.write(`${E}No checkpoints found to roll back to.${H}
430
+ `),1;return process.stdout.write(`Rolling back to latest checkpoint: ${O}${Z.id}${H}
431
+ `),r0(Z.id)}default:return process.stderr.write(`Unknown subcommand: ${$}
432
+ `),process.stderr.write(s0),2}}function r0(K){let $;try{$=o0(K)}catch(X){return process.stderr.write(`${E}Cannot plan rollback:${H} ${X.message}
433
+ `),1}if($.restore.length===0)return process.stdout.write(`${F}Checkpoint ${K} has no restorable state files; nothing to do.${H}
434
+ `),0;let Q=n0($);if(Q.errors.length>0){for(let X of Q.errors)process.stderr.write(`${E}restore error:${H} ${X}
435
+ `);return process.stderr.write(`${E}Partial rollback: ${Q.restored}/${$.restore.length} files restored.${H}
436
+ `),1}return process.stdout.write(`${b}Rolled back ${Q.restored}/${$.restore.length} state files from ${K}.${H}
436
437
  `),process.stdout.write("Run `loki start` to resume from the restored state.\n"),0}var s0=`Usage: loki rollback <subcommand>
437
438
 
438
439
  Subcommands:
@@ -448,8 +449,8 @@ Restored files (matches autonomy/run.sh:7028 byte-for-byte):
448
449
  Note: only state files are restored. Source code, git history, and the
449
450
  session's autonomy-state.json are unchanged. Re-run \`loki start\` to
450
451
  resume from the restored state.
451
- `;var i0=R(()=>{a0();a()});var d1={};b(d1,{renderFindingsForPrompt:()=>GK,loadPreviousFindings:()=>l1,findLatestReviewDir:()=>Q7,_parseReviewerOutputForTests:()=>BK});import{existsSync as K7,readFileSync as e0,readdirSync as $7,statSync as qK}from"fs";import{join as R1}from"path";function JK(K){let $=K.toLowerCase();if($==="critical")return"Critical";if($==="high")return"High";if($==="medium")return"Medium";return"Low"}function z7(K,$,z,Q){let X=[],H=K.split(/\r?\n/);for(let Z of H){let q=Z.trim();if(q.length===0)continue;let U=q.replace(/^[-*]\s*/,""),V=UK.exec(U);if(!V||!V[1]||!V[2])continue;let G=JK(V[1]),B=V[2].trim(),J=VK.exec(B),Y=J&&J[1]?J[1]:null,j=J&&J[2]?Number.parseInt(J[2],10):null;X.push({reviewId:z,iteration:Q,reviewer:$,severity:G,description:B,file:Y,line:Number.isFinite(j)?j:null,raw:q})}return X}function Q7(K,$){let z=R1(K,"quality","reviews");if(!K7(z))return null;let Q;try{Q=$7(z)}catch{return null}let X=$===void 0?Q.filter((q)=>q.startsWith("review-")):Q.filter((q)=>q.endsWith(`-${$}`)&&q.startsWith("review-"));if(X.length===0)return null;X.sort();let H=X[X.length-1];if(!H)return null;let Z=R1(z,H);try{if(!qK(Z).isDirectory())return null}catch{return null}return Z}function l1(K,$){let z=Q7(K,$);if(z===null)return{reviewDir:null,reviewId:null,iteration:null,findings:[]};let Q=null,X=null,H=R1(z,"aggregate.json");if(K7(H))try{let V=e0(H,"utf-8"),G=JSON.parse(V);if(typeof G.review_id==="string")Q=G.review_id;if(typeof G.iteration==="number")X=G.iteration}catch{}let Z;try{Z=$7(z)}catch{return{reviewDir:z,reviewId:Q,iteration:X,findings:[]}}let q=new Set(["diff.txt","files.txt","anti-sycophancy.txt"]),U=[];for(let V of Z){if(!V.endsWith(".txt"))continue;if(q.has(V))continue;if(V.endsWith("-prompt.txt"))continue;let G=V.replace(/\.txt$/,""),B;try{B=e0(R1(z,V),"utf-8")}catch{continue}U.push(...z7(B,G,Q??"",X??-1))}return{reviewDir:z,reviewId:Q,iteration:X,findings:U}}function GK(K){if(K.length===0)return"";let $=["Critical","High","Medium","Low"],z=new Map;for(let X of $)z.set(X,[]);for(let X of K){let H=z.get(X.severity);if(H)H.push(X)}let Q=[];Q.push("PREVIOUS REVIEWER FINDINGS (must address each, or supply counter-evidence in .loki/state/counter-evidence-<iter>.json):");for(let X of $){let H=z.get(X)??[];if(H.length===0)continue;Q.push(` [${X}] (${H.length}):`);for(let Z of H){let q=Z.file?` (${Z.file}${Z.line!==null?":"+Z.line:""})`:"";Q.push(` - ${Z.description}${q} -- via ${Z.reviewer}`)}}return Q.join(`
452
- `)}function BK(K,$,z="review-test",Q=0){return z7(K,$,z,Q)}var UK,VK;var E1=R(()=>{UK=/\[(Critical|High|Medium|Low)\]\s*(.+)/i,VK=/([\w.\-/]+\.[a-zA-Z]+):(\d+)/});import{existsSync as MK}from"fs";import{join as YK}from"path";async function X7(K,$){let z=YK(K,"memory");if(!MK(z))return{stored:!1,reason:"memory dir not initialized"};let Q=Math.max(0,Math.floor($.durationSeconds??0)),X={_LOKI_PROJECT_DIR:p,_LOKI_TARGET_DIR:process.cwd(),_LOKI_TASK_ID:$.taskId,_LOKI_OUTCOME:$.outcome,_LOKI_PHASE:$.phase,_LOKI_GOAL:$.goal,_LOKI_DURATION:String(Q),_LOKI_LOKI_DIR:K},Z=await s(`
452
+ `;var i0=R(()=>{a0();a()});var d1={};v(d1,{renderFindingsForPrompt:()=>BK,loadPreviousFindings:()=>l1,findLatestReviewDir:()=>X7,_parseReviewerOutputForTests:()=>MK});import{existsSync as K7,readFileSync as e0,readdirSync as $7,statSync as UK}from"fs";import{join as R1}from"path";function GK(K){let $=K.toLowerCase();if($==="critical")return"Critical";if($==="high")return"High";if($==="medium")return"Medium";return"Low"}function Q7(K,$,Q,X){let Z=[],W=K.split(/\r?\n/);for(let z of W){let q=z.trim();if(q.length===0)continue;let U=q.replace(/^[-*]\s*/,""),V=VK.exec(U);if(!V||!V[1]||!V[2])continue;let G=GK(V[1]),B=V[2].trim(),J=JK.exec(B),Y=J&&J[1]?J[1]:null,T=J&&J[2]?Number.parseInt(J[2],10):null;Z.push({reviewId:Q,iteration:X,reviewer:$,severity:G,description:B,file:Y,line:Number.isFinite(T)?T:null,raw:q})}return Z}function X7(K,$){let Q=R1(K,"quality","reviews");if(!K7(Q))return null;let X;try{X=$7(Q)}catch{return null}let Z=$===void 0?X.filter((q)=>q.startsWith("review-")):X.filter((q)=>q.endsWith(`-${$}`)&&q.startsWith("review-"));if(Z.length===0)return null;Z.sort();let W=Z[Z.length-1];if(!W)return null;let z=R1(Q,W);try{if(!UK(z).isDirectory())return null}catch{return null}return z}function l1(K,$){let Q=X7(K,$);if(Q===null)return{reviewDir:null,reviewId:null,iteration:null,findings:[]};let X=null,Z=null,W=R1(Q,"aggregate.json");if(K7(W))try{let V=e0(W,"utf-8"),G=JSON.parse(V);if(typeof G.review_id==="string")X=G.review_id;if(typeof G.iteration==="number")Z=G.iteration}catch{}let z;try{z=$7(Q)}catch{return{reviewDir:Q,reviewId:X,iteration:Z,findings:[]}}let q=new Set(["diff.txt","files.txt","anti-sycophancy.txt"]),U=[];for(let V of z){if(!V.endsWith(".txt"))continue;if(q.has(V))continue;if(V.endsWith("-prompt.txt"))continue;let G=V.replace(/\.txt$/,""),B;try{B=e0(R1(Q,V),"utf-8")}catch{continue}U.push(...Q7(B,G,X??"",Z??-1))}return{reviewDir:Q,reviewId:X,iteration:Z,findings:U}}function BK(K){if(K.length===0)return"";let $=["Critical","High","Medium","Low"],Q=new Map;for(let Z of $)Q.set(Z,[]);for(let Z of K){let W=Q.get(Z.severity);if(W)W.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 $){let W=Q.get(Z)??[];if(W.length===0)continue;X.push(` [${Z}] (${W.length}):`);for(let z of W){let q=z.file?` (${z.file}${z.line!==null?":"+z.line:""})`:"";X.push(` - ${z.description}${q} -- via ${z.reviewer}`)}}return X.join(`
453
+ `)}function MK(K,$,Q="review-test",X=0){return Q7(K,$,Q,X)}var VK,JK;var E1=R(()=>{VK=/\[(Critical|High|Medium|Low)\]\s*(.+)/i,JK=/([\w.\-/]+\.[a-zA-Z]+):(\d+)/});import{existsSync as YK}from"fs";import{join as OK}from"path";async function Z7(K,$){let Q=OK(K,"memory");if(!YK(Q))return{stored:!1,reason:"memory dir not initialized"};let X=Math.max(0,Math.floor($.durationSeconds??0)),Z={_LOKI_PROJECT_DIR:p,_LOKI_TARGET_DIR:process.cwd(),_LOKI_TASK_ID:$.taskId,_LOKI_OUTCOME:$.outcome,_LOKI_PHASE:$.phase,_LOKI_GOAL:$.goal,_LOKI_DURATION:String(X),_LOKI_LOKI_DIR:K},z=await s(`
453
454
  import os, sys
454
455
  project = os.environ.get('_LOKI_PROJECT_DIR', '')
455
456
  loki = os.environ.get('_LOKI_LOKI_DIR', '.loki')
@@ -476,22 +477,22 @@ try:
476
477
  print('OK')
477
478
  except Exception as e:
478
479
  print('ERR:' + str(e))
479
- `,{env:X,timeoutMs:15000});if(Z.exitCode===127)return{stored:!1,reason:"python3 not found"};let q=Z.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:Z.stderr.trim()||"unknown"}}var Z7=R(()=>{Q1();y()});var U7={};b(U7,{loadLearnings:()=>o1,appendLearning:()=>q1,appendFromGateFailure:()=>LK});import{existsSync as OK,readFileSync as TK}from"fs";import{join as H7}from"path";import{createHash as AK}from"crypto";function W7(K){return H7(K,jK)}function _K(K){if(K===null||typeof K!=="object")return!1;let $=K;return typeof $.id==="string"&&typeof $.timestamp==="string"&&typeof $.iteration==="number"&&typeof $.trigger==="string"&&typeof $.rootCause==="string"&&typeof $.fix==="string"&&typeof $.preventInFuture==="string"&&typeof $.evidence==="object"&&$.evidence!==null}function q7(K){if(!OK(K))return{version:1,learnings:[]};try{let $=TK(K,"utf-8"),z=JSON.parse($);if(z.version===1&&Array.isArray(z.learnings))return{version:1,learnings:z.learnings.filter(_K)}}catch{}return{version:1,learnings:[]}}function IK(K,$){return AK("sha256").update(`${K}\x00${$}`).digest("hex").slice(0,16)}async function q1(K,$,z={}){let Q=IK($.trigger,$.rootCause),X=new Date().toISOString(),H={id:Q,timestamp:X,...$},Z=W7(K);if(await p0(Z,()=>{let U=q7(Z),V=U.learnings.findIndex((G)=>G.id===Q);if(V>=0){let G=U.learnings[V];U.learnings[V]={...G,timestamp:X,iteration:H.iteration}}else U.learnings.push(H);W1(Z,U)}),z.episodeBridge!==null&&(z.episodeBridge!==void 0||process.env.LOKI_AUTO_LEARNINGS_EPISODE==="1")){let U=z.episodeBridge??X7,V=z.bridgeFailureLog??PK;try{let G=await U(K,{taskId:`learning-${Q}`,outcome:"failure",phase:"VERIFY",goal:`${$.trigger}: ${$.rootCause}`});if(G&&!G.stored){if(!new Set(["memory dir not initialized","stub"]).has(G.reason))V(`episode_bridge skipped: ${G.reason}`)}}catch(G){V(`episode_bridge threw: ${G.message}`)}}return H}function PK(K){process.stderr.write(`[learnings_writer] ${K}
480
- `)}async function LK(K,$,z,Q={}){let X=`[${z.severity}] ${z.description}`;return q1(K,{iteration:$,trigger:"gate_failure",rootCause:X,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}},Q)}function o1(K){return q7(W7(K))}var jK;var x1=R(()=>{I1();Z7();jK=H7("state","relevant-learnings.json")});var J7={};b(J7,{runOverrideCouncil:()=>SK,recordOverrideOutcome:()=>NK,loadCounterEvidence:()=>wK,canonicalFindingId:()=>n1,DEFAULT_OVERRIDE_JUDGES:()=>V7});import{existsSync as RK,readFileSync as EK}from"fs";import{join as xK}from"path";function wK(K,$){let z=xK(K,"state",`counter-evidence-${$}.json`);if(!RK(z))return null;try{let Q=EK(z,"utf-8"),X=JSON.parse(Q);if(typeof X.iteration!=="number")return null;let H=Array.isArray(X.evidence)?X.evidence:[],Z=[];for(let q of H){if(typeof q!=="object"||q===null)continue;let U=q;if(typeof U.findingId!=="string")continue;if(typeof U.claim!=="string")continue;let V=U.proofType;if(typeof V!=="string"||!FK.has(V))continue;let G=V,B=Array.isArray(U.artifacts)?U.artifacts:[];Z.push({findingId:U.findingId,claim:U.claim,proofType:G,artifacts:B.filter((J)=>typeof J==="string")})}return{iteration:X.iteration,evidence:Z}}catch{return null}}async function SK(K,$,z,Q={}){let X=Q.judges??V7,H=new Set,Z=new Set,q={},U=new Map;for(let V of $.evidence)U.set(V.findingId,V);for(let V of K){let G=n1(V),B=U.get(G);if(!B){Z.add(G);continue}let J=await Promise.all(X.map((j)=>z({finding:V,evidence:B,judge:j})));if(q[G]=J,J.filter((j)=>j.verdict==="APPROVE_OVERRIDE").length>=2)H.add(G);else Z.add(G)}return{approvedFindingIds:H,rejectedFindingIds:Z,votes:q}}function n1(K){let $=K.raw.slice(0,80).replace(/\s+/g," ").trim();return`${K.reviewer}::${$}`}async function NK(K,$,z,Q,X={}){let H={episodeBridge:X.episodeBridge===void 0?null:X.episodeBridge};for(let Z of Q){let q=n1(Z);if(z.approvedFindingIds.has(q))await q1(K,{iteration:$,trigger:"override_approved",rootCause:`[${Z.severity}] ${Z.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:Z.reviewId,file:Z.file??void 0,line:Z.line??void 0,severity:Z.severity,reviewer:Z.reviewer}},H);else if(z.rejectedFindingIds.has(q))await q1(K,{iteration:$,trigger:"override_rejected",rootCause:`[${Z.severity}] ${Z.description}`,fix:"override council rejected -- dev agent must fix the finding",preventInFuture:"address this finding in the next iteration",evidence:{findingId:q,reviewId:Z.reviewId,file:Z.file??void 0,line:Z.line??void 0,severity:Z.severity,reviewer:Z.reviewer}},H)}}var FK,V7;var G7=R(()=>{x1();FK=new Set(["file-exists","test-passes","grep-miss","reviewer-misread","duplicate-code-path","out-of-scope"]);V7=["judge-primary","judge-secondary","judge-tertiary"]});var Y7={};b(Y7,{writeEscalationHandoff:()=>pK,renderHandoff:()=>B7,readLatestHandoff:()=>cK});import{existsSync as kK,mkdirSync as DK,readdirSync as CK,readFileSync as hK,renameSync as bK,writeFileSync as yK}from"fs";import{dirname as vK,join as F1}from"path";function gK(){return new Date().toISOString()}function mK(K){let $=K.file?` (${K.file}${K.line!==null?":"+K.line:""})`:"";return` - [${K.severity}] ${K.description}${$} -- ${K.reviewer}`}function fK(K){let $=K.evidence,z=$.file?` ${$.file}${$.line!==void 0?":"+$.line:""}`:"";return` - **${K.trigger}** (iter ${K.iteration})${z}: ${K.rootCause}`}function B7(K,$,z){let Q=[];if(Q.push(`# Loki escalation handoff -- ${gK()}`),Q.push(""),Q.push(`Gate **${K.gateName}** has failed ${K.consecutiveFailures} consecutive times at iteration ${K.iteration}.`),Q.push(""),Q.push(`Reason: ${K.detail}`),Q.push(""),$.length>0){Q.push(`## Outstanding findings (${$.length})`),Q.push("");for(let X of $)Q.push(mK(X));Q.push("")}else Q.push("## Outstanding findings"),Q.push(""),Q.push("(no per-finding records captured -- gate failed without populating reviewer outputs)"),Q.push("");if(z.length>0){Q.push(`## Recent learnings (${Math.min(z.length,10)})`),Q.push("");for(let X of z.slice(-10))Q.push(fK(X));Q.push("")}return Q.push("## What the human must decide"),Q.push(""),Q.push("- Approve override? Write `.loki/state/counter-evidence-<iter>.json` with one entry per finding to dispute, then `rm .loki/PAUSE` to resume."),Q.push("- Disable a gate? Set `LOKI_GATE_<NAME>=false` in env (see skills/quality-gates.md)."),Q.push("- Tweak escalation? Set `LOKI_GATE_PAUSE_LIMIT` or `LOKI_GATE_ESCALATE_LIMIT`."),Q.push("- Roll back? Switch to `LOKI_LEGACY_BASH=1` and re-run; the bash route does not consult this handoff doc."),Q.push(""),Q.push("To resume: address the findings (or supply counter-evidence) and `rm .loki/PAUSE`."),Q.join(`
481
- `)}function uK(K,$){DK(vK(K),{recursive:!0});let z=`${K}.tmp.${process.pid}.${++M7}`;yK(z,$),bK(z,K)}function pK(K,$,z={}){let Q=z.findings??l1(K,$.iteration).findings,X=z.learnings??o1(K).learnings,H=B7($,Q,X),Z=(z.now?.()??new Date).toISOString().replace(/[-:.]/g,""),q=F1(K,"escalations"),U=++M7,V=F1(q,`handoff-${Z}-${process.pid}-${U}-${$.gateName}.md`);return uK(V,H),{path:V,bytes:H.length}}function cK(K){let $=F1(K,"escalations");if(!kK($))return null;let z;try{z=CK($).filter((H)=>H.endsWith(".md"))}catch{return null}if(z.length===0)return null;z.sort();let Q=z[z.length-1];if(!Q)return null;let X=F1($,Q);try{return{path:X,body:hK(X,"utf-8")}}catch{return null}}var M7=0;var O7=R(()=>{E1();x1()});var T7={};b(T7,{runInternalPhase1Hooks:()=>sK,_resolveForTests:()=>aK,_internalPhase1HooksHelp:()=>eK});import{existsSync as lK,mkdirSync as dK,readdirSync as oK,statSync as nK}from"fs";import{join as U1,resolve as aK}from"path";async function sK(K){let[$,...z]=K;switch($){case void 0:case"help":case"--help":case"-h":return process.stdout.write(a1),$===void 0?1:0;case"reflect":return rK(z);case"override":return tK(z);case"handoff":return iK(z);default:return process.stderr.write(`Unknown subcommand: ${$}
482
- `),process.stderr.write(a1),2}}async function rK(K){let $=s1(K[0]);if($===null)return process.stderr.write(`reflect: missing or invalid <iter>
483
- `),2;let z=P();try{let X=(await Promise.resolve().then(() => (E1(),d1))).loadPreviousFindings(z,$);if(X.findings.length===0)return process.stdout.write(`reflect: no findings for iter ${$} (nothing to do)
484
- `),0;let H=U1(z,"state");dK(H,{recursive:!0}),W1(U1(H,`findings-${$}.json`),{review_id:X.reviewId,iteration:$,findings:X.findings});let Z=await Promise.resolve().then(() => (x1(),U7)),q=0;if(process.env.LOKI_AUTO_LEARNINGS!=="0"){for(let U of X.findings)if(U.severity==="Critical"||U.severity==="High")await Z.appendFromGateFailure(z,$,U,{episodeBridge:null}),q+=1}return process.stdout.write(`reflect: persisted ${X.findings.length} findings + ${q} learnings (iter ${$})
485
- `),0}catch(Q){return process.stderr.write(`reflect: ${Q.message}
486
- `),1}}async function tK(K){let $=s1(K[0]);if($===null)return process.stderr.write(`override: missing or invalid <iter>
487
- `),2;let z=P();try{let Q=await Promise.resolve().then(() => (G7(),J7)),X=Q.loadCounterEvidence(z,$);if(X===null||X.evidence.length===0)return process.stdout.write(`override: no counter-evidence for iter ${$} (skip)
488
- `),0;let Z=(await Promise.resolve().then(() => (E1(),d1))).loadPreviousFindings(z,$),q=Z.findings.filter((_)=>_.severity==="Critical"||_.severity==="High");if(q.length===0)return process.stdout.write(`override: no blocking findings for iter ${$} (skip)
489
- `),0;let U=new Set(["duplicate-code-path","file-exists","test-passes","grep-miss","out-of-scope"]),V=async(_)=>{let h=U.has(_.evidence.proofType);return{judge:_.judge,verdict:h?"APPROVE_OVERRIDE":"REJECT_OVERRIDE",reasoning:h?`[stub] proofType=${_.evidence.proofType} trusted`:`[stub] proofType=${_.evidence.proofType} requires manual review`}},G=await Q.runOverrideCouncil(q,X,V);await Q.recordOverrideOutcome(z,$,G,q);let B=U1(z,"quality","reviews");if(lK(B))try{let _=oK(B).filter((g)=>g.startsWith("review-")).sort(),h=_[_.length-1];if(h&&nK(U1(B,h)).isDirectory())W1(U1(B,h,`override-${$}.json`),{review_id:Z.reviewId,iteration:$,approved_finding_ids:Array.from(G.approvedFindingIds),rejected_finding_ids:Array.from(G.rejectedFindingIds),votes:G.votes})}catch{}let J=G.approvedFindingIds.size,Y=G.rejectedFindingIds.size;if(Y===0&&J>0)process.stdout.write(`override: LIFTED -- ${J} approved, ${Y} rejected
480
+ `,{env:Z,timeoutMs:15000});if(z.exitCode===127)return{stored:!1,reason:"python3 not found"};let q=z.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:z.stderr.trim()||"unknown"}}var z7=R(()=>{Z1();g()});var U7={};v(U7,{loadLearnings:()=>o1,appendLearning:()=>U1,appendFromGateFailure:()=>RK});import{existsSync as TK,readFileSync as AK}from"fs";import{join as W7}from"path";import{createHash as jK}from"crypto";function H7(K){return W7(K,_K)}function IK(K){if(K===null||typeof K!=="object")return!1;let $=K;return typeof $.id==="string"&&typeof $.timestamp==="string"&&typeof $.iteration==="number"&&typeof $.trigger==="string"&&typeof $.rootCause==="string"&&typeof $.fix==="string"&&typeof $.preventInFuture==="string"&&typeof $.evidence==="object"&&$.evidence!==null}function q7(K){if(!TK(K))return{version:1,learnings:[]};try{let $=AK(K,"utf-8"),Q=JSON.parse($);if(Q.version===1&&Array.isArray(Q.learnings))return{version:1,learnings:Q.learnings.filter(IK)}}catch{}return{version:1,learnings:[]}}function PK(K,$){return jK("sha256").update(`${K}\x00${$}`).digest("hex").slice(0,16)}async function U1(K,$,Q={}){let X=PK($.trigger,$.rootCause),Z=new Date().toISOString(),W={id:X,timestamp:Z,...$},z=H7(K);if(await p0(z,()=>{let U=q7(z),V=U.learnings.findIndex((G)=>G.id===X);if(V>=0){let G=U.learnings[V];U.learnings[V]={...G,timestamp:Z,iteration:W.iteration}}else U.learnings.push(W);q1(z,U)}),Q.episodeBridge!==null&&(Q.episodeBridge!==void 0||process.env.LOKI_AUTO_LEARNINGS_EPISODE==="1")){let U=Q.episodeBridge??Z7,V=Q.bridgeFailureLog??LK;try{let G=await U(K,{taskId:`learning-${X}`,outcome:"failure",phase:"VERIFY",goal:`${$.trigger}: ${$.rootCause}`});if(G&&!G.stored){if(!new Set(["memory dir not initialized","stub"]).has(G.reason))V(`episode_bridge skipped: ${G.reason}`)}}catch(G){V(`episode_bridge threw: ${G.message}`)}}return W}function LK(K){process.stderr.write(`[learnings_writer] ${K}
481
+ `)}async function RK(K,$,Q,X={}){let Z=`[${Q.severity}] ${Q.description}`;return U1(K,{iteration:$,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 o1(K){return q7(H7(K))}var _K;var F1=R(()=>{I1();z7();_K=W7("state","relevant-learnings.json")});var J7={};v(J7,{runOverrideCouncil:()=>NK,recordOverrideOutcome:()=>kK,loadCounterEvidence:()=>SK,canonicalFindingId:()=>n1,DEFAULT_OVERRIDE_JUDGES:()=>V7});import{existsSync as EK,readFileSync as FK}from"fs";import{join as wK}from"path";function SK(K,$){let Q=wK(K,"state",`counter-evidence-${$}.json`);if(!EK(Q))return null;try{let X=FK(Q,"utf-8"),Z=JSON.parse(X);if(typeof Z.iteration!=="number")return null;let W=Array.isArray(Z.evidence)?Z.evidence:[],z=[];for(let q of W){if(typeof q!=="object"||q===null)continue;let U=q;if(typeof U.findingId!=="string")continue;if(typeof U.claim!=="string")continue;let V=U.proofType;if(typeof V!=="string"||!xK.has(V))continue;let G=V,B=Array.isArray(U.artifacts)?U.artifacts:[];z.push({findingId:U.findingId,claim:U.claim,proofType:G,artifacts:B.filter((J)=>typeof J==="string")})}return{iteration:Z.iteration,evidence:z}}catch{return null}}async function NK(K,$,Q,X={}){let Z=X.judges??V7,W=new Set,z=new Set,q={},U=new Map;for(let V of $.evidence)U.set(V.findingId,V);for(let V of K){let G=n1(V),B=U.get(G);if(!B){z.add(G);continue}let J=await Promise.all(Z.map((T)=>Q({finding:V,evidence:B,judge:T})));if(q[G]=J,J.filter((T)=>T.verdict==="APPROVE_OVERRIDE").length>=2)W.add(G);else z.add(G)}return{approvedFindingIds:W,rejectedFindingIds:z,votes:q}}function n1(K){let $=K.raw.slice(0,80).replace(/\s+/g," ").trim();return`${K.reviewer}::${$}`}async function kK(K,$,Q,X,Z={}){let W={episodeBridge:Z.episodeBridge===void 0?null:Z.episodeBridge};for(let z of X){let q=n1(z);if(Q.approvedFindingIds.has(q))await U1(K,{iteration:$,trigger:"override_approved",rootCause:`[${z.severity}] ${z.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:z.reviewId,file:z.file??void 0,line:z.line??void 0,severity:z.severity,reviewer:z.reviewer}},W);else if(Q.rejectedFindingIds.has(q))await U1(K,{iteration:$,trigger:"override_rejected",rootCause:`[${z.severity}] ${z.description}`,fix:"override council rejected -- dev agent must fix the finding",preventInFuture:"address this finding in the next iteration",evidence:{findingId:q,reviewId:z.reviewId,file:z.file??void 0,line:z.line??void 0,severity:z.severity,reviewer:z.reviewer}},W)}}var xK,V7;var G7=R(()=>{F1();xK=new Set(["file-exists","test-passes","grep-miss","reviewer-misread","duplicate-code-path","out-of-scope"]);V7=["judge-primary","judge-secondary","judge-tertiary"]});var Y7={};v(Y7,{writeEscalationHandoff:()=>cK,renderHandoff:()=>B7,readLatestHandoff:()=>lK});import{existsSync as DK,mkdirSync as CK,readdirSync as hK,readFileSync as bK,renameSync as yK,writeFileSync as vK}from"fs";import{dirname as gK,join as w1}from"path";function mK(){return new Date().toISOString()}function fK(K){let $=K.file?` (${K.file}${K.line!==null?":"+K.line:""})`:"";return` - [${K.severity}] ${K.description}${$} -- ${K.reviewer}`}function uK(K){let $=K.evidence,Q=$.file?` ${$.file}${$.line!==void 0?":"+$.line:""}`:"";return` - **${K.trigger}** (iter ${K.iteration})${Q}: ${K.rootCause}`}function B7(K,$,Q){let X=[];if(X.push(`# Loki escalation handoff -- ${mK()}`),X.push(""),X.push(`Gate **${K.gateName}** has failed ${K.consecutiveFailures} consecutive times at iteration ${K.iteration}.`),X.push(""),X.push(`Reason: ${K.detail}`),X.push(""),$.length>0){X.push(`## Outstanding findings (${$.length})`),X.push("");for(let Z of $)X.push(fK(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(uK(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(`
482
+ `)}function pK(K,$){CK(gK(K),{recursive:!0});let Q=`${K}.tmp.${process.pid}.${++M7}`;vK(Q,$),yK(Q,K)}function cK(K,$,Q={}){let X=Q.findings??l1(K,$.iteration).findings,Z=Q.learnings??o1(K).learnings,W=B7($,X,Z),z=(Q.now?.()??new Date).toISOString().replace(/[-:.]/g,""),q=w1(K,"escalations"),U=++M7,V=w1(q,`handoff-${z}-${process.pid}-${U}-${$.gateName}.md`);return pK(V,W),{path:V,bytes:W.length}}function lK(K){let $=w1(K,"escalations");if(!DK($))return null;let Q;try{Q=hK($).filter((W)=>W.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=w1($,X);try{return{path:Z,body:bK(Z,"utf-8")}}catch{return null}}var M7=0;var O7=R(()=>{E1();F1()});var T7={};v(T7,{runInternalPhase1Hooks:()=>rK,_resolveForTests:()=>sK,_internalPhase1HooksHelp:()=>K6});import{existsSync as dK,mkdirSync as oK,readdirSync as nK,statSync as aK}from"fs";import{join as V1,resolve as sK}from"path";async function rK(K){let[$,...Q]=K;switch($){case void 0:case"help":case"--help":case"-h":return process.stdout.write(a1),$===void 0?1:0;case"reflect":return tK(Q);case"override":return iK(Q);case"handoff":return eK(Q);default:return process.stderr.write(`Unknown subcommand: ${$}
483
+ `),process.stderr.write(a1),2}}async function tK(K){let $=s1(K[0]);if($===null)return process.stderr.write(`reflect: missing or invalid <iter>
484
+ `),2;let Q=P();try{let Z=(await Promise.resolve().then(() => (E1(),d1))).loadPreviousFindings(Q,$);if(Z.findings.length===0)return process.stdout.write(`reflect: no findings for iter ${$} (nothing to do)
485
+ `),0;let W=V1(Q,"state");oK(W,{recursive:!0}),q1(V1(W,`findings-${$}.json`),{review_id:Z.reviewId,iteration:$,findings:Z.findings});let z=await Promise.resolve().then(() => (F1(),U7)),q=0;if(process.env.LOKI_AUTO_LEARNINGS!=="0"){for(let U of Z.findings)if(U.severity==="Critical"||U.severity==="High")await z.appendFromGateFailure(Q,$,U,{episodeBridge:null}),q+=1}return process.stdout.write(`reflect: persisted ${Z.findings.length} findings + ${q} learnings (iter ${$})
486
+ `),0}catch(X){return process.stderr.write(`reflect: ${X.message}
487
+ `),1}}async function iK(K){let $=s1(K[0]);if($===null)return process.stderr.write(`override: missing or invalid <iter>
488
+ `),2;let Q=P();try{let X=await Promise.resolve().then(() => (G7(),J7)),Z=X.loadCounterEvidence(Q,$);if(Z===null||Z.evidence.length===0)return process.stdout.write(`override: no counter-evidence for iter ${$} (skip)
489
+ `),0;let z=(await Promise.resolve().then(() => (E1(),d1))).loadPreviousFindings(Q,$),q=z.findings.filter((j)=>j.severity==="Critical"||j.severity==="High");if(q.length===0)return process.stdout.write(`override: no blocking findings for iter ${$} (skip)
490
+ `),0;let U=new Set(["duplicate-code-path","file-exists","test-passes","grep-miss","out-of-scope"]),V=async(j)=>{let C=U.has(j.evidence.proofType);return{judge:j.judge,verdict:C?"APPROVE_OVERRIDE":"REJECT_OVERRIDE",reasoning:C?`[stub] proofType=${j.evidence.proofType} trusted`:`[stub] proofType=${j.evidence.proofType} requires manual review`}},G=await X.runOverrideCouncil(q,Z,V);await X.recordOverrideOutcome(Q,$,G,q);let B=V1(Q,"quality","reviews");if(dK(B))try{let j=nK(B).filter((N)=>N.startsWith("review-")).sort(),C=j[j.length-1];if(C&&aK(V1(B,C)).isDirectory())q1(V1(B,C,`override-${$}.json`),{review_id:z.reviewId,iteration:$,approved_finding_ids:Array.from(G.approvedFindingIds),rejected_finding_ids:Array.from(G.rejectedFindingIds),votes:G.votes})}catch{}let J=G.approvedFindingIds.size,Y=G.rejectedFindingIds.size;if(Y===0&&J>0)process.stdout.write(`override: LIFTED -- ${J} approved, ${Y} rejected
490
491
  `);else process.stdout.write(`override: BLOCKED -- ${J} approved, ${Y} rejected
491
- `);return 0}catch(Q){return process.stderr.write(`override: ${Q.message}
492
- `),1}}async function iK(K){let $=K[0],z=Number.parseInt(K[1]??"0",10),Q=s1(K[2]);if(!$||!Number.isFinite(z)||Q===null)return process.stderr.write(`handoff: usage: handoff <gate> <consecutive-failures> <iter>
493
- `),2;let X=P();try{let Z=(await Promise.resolve().then(() => (O7(),Y7))).writeEscalationHandoff(X,{gateName:$,iteration:Q,consecutiveFailures:z,detail:`${$} hit PAUSE_LIMIT (${z} consecutive failures)`});return process.stdout.write(`handoff: wrote ${Z.path} (${Z.bytes}B)
494
- `),0}catch(H){return process.stderr.write(`handoff: ${H.message}
492
+ `);return 0}catch(X){return process.stderr.write(`override: ${X.message}
493
+ `),1}}async function eK(K){let $=K[0],Q=Number.parseInt(K[1]??"0",10),X=s1(K[2]);if(!$||!Number.isFinite(Q)||X===null)return process.stderr.write(`handoff: usage: handoff <gate> <consecutive-failures> <iter>
494
+ `),2;let Z=P();try{let z=(await Promise.resolve().then(() => (O7(),Y7))).writeEscalationHandoff(Z,{gateName:$,iteration:X,consecutiveFailures:Q,detail:`${$} hit PAUSE_LIMIT (${Q} consecutive failures)`});return process.stdout.write(`handoff: wrote ${z.path} (${z.bytes}B)
495
+ `),0}catch(W){return process.stderr.write(`handoff: ${W.message}
495
496
  `),1}}function s1(K){if(K===void 0)return null;let $=Number.parseInt(K,10);return Number.isFinite($)&&$>=0?$:null}var a1=`loki internal phase1-hooks <subcommand>
496
497
 
497
498
  Subcommands:
@@ -501,24 +502,24 @@ Subcommands:
501
502
 
502
503
  This command is invoked by autonomy/run.sh between iterations. Users
503
504
  should not run it directly -- run \`loki start\` instead.
504
- `,eK;var A7=R(()=>{y();I1();eK=a1});D1();function K0(){return process.stdout.write(`Loki Mode v${G1()}
505
- `),0}c();a();y();import{readFileSync as h7,existsSync as b7}from"fs";import{resolve as y7}from"path";var v7=["claude","codex","cline","aider"];function z0(){let K=y7(P(),"state","provider");if(!b7(K))return"";try{return h7(K,"utf-8").trim()}catch{return""}}function g7(K,$){return K||$||process.env.LOKI_PROVIDER||"claude"}function m7(K){let $=z0(),z=g7(K,$);switch(process.stdout.write(`${k}Current Provider${W}
505
+ `,K6;var A7=R(()=>{g();I1();K6=a1});D1();function K0(){return process.stdout.write(`Loki Mode v${G1()}
506
+ `),0}c();a();g();import{readFileSync as h7,existsSync as b7}from"fs";import{resolve as y7}from"path";var v7=["claude","codex","cline","aider"];function Q0(){let K=y7(P(),"state","provider");if(!b7(K))return"";try{return h7(K,"utf-8").trim()}catch{return""}}function g7(K,$){return K||$||process.env.LOKI_PROVIDER||"claude"}function m7(K){let $=Q0(),Q=g7(K,$);switch(process.stdout.write(`${D}Current Provider${H}
506
507
  `),process.stdout.write(`
507
- `),process.stdout.write(`${O}Provider:${W} ${z}
508
- `),z){case"claude":process.stdout.write(`${C}Status:${W} Full features (subagents, parallel, MCP)
509
- `);break;case"cline":process.stdout.write(`${C}Status:${W} Near-full mode (subagents, MCP, 12+ providers)
510
- `);break;case"codex":case"aider":process.stdout.write(`${x}Status:${W} Degraded mode (sequential only)
511
- `);break;default:break}if($)process.stdout.write(`${F}(saved in .loki/state/provider)${W}
512
- `);else process.stdout.write(`${F}(default - not explicitly set)${W}
508
+ `),process.stdout.write(`${O}Provider:${H} ${Q}
509
+ `),Q){case"claude":process.stdout.write(`${b}Status:${H} Full features (subagents, parallel, MCP)
510
+ `);break;case"cline":process.stdout.write(`${b}Status:${H} Near-full mode (subagents, MCP, 12+ providers)
511
+ `);break;case"codex":case"aider":process.stdout.write(`${F}Status:${H} Degraded mode (sequential only)
512
+ `);break;default:break}if($)process.stdout.write(`${w}(saved in .loki/state/provider)${H}
513
+ `);else process.stdout.write(`${w}(default - not explicitly set)${H}
513
514
  `);return process.stdout.write(`
514
- `),process.stdout.write(`Switch provider: ${O}loki provider set <name>${W}
515
- `),process.stdout.write(`Available: ${O}loki provider list${W}
516
- `),0}async function f7(){let $=z0()||process.env.LOKI_PROVIDER||"claude";process.stdout.write(`${k}Available Providers${W}
515
+ `),process.stdout.write(`Switch provider: ${O}loki provider set <name>${H}
516
+ `),process.stdout.write(`Available: ${O}loki provider list${H}
517
+ `),0}async function f7(){let $=Q0()||process.env.LOKI_PROVIDER||"claude";process.stdout.write(`${D}Available Providers${H}
517
518
  `),process.stdout.write(`
518
- `);let z=await Promise.all(v7.map(async(H)=>[H,await D(H)!==null])),Q=new Map;for(let[H,Z]of z)Q.set(H,Z?`${C}installed${W}`:`${E}not installed${W}`);let X=[["claude","claude - Claude Code (Anthropic) "],["codex","codex - Codex CLI (OpenAI) "],["cline","cline - Cline (multi-provider) "],["aider","aider - Aider (terminal pair prog) "]];for(let[H,Z]of X){let q=$===H?` ${O}(current)${W}`:"";process.stdout.write(` ${Z} ${Q.get(H)}${q}
519
+ `);let Q=await Promise.all(v7.map(async(W)=>[W,await h(W)!==null])),X=new Map;for(let[W,z]of Q)X.set(W,z?`${b}installed${H}`:`${E}not installed${H}`);let Z=[["claude","claude - Claude Code (Anthropic) "],["codex","codex - Codex CLI (OpenAI) "],["cline","cline - Cline (multi-provider) "],["aider","aider - Aider (terminal pair prog) "]];for(let[W,z]of Z){let q=$===W?` ${O}(current)${H}`:"";process.stdout.write(` ${z} ${X.get(W)}${q}
519
520
  `)}return process.stdout.write(`
520
- `),process.stdout.write(`Set provider: ${O}loki provider set <name>${W}
521
- `),0}function u7(){return process.stdout.write(`${k}Loki Mode Provider Management${W}
521
+ `),process.stdout.write(`Set provider: ${O}loki provider set <name>${H}
522
+ `),0}function u7(){return process.stdout.write(`${D}Loki Mode Provider Management${H}
522
523
  `),process.stdout.write(`
523
524
  `),process.stdout.write(`Usage: loki provider <command>
524
525
  `),process.stdout.write(`
@@ -536,17 +537,17 @@ should not run it directly -- run \`loki start\` instead.
536
537
  `),process.stdout.write(` loki provider list
537
538
  `),process.stdout.write(` loki provider info codex
538
539
  `),process.stdout.write(` loki provider models
539
- `),0}async function Q0(K){let $=K[0]??"show",z=K.slice(1);switch($){case"show":case"current":return m7(z[0]);case"list":return f7();case"set":case"info":case"models":return p7(["provider",$,...z]);default:return u7()}}async function p7(K){let{run:$}=await Promise.resolve().then(() => (c(),$0)),{resolve:z}=await import("path"),{REPO_ROOT:Q}=await Promise.resolve().then(() => (y(),e1)),X=z(Q,"autonomy","loki"),H=await $([X,...K],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(H.stdout),process.stderr.write(H.stderr),H.exitCode}a();y();Q1();c();import{existsSync as X0,readFileSync as l7}from"fs";import{resolve as i}from"path";import{mkdir as d7}from"fs/promises";var X1=i(k1(),"learnings");function h1(K){if(!X0(K))return 0;try{let $=l7(K,"utf-8"),z=0;for(let Q of $.split(`
540
- `))if(Q.includes('"description"'))z++;return z}catch{return 0}}async function o7(){await d7(X1,{recursive:!0});let K=h1(i(X1,"patterns.jsonl")),$=h1(i(X1,"mistakes.jsonl")),z=h1(i(X1,"successes.jsonl"));return process.stdout.write(`${k}Cross-Project Learnings${W}
540
+ `),0}async function X0(K){let $=K[0]??"show",Q=K.slice(1);switch($){case"show":case"current":return m7(Q[0]);case"list":return f7();case"set":case"info":case"models":return p7(["provider",$,...Q]);default:return u7()}}async function p7(K){let{run:$}=await Promise.resolve().then(() => (c(),$0)),{resolve:Q}=await import("path"),{REPO_ROOT:X}=await Promise.resolve().then(() => (g(),e1)),Z=Q(X,"autonomy","loki"),W=await $([Z,...K],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(W.stdout),process.stderr.write(W.stderr),W.exitCode}a();g();Z1();c();import{existsSync as Z0,readFileSync as l7}from"fs";import{resolve as e}from"path";import{mkdir as d7}from"fs/promises";var z1=e(k1(),"learnings");function h1(K){if(!Z0(K))return 0;try{let $=l7(K,"utf-8"),Q=0;for(let X of $.split(`
541
+ `))if(X.includes('"description"'))Q++;return Q}catch{return 0}}async function o7(){await d7(z1,{recursive:!0});let K=h1(e(z1,"patterns.jsonl")),$=h1(e(z1,"mistakes.jsonl")),Q=h1(e(z1,"successes.jsonl"));return process.stdout.write(`${D}Cross-Project Learnings${H}
541
542
  `),process.stdout.write(`
542
- `),process.stdout.write(` Patterns: ${C}${K}${W}
543
- `),process.stdout.write(` Mistakes: ${x}${$}${W}
544
- `),process.stdout.write(` Successes: ${O}${z}${W}
543
+ `),process.stdout.write(` Patterns: ${b}${K}${H}
544
+ `),process.stdout.write(` Mistakes: ${F}${$}${H}
545
+ `),process.stdout.write(` Successes: ${O}${Q}${H}
545
546
  `),process.stdout.write(`
546
- `),process.stdout.write(`Location: ${X1}
547
+ `),process.stdout.write(`Location: ${z1}
547
548
  `),process.stdout.write(`
548
549
  `),process.stdout.write(`Use 'loki memory show <type>' to view entries
549
- `),0}async function n7(K){if(K){let Q=`
550
+ `),0}async function n7(K){if(K){let X=`
550
551
  try:
551
552
  from memory.layers import IndexLayer
552
553
  layer = IndexLayer('.loki/memory')
@@ -556,9 +557,9 @@ except ImportError:
556
557
  print('Error: memory.layers module not found')
557
558
  except Exception as e:
558
559
  print(f'Error: {e}')
559
- `.trim(),X=await s(Q,{cwd:p});return process.stdout.write(X.stdout),0}let $=i(P(),"memory","index.json");if(!X0($))return process.stdout.write(`No index found
560
- `),0;let z=await s(`import json, sys; sys.stdout.write(json.dumps(json.load(open(${JSON.stringify($)})), indent=4) + "\\n")`);if(z.exitCode!==0)return process.stdout.write(`No index found
561
- `),0;return process.stdout.write(z.stdout),0}async function Z0(K){switch(K[0]??"list"){case"list":case"ls":return o7();case"index":return n7(K[1]==="rebuild");default:{let z=i(p,"autonomy","loki"),Q=await S([z,"memory",...K],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(Q.stdout),process.stderr.write(Q.stderr),Q.exitCode}}}var j7=`Loki Mode (TypeScript port, Phase 2 of bash->Bun migration)
560
+ `.trim(),Z=await s(X,{cwd:p});return process.stdout.write(Z.stdout),0}let $=e(P(),"memory","index.json");if(!Z0($))return process.stdout.write(`No index found
561
+ `),0;let Q=await s(`import json, sys; sys.stdout.write(json.dumps(json.load(open(${JSON.stringify($)})), indent=4) + "\\n")`);if(Q.exitCode!==0)return process.stdout.write(`No index found
562
+ `),0;return process.stdout.write(Q.stdout),0}async function z0(K){switch(K[0]??"list"){case"list":case"ls":return o7();case"index":return n7(K[1]==="rebuild");default:{let Q=e(p,"autonomy","loki"),X=await S([Q,"memory",...K],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(X.stdout),process.stderr.write(X.stderr),X.exitCode}}}var j7=`Loki Mode (TypeScript port, Phase 2 of bash->Bun migration)
562
563
 
563
564
  Usage: loki <command> [args...]
564
565
 
@@ -576,12 +577,12 @@ Phase 2 ported (Bun-native, fast):
576
577
 
577
578
  All other commands fall through to the bash CLI (autonomy/loki).
578
579
  Set LOKI_LEGACY_BASH=1 to force the bash CLI for every command.
579
- `;function K6(){let K=process.env.LOKI_LEGACY_BASH;if(K===void 0)return;let $=K.trim().toLowerCase();if($!=="1"&&$!=="true"&&$!=="yes"&&$!=="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.
580
- `)}async function $6(K){K6();let $=K[0],z=K.slice(1);switch($){case void 0:case"help":case"--help":case"-h":return process.stdout.write(j7),0;case"version":case"--version":case"-v":return K0();case"provider":return Q0(z);case"memory":return Z0(z);case"status":{let{runStatus:Q}=await Promise.resolve().then(() => (B0(),G0));return Q(z)}case"stats":{let{runStats:Q}=await Promise.resolve().then(() => (A0(),T0));return Q(z)}case"doctor":{let{runDoctor:Q}=await Promise.resolve().then(() => (w0(),F0));return Q(z)}case"kpis":{let{runKpis:Q}=await Promise.resolve().then(() => (u0(),f0));return Q(z)}case"rollback":{let{runRollback:Q}=await Promise.resolve().then(() => (i0(),t0));return Q(z)}case"internal":{let Q=z[0];if(!Q||Q==="--help"||Q==="-h"||Q==="help"){let H=["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(`
581
- `);return process.stdout.write(`${H}
582
- `),0}if(Q==="phase1-hooks"){let{runInternalPhase1Hooks:H}=await Promise.resolve().then(() => (A7(),T7));return H(z.slice(1))}return process.stderr.write(`Unknown internal subcommand: ${Q}
580
+ `;function $6(){let K=process.env.LOKI_LEGACY_BASH;if(K===void 0)return;let $=K.trim().toLowerCase();if($!=="1"&&$!=="true"&&$!=="yes"&&$!=="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.
581
+ `)}async function Q6(K){$6();let $=K[0],Q=K.slice(1);switch($){case void 0:case"help":case"--help":case"-h":return process.stdout.write(j7),0;case"version":case"--version":case"-v":return K0();case"provider":return X0(Q);case"memory":return z0(Q);case"status":{let{runStatus:X}=await Promise.resolve().then(() => (B0(),G0));return X(Q)}case"stats":{let{runStats:X}=await Promise.resolve().then(() => (A0(),T0));return X(Q)}case"doctor":{let{runDoctor:X}=await Promise.resolve().then(() => (x0(),w0));return X(Q)}case"kpis":{let{runKpis:X}=await Promise.resolve().then(() => (u0(),f0));return X(Q)}case"rollback":{let{runRollback:X}=await Promise.resolve().then(() => (i0(),t0));return X(Q)}case"internal":{let X=Q[0];if(!X||X==="--help"||X==="-h"||X==="help"){let W=["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(`
582
+ `);return process.stdout.write(`${W}
583
+ `),0}if(X==="phase1-hooks"){let{runInternalPhase1Hooks:W}=await Promise.resolve().then(() => (A7(),T7));return W(Q.slice(1))}return process.stderr.write(`Unknown internal subcommand: ${X}
583
584
  `),process.stderr.write(`Run 'loki internal --help' for the supported list.
584
585
  `),2}default:return process.stderr.write(`Unknown command: ${$}
585
- `),process.stderr.write(j7),2}}process.on("SIGINT",()=>process.exit(130));process.on("SIGTERM",()=>process.exit(143));var z6=await $6(Bun.argv.slice(2));process.exit(z6);
586
+ `),process.stderr.write(j7),2}}process.on("SIGINT",()=>process.exit(130));process.on("SIGTERM",()=>process.exit(143));var X6=await Q6(Bun.argv.slice(2));process.exit(X6);
586
587
 
587
- //# debugId=E61F640A3343E84664756E2164756E21
588
+ //# debugId=508CDF0AB125EA5664756E2164756E21
package/mcp/__init__.py CHANGED
@@ -57,4 +57,4 @@ try:
57
57
  except ImportError:
58
58
  __all__ = ['mcp']
59
59
 
60
- __version__ = '7.7.15'
60
+ __version__ = '7.7.17'
@@ -0,0 +1,182 @@
1
+ """v7.7.17: structured error log for memory subsystem failures.
2
+
3
+ Replaces the silent-fail (`except Exception: pass`) pattern in
4
+ `autonomy/run.sh` memory call sites with explicit error logging to
5
+ `.loki/memory/.errors.log`. Surfaces in `loki doctor --json` so
6
+ developers see regressions early.
7
+
8
+ Record format (tab-separated, one record per line):
9
+ <iso_timestamp>\\t<function_name>\\t<error_class>\\t<message>\\t<traceback_snippet>
10
+
11
+ Rotation:
12
+ - At 10 MB the current file is renamed to `.errors.log.1`.
13
+ - Older generations shift down (`.1` -> `.2` -> `.3`).
14
+ - Up to 3 historical files retained; older ones are dropped.
15
+
16
+ Failure mode:
17
+ - Logging never raises. If the log itself is unwriteable (perms,
18
+ read-only fs, full disk) the call silently drops the record.
19
+ Observability must not break the memory pipeline.
20
+ """
21
+ from __future__ import annotations
22
+
23
+ import re
24
+ import traceback
25
+ from datetime import datetime, timezone
26
+ from pathlib import Path
27
+ from typing import List
28
+
29
+ MAX_LOG_SIZE_BYTES = 10 * 1024 * 1024 # 10 MB
30
+ MAX_HISTORICAL_FILES = 3
31
+ # Doctor surface and read_recent_errors only consider the tail of the file.
32
+ # Prevents an OOM on the doctor command if rotation ever fails (perms,
33
+ # disk full mid-rotation, or external writer appending).
34
+ TAIL_READ_BYTES = 64 * 1024 # 64 KB
35
+
36
+ # v7.7.17 (council fix Opus 2): scrub credential-shaped substrings from
37
+ # exception messages before writing to .errors.log. Doctor --json is
38
+ # commonly pasted into bug reports / chats; without this scrub, a stray
39
+ # `RuntimeError: Bearer sk-...` would leak.
40
+ # Mirrors the v7.7.10 USAGE.md regen scrubber (autonomy/run.sh).
41
+ _CREDENTIAL_KEYWORD_RE = re.compile(
42
+ r"(?i)(api[_-]?key|secret|password|token|private[_-]?key|credential|bearer)"
43
+ )
44
+ _HIGH_ENTROPY_TOKEN_RE = re.compile(
45
+ r"sk-[A-Za-z0-9_-]{16,}|pk_[A-Za-z0-9_-]{16,}|ghp_[A-Za-z0-9]{16,}|"
46
+ r"ghs_[A-Za-z0-9]{16,}|xox[bpoa]-[A-Za-z0-9-]{16,}|AIza[A-Za-z0-9_-]{32,}|"
47
+ r"AKIA[A-Z0-9]{12,}"
48
+ )
49
+
50
+
51
+ def _scrub(text: str) -> str:
52
+ """Redact credential-shaped substrings from `text`.
53
+
54
+ Two passes:
55
+ 1. Any token containing a credential keyword: replace the whole
56
+ token (whitespace-delimited) with `[REDACTED]`.
57
+ 2. Any literal high-entropy token shape (Stripe sk-, GitHub
58
+ ghp_/ghs_, Slack xox*, GCP AIza, AWS AKIA): replace inline
59
+ with `[REDACTED]`.
60
+ """
61
+ if not text:
62
+ return text
63
+ scrubbed_tokens = []
64
+ for token in text.split():
65
+ if _CREDENTIAL_KEYWORD_RE.search(token):
66
+ scrubbed_tokens.append("[REDACTED]")
67
+ else:
68
+ scrubbed_tokens.append(_HIGH_ENTROPY_TOKEN_RE.sub("[REDACTED]", token))
69
+ return " ".join(scrubbed_tokens)
70
+
71
+
72
+ def _errors_log_path(memory_base: str) -> Path:
73
+ return Path(memory_base) / ".errors.log"
74
+
75
+
76
+ def _rotate_if_needed(path: Path) -> None:
77
+ """Rotate `path` if it has grown past MAX_LOG_SIZE_BYTES.
78
+
79
+ Sequence (for MAX_HISTORICAL_FILES=3):
80
+ 1. If `path.log.3` exists, delete it (it falls off the back).
81
+ 2. Shift `path.log.2` -> `path.log.3`, `path.log.1` -> `path.log.2`.
82
+ 3. Rename `path` -> `path.log.1`.
83
+
84
+ All failures are silently absorbed via fallback truncation. The
85
+ caller cannot afford rotation errors to propagate.
86
+ """
87
+ try:
88
+ if not path.exists() or path.stat().st_size < MAX_LOG_SIZE_BYTES:
89
+ return
90
+ # Drop oldest generation
91
+ oldest = path.with_suffix(f".log.{MAX_HISTORICAL_FILES}")
92
+ if oldest.exists():
93
+ oldest.unlink()
94
+ # Shift older -> newer (going from N-1 down to 1)
95
+ for n in range(MAX_HISTORICAL_FILES - 1, 0, -1):
96
+ src = path.with_suffix(f".log.{n}")
97
+ dst = path.with_suffix(f".log.{n + 1}")
98
+ if src.exists():
99
+ src.rename(dst)
100
+ # Move current -> .log.1
101
+ path.rename(path.with_suffix(".log.1"))
102
+ except OSError:
103
+ # Last-ditch: try to truncate the current file so future writes
104
+ # do not keep failing. Silent on failure.
105
+ try:
106
+ path.unlink()
107
+ except OSError:
108
+ pass
109
+
110
+
111
+ def log_memory_error(memory_base: str, function_name: str, exc: BaseException) -> None:
112
+ """Append a structured error record to `<memory_base>/.errors.log`.
113
+
114
+ Never raises. Falls back to silent drop if the log is unwriteable.
115
+
116
+ Args:
117
+ memory_base: Absolute or relative path to a `.loki/memory/`
118
+ directory. Will be created if missing.
119
+ function_name: Short identifier of the call site (e.g.
120
+ "store_episode_trace", "auto_capture_episode").
121
+ exc: The caught exception to record.
122
+ """
123
+ try:
124
+ memory_dir = Path(memory_base)
125
+ memory_dir.mkdir(parents=True, exist_ok=True)
126
+ log_path = _errors_log_path(memory_base)
127
+ _rotate_if_needed(log_path)
128
+ # Tab-separated single-line record. Replace embedded tabs/newlines
129
+ # in user-provided strings so the format stays parseable.
130
+ tb_snippet = "".join(
131
+ traceback.format_exception_only(type(exc), exc)
132
+ ).strip()[:200].replace("\t", " ").replace("\n", " ")
133
+ msg = str(exc).replace("\t", " ").replace("\n", " ")[:200]
134
+ # v7.7.17 council fix (Opus 2): scrub credentials BEFORE writing
135
+ # to disk so doctor --json (often pasted in bug reports) cannot
136
+ # leak Bearer tokens, API keys, etc. embedded in exception text.
137
+ msg = _scrub(msg)
138
+ tb_snippet = _scrub(tb_snippet)
139
+ record = "\t".join([
140
+ datetime.now(timezone.utc).isoformat(),
141
+ function_name,
142
+ type(exc).__name__,
143
+ msg,
144
+ tb_snippet,
145
+ ])
146
+ with open(log_path, "a", encoding="utf-8") as f:
147
+ f.write(record + "\n")
148
+ except Exception:
149
+ return
150
+
151
+
152
+ def read_recent_errors(memory_base: str, limit: int = 5) -> List[str]:
153
+ """Read the last `limit` error records as raw lines (oldest-to-newest).
154
+
155
+ Returns an empty list if the log does not exist or cannot be read.
156
+ Never raises. The current file only is read (historical rotated
157
+ files are not consulted; recent enough for the doctor surface).
158
+
159
+ v7.7.17 council fix (Opus 2): seeks from end and reads at most
160
+ TAIL_READ_BYTES so an oversize log (rotation failed, disk-full
161
+ mid-rotate, external writer) cannot OOM the doctor command.
162
+ """
163
+ log_path = _errors_log_path(memory_base)
164
+ if not log_path.exists():
165
+ return []
166
+ try:
167
+ with open(log_path, "rb") as f:
168
+ f.seek(0, 2) # end
169
+ size = f.tell()
170
+ offset = max(0, size - TAIL_READ_BYTES)
171
+ f.seek(offset)
172
+ chunk = f.read()
173
+ text = chunk.decode("utf-8", errors="replace")
174
+ # If we seeked into the middle of a line, drop the (possibly
175
+ # corrupt) first partial line.
176
+ lines = text.split("\n")
177
+ if offset > 0 and lines:
178
+ lines = lines[1:]
179
+ lines = [ln.strip() for ln in lines if ln.strip()]
180
+ return lines[-limit:]
181
+ except OSError:
182
+ return []
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "loki-mode",
3
- "version": "7.7.15",
3
+ "version": "7.7.17",
4
4
  "description": "Loki Mode by Autonomi. Multi-agent autonomous SDLC framework. Spec to deployed app: PRD, GitHub issue, OpenAPI/JSON/YAML, or one-line brief. 4 AI providers (Claude Code, OpenAI Codex, Cline, Aider). 11 quality gates.",
5
5
  "keywords": [
6
6
  "agent",