loki-mode 7.5.1 → 7.5.2

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 PRD to deployed product with minimal human intervention. Requires --dangerously-skip-permissions flag.
4
4
  ---
5
5
 
6
- # Loki Mode v7.5.1
6
+ # Loki Mode v7.5.2
7
7
 
8
8
  **You are an autonomous agent. You make decisions. You do not ask questions. You do not stop.**
9
9
 
@@ -322,4 +322,4 @@ The following features are documented in skill modules but not yet fully automat
322
322
  | Quality gates 3-reviewer system | Implemented (v5.35.0) | 5 specialist reviewers in `skills/quality-gates.md`; execution in run.sh |
323
323
  | Benchmarks (HumanEval, SWE-bench) | Infrastructure only | Runner scripts and datasets exist in `benchmarks/`; no published results |
324
324
 
325
- **v7.5.1 | [Autonomi](https://www.autonomi.dev/) flagship product | ~260 lines core**
325
+ **v7.5.2 | [Autonomi](https://www.autonomi.dev/) flagship product | ~260 lines core**
package/VERSION CHANGED
@@ -1 +1 @@
1
- 7.5.1
1
+ 7.5.2
package/autonomy/loki CHANGED
@@ -6764,6 +6764,16 @@ cmd_doctor() {
6764
6764
  if [ "${BUN_FROM_SOURCE:-}" = "1" ] || [ "${BUN_FROM_SOURCE:-}" = "true" ]; then
6765
6765
  echo -e " ${GREEN}PASS${NC} BUN_FROM_SOURCE set: shim prefers loki-ts/src/ over dist/"
6766
6766
  fi
6767
+ # v7.5.2: explicit python3.12 probe -- chromadb + sentence-transformers
6768
+ # require it; the generic Python 3 check passes Python 3.13/3.14 and
6769
+ # silently fails at runtime. Informational only.
6770
+ py312_path=$(command -v python3.12 2>/dev/null || true)
6771
+ if [ -n "$py312_path" ]; then
6772
+ py312_ver=$("$py312_path" -c 'import sys; print(f"{sys.version_info.major}.{sys.version_info.minor}.{sys.version_info.micro}")' 2>/dev/null)
6773
+ echo -e " ${GREEN}PASS${NC} Python 3.12 (chromadb / sentence-transformers): ${py312_ver} at ${py312_path}"
6774
+ else
6775
+ echo -e " ${YELLOW}WARN${NC} Python 3.12 NOT on PATH -- chromadb / sentence-transformers may fail. Install via brew install python@3.12 or apt install python3.12."
6776
+ fi
6767
6777
  echo ""
6768
6778
 
6769
6779
  # Summary
package/bin/loki CHANGED
@@ -103,7 +103,8 @@ fi
103
103
  # Two-token routes (provider show/list, memory list/index) match on the first
104
104
  # token only; the Bun dispatcher handles subcommand routing internally.
105
105
  case "${1:-}" in
106
- version|--version|-v|status|stats|doctor|provider|memory)
106
+ version|--version|-v|status|stats|doctor|provider|memory|rollback)
107
+ # v7.5.2: rollback added (wires loki-ts/src/commands/rollback.ts).
107
108
  exec bun "$BUN_CLI" "$@"
108
109
  ;;
109
110
  *)
@@ -7,7 +7,7 @@ Modules:
7
7
  control: Session control API (start/stop/pause/resume)
8
8
  """
9
9
 
10
- __version__ = "7.5.1"
10
+ __version__ = "7.5.2"
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.5.1
5
+ **Version:** v7.5.2
6
6
 
7
7
  ---
8
8
 
@@ -1,55 +1,55 @@
1
1
  // @bun
2
- var Q0=Object.defineProperty;var X0=(z)=>z;function Z0(z,Q){this[z]=X0.bind(null,Q)}var t=(z,Q)=>{for(var $ in Q)Q0(z,$,{get:Q[$],enumerable:!0,configurable:!0,set:Z0.bind(Q,$)})};var h=(z,Q)=>()=>(z&&(Q=z(z=0)),Q);var K0=import.meta.require;var P1={};t(P1,{lokiDir:()=>C,homeLokiDir:()=>B1,findSkillDir:()=>U0,findRepoRootForVersion:()=>V1,REPO_ROOT:()=>v});import{resolve as R,dirname as U1}from"path";import{fileURLToPath as H0}from"url";import{existsSync as c}from"fs";import{homedir as L1}from"os";function W0(){let z=x1;for(let Q=0;Q<6;Q++){if(c(R(z,"VERSION"))&&c(R(z,"autonomy/run.sh")))return z;let $=U1(z);if($===z)break;z=$}return R(x1,"..","..","..")}function V1(z){let Q=z;for(let $=0;$<6;$++){if(c(R(Q,"VERSION"))&&c(R(Q,"autonomy/run.sh")))return Q;let X=U1(Q);if(X===Q)break;Q=X}return R(z,"..","..","..")}function C(){return process.env.LOKI_DIR??R(process.cwd(),".loki")}function B1(){return R(L1(),".loki")}function U0(){let z=[v,R(L1(),".claude/skills/loki-mode"),process.cwd()];for(let Q of z)if(c(R(Q,"SKILL.md"))&&c(R(Q,"autonomy/run.sh")))return Q;return null}var x1,v;var m=h(()=>{x1=U1(H0(import.meta.url));v=W0()});var S1={};t(S1,{runOrThrow:()=>J0,run:()=>w,commandVersion:()=>M0,commandExists:()=>S,ShellError:()=>q1});async function w(z,Q={}){let $=Bun.spawn({cmd:[...z],stdout:"pipe",stderr:"pipe",env:Q.env?{...process.env,...Q.env}:process.env,cwd:Q.cwd}),X;if(Q.timeoutMs&&Q.timeoutMs>0)X=setTimeout(()=>$.kill(),Q.timeoutMs);let[Z,H,W]=await Promise.all([new Response($.stdout).text(),new Response($.stderr).text(),$.exited]);if(X)clearTimeout(X);return{stdout:Z,stderr:H,exitCode:W}}async function J0(z,Q={}){let $=await w(z,Q);if($.exitCode!==0)throw new q1(`command failed (${$.exitCode}): ${z.join(" ")}`,$.exitCode,$.stdout,$.stderr);return $}async function S(z){let Q=j0(z),$=await w(["sh","-c",`command -v ${Q}`],{timeoutMs:5000});if($.exitCode===0)return $.stdout.trim()||null;return null}function j0(z){if(!/^[A-Za-z0-9._/-]+$/.test(z))throw Error(`refused to shell-escape suspect token: ${z}`);return z}async function M0(z,Q="--version"){if(!await S(z))return null;let X=await w([z,Q],{timeoutMs:5000});if(X.exitCode!==0)return null;return((X.stdout||X.stderr).split(/\r?\n/)[0]?.trim()??"")||null}var q1;var p=h(()=>{q1=class q1 extends Error{message;exitCode;stdout;stderr;constructor(z,Q,$,X){super(z);this.message=z;this.exitCode=Q;this.stdout=$;this.stderr=X;this.name="ShellError"}}});function f(z){return Y0?"":z}var Y0,D,k,_,A3,G,x,A,K;var l=h(()=>{Y0=(process.env.NO_COLOR??"").length>0;D=f("\x1B[0;31m"),k=f("\x1B[0;32m"),_=f("\x1B[1;33m"),A3=f("\x1B[0;34m"),G=f("\x1B[0;36m"),x=f("\x1B[1m"),A=f("\x1B[2m"),K=f("\x1B[0m")});import{existsSync as P0}from"fs";async function s(){if(a!==void 0)return a;let z="/opt/homebrew/bin/python3.12";if(P0(z))return a=z,z;let Q=await S("python3.12");if(Q)return a=Q,Q;let $=await S("python3");return a=$,$}async function i(z,Q={}){let $=await s();if(!$)return{stdout:"",stderr:"python3 not found",exitCode:127};return w([$,"-c",z],Q)}var a;var Q1=h(()=>{p()});var v1={};t(v1,{runStatus:()=>y0});import{existsSync as L,readFileSync as z1,readdirSync as C1,statSync as f1}from"fs";import{resolve as F,basename as E0}from"path";async function b0(){if(await S("jq"))return!0;return process.stdout.write(`${D}Error: jq is required but not installed.${K}
2
+ var _0=Object.defineProperty;var O0=($)=>$;function T0($,Z){this[$]=O0.bind(null,Z)}var o=($,Z)=>{for(var Q in Z)_0($,Q,{get:Z[Q],enumerable:!0,configurable:!0,set:T0.bind(Z,Q)})};var b=($,Z)=>()=>($&&(Z=$($=0)),Z);var w0=import.meta.require;var N1={};o(N1,{lokiDir:()=>P,homeLokiDir:()=>M1,findSkillDir:()=>F0,findRepoRootForVersion:()=>B1,REPO_ROOT:()=>p});import{resolve as E,dirname as G1}from"path";import{fileURLToPath as I0}from"url";import{existsSync as n}from"fs";import{homedir as k1}from"os";function L0(){let $=E1;for(let Z=0;Z<6;Z++){if(n(E($,"VERSION"))&&n(E($,"autonomy/run.sh")))return $;let Q=G1($);if(Q===$)break;$=Q}return E(E1,"..","..","..")}function B1($){let Z=$;for(let Q=0;Q<6;Q++){if(n(E(Z,"VERSION"))&&n(E(Z,"autonomy/run.sh")))return Z;let X=G1(Z);if(X===Z)break;Z=X}return E($,"..","..","..")}function P(){return process.env.LOKI_DIR??E(process.cwd(),".loki")}function M1(){return E(k1(),".loki")}function F0(){let $=[p,E(k1(),".claude/skills/loki-mode"),process.cwd()];for(let Z of $)if(n(E(Z,"SKILL.md"))&&n(E(Z,"autonomy/run.sh")))return Z;return null}var E1,p;var v=b(()=>{E1=G1(I0(import.meta.url));p=L0()});var g1={};o(g1,{runOrThrow:()=>D0,run:()=>T,commandVersion:()=>k0,commandExists:()=>k,ShellError:()=>Y1});async function T($,Z={}){let Q=Bun.spawn({cmd:[...$],stdout:"pipe",stderr:"pipe",env:Z.env?{...process.env,...Z.env}:process.env,cwd:Z.cwd}),X,z;if(Z.timeoutMs&&Z.timeoutMs>0)X=setTimeout(()=>{try{Q.kill("SIGTERM")}catch{}z=setTimeout(()=>{try{Q.kill("SIGKILL")}catch{}},2000)},Z.timeoutMs);try{let[H,V,U]=await Promise.all([new Response(Q.stdout).text(),new Response(Q.stderr).text(),Q.exited]);return{stdout:H,stderr:V,exitCode:U}}finally{if(X)clearTimeout(X);if(z)clearTimeout(z)}}async function D0($,Z={}){let Q=await T($,Z);if(Q.exitCode!==0)throw new Y1(`command failed (${Q.exitCode}): ${$.join(" ")}`,Q.exitCode,Q.stdout,Q.stderr);return Q}async function k($){let Z=E0($),Q=await T(["sh","-c",`command -v ${Z}`],{timeoutMs:5000});if(Q.exitCode===0)return Q.stdout.trim()||null;return null}function E0($){if(!/^[A-Za-z0-9._/-]+$/.test($))throw Error(`refused to shell-escape suspect token: ${$}`);return $}async function k0($,Z="--version"){if(!await k($))return null;let X=await T([$,Z],{timeoutMs:5000});if(X.exitCode!==0)return null;return((X.stdout||X.stderr).split(/\r?\n/)[0]?.trim()??"")||null}var Y1;var y=b(()=>{Y1=class Y1 extends Error{message;exitCode;stdout;stderr;constructor($,Z,Q,X){super($);this.message=$;this.exitCode=Z;this.stdout=Q;this.stderr=X;this.name="ShellError"}}});function h($){return N0?"":$}var N0,j,R,O,o3,q,I,_,K;var l=b(()=>{N0=(process.env.NO_COLOR??"").length>0;j=h("\x1B[0;31m"),R=h("\x1B[0;32m"),O=h("\x1B[1;33m"),o3=h("\x1B[0;34m"),q=h("\x1B[0;36m"),I=h("\x1B[1m"),_=h("\x1B[2m"),K=h("\x1B[0m")});import{existsSync as p0}from"fs";async function r(){if(e!==void 0)return e;let $="/opt/homebrew/bin/python3.12";if(p0($))return e=$,$;let Z=await k("python3.12");if(Z)return e=Z,Z;let Q=await k("python3");return e=Q,Q}async function $1($,Z={}){let Q=await r();if(!Q)return{stdout:"",stderr:"python3 not found",exitCode:127};return T([Q,"-c",$],Z)}var e;var z1=b(()=>{y()});var d1={};o(d1,{runStatus:()=>e0});import{existsSync as D,readFileSync as Z1,readdirSync as u1,statSync as m1}from"fs";import{resolve as S,basename as n0}from"path";async function t0(){if(await k("jq"))return!0;return process.stdout.write(`${j}Error: jq is required but not installed.${K}
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 X1(z){if(!Number.isFinite(z)||z<=0)return!1;try{return process.kill(z,0),!0}catch{return!1}}function Z1(z){if(!L(z))return null;try{let Q=z1(z,"utf-8").trim();if(!Q)return null;let $=Number.parseInt(Q,10);return Number.isFinite($)?$:null}catch{return null}}function C0(z){let Q=[],$=Z1(F(z,"loki.pid"));if($!==null&&X1($))Q.push(`global:${$}`);let X=F(z,"sessions");if(L(X)){let Z=[];try{Z=C1(X)}catch{Z=[]}for(let H of Z){let W=F(X,H);try{if(!f1(W).isDirectory())continue}catch{continue}let U=F(W,"loki.pid"),J=Z1(U);if(J!==null&&X1(J))Q.push(`${H}:${J}`)}}if(L(z)){let Z=[];try{Z=C1(z)}catch{Z=[]}for(let H of Z){if(!H.startsWith("run-")||!H.endsWith(".pid"))continue;let W=F(z,H);try{if(!f1(W).isFile())continue}catch{continue}let U=E0(H,".pid").slice(4),J=Z1(W);if(J!==null&&X1(J)){if(!Q.some((T)=>T.startsWith(`${U}:`)))Q.push(`${U}:${J}`)}}}return Q}async function g1(z,Q){let $=await w(["jq","-r",z,Q]);if($.exitCode!==0)return null;return $.stdout.trim()}function y1(z,Q){try{let $=z1(z,"utf-8"),Z=JSON.parse($)[Q];if(typeof Z==="number"){if(Q==="budget_used"){let H=Math.round(Z*100)/100;if(Number.isInteger(H))return String(H);return String(H)}return String(Z)}if(Z===void 0||Z===null)return"0";return String(Z)}catch{return"0"}}function h1(z,Q,$){try{let X=z1(z,"utf-8"),H=JSON.parse(X)[Q];if(typeof H==="number"&&Number.isFinite(H))return H;return $}catch{return $}}async function f0(){let z=C();if(!await b0())return 1;if(!L(z))return process.stdout.write(`${x}Loki Mode Status${K}
7
+ `),!1}function K1($){if(!Number.isFinite($)||$<=0)return!1;try{return process.kill($,0),!0}catch{return!1}}function H1($){if(!D($))return null;try{let Z=Z1($,"utf-8").trim();if(!Z)return null;let Q=Number.parseInt(Z,10);return Number.isFinite(Q)?Q:null}catch{return null}}function a0($){let Z=[],Q=H1(S($,"loki.pid"));if(Q!==null&&K1(Q))Z.push(`global:${Q}`);let X=S($,"sessions");if(D(X)){let z=[];try{z=u1(X)}catch{z=[]}for(let H of z){let V=S(X,H);try{if(!m1(V).isDirectory())continue}catch{continue}let U=S(V,"loki.pid"),B=H1(U);if(B!==null&&K1(B))Z.push(`${H}:${B}`)}}if(D($)){let z=[];try{z=u1($)}catch{z=[]}for(let H of z){if(!H.startsWith("run-")||!H.endsWith(".pid"))continue;let V=S($,H);try{if(!m1(V).isFile())continue}catch{continue}let U=n0(H,".pid").slice(4),B=H1(V);if(B!==null&&K1(B)){if(!Z.some((L)=>L.startsWith(`${U}:`)))Z.push(`${U}:${B}`)}}}return Z}async function p1($,Z){let Q=await T(["jq","-r",$,Z]);if(Q.exitCode!==0)return null;return Q.stdout.trim()}function c1($,Z){try{let Q=Z1($,"utf-8"),z=JSON.parse(Q)[Z];if(typeof z==="number"){if(Z==="budget_used"){let H=Math.round(z*100)/100;if(Number.isInteger(H))return String(H);return String(H)}return String(z)}if(z===void 0||z===null)return"0";return String(z)}catch{return"0"}}function l1($,Z,Q){try{let X=Z1($,"utf-8"),H=JSON.parse(X)[Z];if(typeof H==="number"&&Number.isFinite(H))return H;return Q}catch{return Q}}async function s0(){let $=P();if(!await t0())return 1;if(!D($))return process.stdout.write(`${I}Loki Mode Status${K}
8
8
  `),process.stdout.write(`
9
- `),process.stdout.write(`${_}No active session found.${K}
9
+ `),process.stdout.write(`${O}No active session found.${K}
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(`${A}Current directory: ${process.cwd()}${K}
17
- `),0;process.stdout.write(`${x}Loki Mode Status${K}
16
+ `),process.stdout.write(`${_}Current directory: ${process.cwd()}${K}
17
+ `),0;process.stdout.write(`${I}Loki Mode Status${K}
18
18
  `),process.stdout.write(`
19
- `);let Q="",$=F(z,"state","provider");if(L($))try{Q=z1($,"utf-8").trim()}catch{Q=""}let X=Q||process.env.LOKI_PROVIDER||"claude",Z="full features";switch(X){case"codex":case"gemini":case"aider":Z="degraded mode";break;case"cline":Z="near-full mode";break;default:Z="full features";break}process.stdout.write(`${G}Provider:${K} ${X} (${Z})
20
- `),process.stdout.write(`${A} Switch with: loki provider set <claude|codex|gemini|cline|aider>${K}
19
+ `);let Z="",Q=S($,"state","provider");if(D(Q))try{Z=Z1(Q,"utf-8").trim()}catch{Z=""}let X=Z||process.env.LOKI_PROVIDER||"claude",z="full features";switch(X){case"codex":case"gemini":case"aider":z="degraded mode";break;case"cline":z="near-full mode";break;default:z="full features";break}process.stdout.write(`${q}Provider:${K} ${X} (${z})
20
+ `),process.stdout.write(`${_} Switch with: loki provider set <claude|codex|gemini|cline|aider>${K}
21
21
  `),process.stdout.write(`
22
- `);let H=C0(z);if(H.length>0){process.stdout.write(`${k}Active Sessions: ${H.length}${K}
23
- `);for(let B of H){let M=B.indexOf(":"),I=M>=0?B.slice(0,M):B,N=M>=0?B.slice(M+1):"";if(I==="global")process.stdout.write(` ${G}[global]${K} PID ${N}
24
- `);else process.stdout.write(` ${G}[#${I}]${K} PID ${N}
22
+ `);let H=a0($);if(H.length>0){process.stdout.write(`${R}Active Sessions: ${H.length}${K}
23
+ `);for(let J of H){let A=J.indexOf(":"),F=A>=0?J.slice(0,A):J,f=A>=0?J.slice(A+1):"";if(F==="global")process.stdout.write(` ${q}[global]${K} PID ${f}
24
+ `);else process.stdout.write(` ${q}[#${F}]${K} PID ${f}
25
25
  `)}process.stdout.write(`
26
- `),process.stdout.write(`${A} Stop specific: loki stop <session-id>${K}
27
- `),process.stdout.write(`${A} Stop all: loki stop${K}
26
+ `),process.stdout.write(`${_} Stop specific: loki stop <session-id>${K}
27
+ `),process.stdout.write(`${_} Stop all: loki stop${K}
28
28
  `),process.stdout.write(`
29
- `)}if(L(F(z,"PAUSE")))process.stdout.write(`${_}Status: PAUSED${K}
30
- `),process.stdout.write(`${A} Resume with: loki resume${K}
29
+ `)}if(D(S($,"PAUSE")))process.stdout.write(`${O}Status: PAUSED${K}
30
+ `),process.stdout.write(`${_} Resume with: loki resume${K}
31
31
  `),process.stdout.write(`
32
- `);else if(L(F(z,"STOP")))process.stdout.write(`${D}Status: STOPPED${K}
33
- `),process.stdout.write(`${A} Clear with: loki resume${K}
32
+ `);else if(D(S($,"STOP")))process.stdout.write(`${j}Status: STOPPED${K}
33
+ `),process.stdout.write(`${_} Clear with: loki resume${K}
34
34
  `),process.stdout.write(`
35
- `);let W=F(z,"STATUS.txt");if(L(W)){process.stdout.write(`${G}Session Info:${K}
36
- `);try{process.stdout.write(z1(W,"utf-8"))}catch{}process.stdout.write(`
37
- `)}let U=F(z,"state","orchestrator.json");if(L(U)){process.stdout.write(`${G}Orchestrator State:${K}
38
- `);let B=await g1('.currentPhase // "unknown"',U);process.stdout.write(`${B??"unknown"}
39
- `)}let J=F(z,"queue","pending.json");if(L(J)){let B=await g1('if type == "array" then length elif .tasks then .tasks | length else 0 end',J);process.stdout.write(`${G}Pending Tasks:${K} ${B??"0"}
40
- `)}let O=F(z,"metrics","budget.json");if(L(O)){let B=y1(O,"budget_limit"),M=y1(O,"budget_used");if(B!=="0")process.stdout.write(`${G}Budget:${K} $${M} / $${B}
41
- `);else process.stdout.write(`${G}Cost:${K} $${M} (no limit)
42
- `)}let T=F(z,"state","context-usage.json");if(L(T)){let B=h1(T,"window_size",200000),M=h1(T,"used_tokens",0),I=0;if(B>0)I=Math.floor(M*100/B);process.stdout.write(`${G}Context:${K} ${I}% (${M} / ${B} tokens)
43
- `)}let E=F(z,"dashboard","dashboard.pid");if(L(E)){let B=Z1(E);if(B!==null&&X1(B)){let M=process.env.LOKI_DASHBOARD_PORT||"57374";process.stdout.write(`${G}Dashboard:${K} http://127.0.0.1:${M}/
35
+ `);let V=S($,"STATUS.txt");if(D(V)){process.stdout.write(`${q}Session Info:${K}
36
+ `);try{process.stdout.write(Z1(V,"utf-8"))}catch{}process.stdout.write(`
37
+ `)}let U=S($,"state","orchestrator.json");if(D(U)){process.stdout.write(`${q}Orchestrator State:${K}
38
+ `);let J=await p1('.currentPhase // "unknown"',U);process.stdout.write(`${J??"unknown"}
39
+ `)}let B=S($,"queue","pending.json");if(D(B)){let J=await p1('if type == "array" then length elif .tasks then .tasks | length else 0 end',B);process.stdout.write(`${q}Pending Tasks:${K} ${J??"0"}
40
+ `)}let w=S($,"metrics","budget.json");if(D(w)){let J=c1(w,"budget_limit"),A=c1(w,"budget_used");if(J!=="0")process.stdout.write(`${q}Budget:${K} $${A} / $${J}
41
+ `);else process.stdout.write(`${q}Cost:${K} $${A} (no limit)
42
+ `)}let L=S($,"state","context-usage.json");if(D(L)){let J=l1(L,"window_size",200000),A=l1(L,"used_tokens",0),F=0;if(J>0)F=Math.floor(A*100/J);process.stdout.write(`${q}Context:${K} ${F}% (${A} / ${J} tokens)
43
+ `)}let N=S($,"dashboard","dashboard.pid");if(D(N)){let J=H1(N);if(J!==null&&K1(J)){let A=process.env.LOKI_DASHBOARD_PORT||"57374";process.stdout.write(`${q}Dashboard:${K} http://127.0.0.1:${A}/
44
44
  `)}}return process.stdout.write(`
45
- `),process.stdout.write(`${A} Tip: loki context show - detailed token breakdown${K}
46
- `),process.stdout.write(`${A} Tip: loki code overview - codebase intelligence${K}
47
- `),0}async function g0(){let z=await s();if(!z)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
48
- `),1;let Q=v,$=C(),X=process.env.LOKI_DASHBOARD_PORT||"57374",Z=process.env.LOKI_PROVIDER||"claude",H=await w([z,"-c",N0,Q,$,X,Z],{timeoutMs:30000});if(H.exitCode!==0)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
49
- `),1;return process.stdout.write(H.stdout),0}async function y0(z){let Q=[...z];while(Q.length>0){let $=Q[0];if($==="--json")return g0();if($==="--help"||$==="-h")return process.stdout.write(`Usage: loki status [--json]
50
- `),0;return process.stdout.write(`${D}Unknown flag: ${$}${K}
45
+ `),process.stdout.write(`${_} Tip: loki context show - detailed token breakdown${K}
46
+ `),process.stdout.write(`${_} Tip: loki code overview - codebase intelligence${K}
47
+ `),0}async function i0(){let $=await r();if(!$)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
48
+ `),1;let Z=p,Q=P(),X=process.env.LOKI_DASHBOARD_PORT||"57374",z=process.env.LOKI_PROVIDER||"claude",H=await T([$,"-c",r0,Z,Q,X,z],{timeoutMs:30000});if(H.exitCode!==0)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
49
+ `),1;return process.stdout.write(H.stdout),0}async function e0($){let Z=[...$];while(Z.length>0){let Q=Z[0];if(Q==="--json")return i0();if(Q==="--help"||Q==="-h")return process.stdout.write(`Usage: loki status [--json]
50
+ `),0;return process.stdout.write(`${j}Unknown flag: ${Q}${K}
51
51
  `),process.stdout.write(`Usage: loki status [--json]
52
- `),1}return f0()}var N0=`
52
+ `),1}return s0()}var r0=`
53
53
  import json, os, sys, time
54
54
 
55
55
  skill_dir = sys.argv[1]
@@ -195,9 +195,9 @@ if os.path.isdir(queue_dir):
195
195
  result['task_counts'] = task_counts
196
196
 
197
197
  print(json.dumps(result, indent=2))
198
- `;var m1=h(()=>{p();Q1();l();m()});var l1={};t(l1,{runStats:()=>c0,computeStats:()=>c1});import{readdirSync as u1,readFileSync as h0,statSync as p1}from"fs";import{join as g}from"path";function o(z){try{if(!p1(z).isFile())return null;return JSON.parse(h0(z,"utf-8"))}catch{return null}}function M1(z){try{return p1(z).isDirectory()}catch{return!1}}function v0(z){if(!M1(z))return[];try{let Q=u1(z).filter(($)=>$.startsWith("iteration-")&&$.endsWith(".json"));return Q.sort(),Q.map(($)=>g(z,$))}catch{return[]}}function n(z){return Math.trunc(z).toLocaleString("en-US")}function J1(z){let Q=Math.trunc(z);if(Q<60)return`${Q}s`;let $=Math.trunc(Q/3600),X=Math.trunc(Q%3600/60),Z=Q%60;if($>0)return`${$}h ${String(X).padStart(2,"0")}m`;return`${X}m ${String(Z).padStart(2,"0")}s`}function y(z,Q=0){let $=Math.pow(10,Q);return Math.round(z*$)/$}function r(z,Q){return z.toFixed(Q)}function j1(z,Q){return z.length>=Q?z:z+" ".repeat(Q-z.length)}function m0(z){let Q="N/A",$=0,X=o(g(z,"state","orchestrator.json"));if(X&&typeof X==="object"){if(typeof X.currentPhase==="string")Q=X.currentPhase;if(typeof X.currentIteration==="number")$=X.currentIteration}let Z=g(z,"metrics","efficiency"),H=v0(Z),W=[];for(let q of H){let j=o(q);if(j&&typeof j==="object")W.push(j)}if(W.length>0)$=Math.max($,W.length);let U=W.reduce((q,j)=>q+(j.input_tokens??0),0),J=W.reduce((q,j)=>q+(j.output_tokens??0),0),O=U+J,T=W.reduce((q,j)=>q+(j.cost_usd??0),0),E=W.reduce((q,j)=>q+(j.duration_seconds??0),0),B=0,M=0,I=o(g(z,"metrics","budget.json"));if(I&&typeof I==="object"){if(typeof I.budget_limit==="number")B=I.budget_limit;if(typeof I.budget_used==="number")M=I.budget_used}let N=0,Y=0,P=o(g(z,"state","quality-gates.json"));if(P&&typeof P==="object"){if(Array.isArray(P)){for(let q of P)if(Y+=1,q===!0)N+=1;else if(q&&typeof q==="object"){let j=q;if(j.passed===!0||j.status==="passed")N+=1}}else for(let q of Object.values(P))if(typeof q==="boolean"){if(Y+=1,q)N+=1}else if(q&&typeof q==="object"){Y+=1;let j=q;if(j.passed===!0||j.status==="passed")N+=1}}let _1={},$1=o(g(z,"quality","gate-failure-count.json"));if($1&&typeof $1==="object"&&!Array.isArray($1)){let q={};for(let[j,b]of Object.entries($1))if(typeof b==="number")q[j]=b;_1=q}let T1=0,I1=0,w1=0,W1=g(z,"quality");if(M1(W1)){let q=[];try{q=u1(W1)}catch{q=[]}for(let j of q){if(!j.endsWith(".json")||j==="gate-failure-count.json")continue;let b=o(g(W1,j));if(!b||typeof b!=="object")continue;if(!(("verdict"in b)||("approved"in b)||("reviewers"in b)))continue;T1+=1;let F1=(b.verdict??"").toString().toLowerCase();if(b.approved===!0||["approved","approve","pass"].includes(F1))I1+=1;else if(["revision","revise","changes_requested","reject"].includes(F1))w1+=1}}return{phase:Q,iterationCount:$,iterations:W,totalInput:U,totalOutput:J,totalTokens:O,totalCost:T,totalDuration:E,budgetLimit:B,budgetUsed:M,gatesPassed:N,gatesTotal:Y,gateFailures:_1,reviewsTotal:T1,reviewsApproved:I1,reviewsRevision:w1}}function u0(z,Q){let $=z.iterationCount,X={session:{iterations:$,duration_seconds:z.totalDuration,phase:z.phase},tokens:{input:z.totalInput,output:z.totalOutput,total:z.totalTokens,cost_usd:y(z.totalCost,2)},quality:{gates_passed:z.gatesPassed,gates_total:z.gatesTotal,reviews_total:z.reviewsTotal,reviews_approved:z.reviewsApproved,reviews_revision:z.reviewsRevision,gate_failures:z.gateFailures},efficiency:{avg_tokens_per_iteration:$>0?y(z.totalTokens/$,0):0,avg_cost_per_iteration:$>0?y(z.totalCost/$,2):0,avg_duration_per_iteration:$>0?y(z.totalDuration/$,1):0},budget:{used:y(z.budgetUsed,2),limit:z.budgetLimit,percent:z.budgetLimit>0?y(z.budgetUsed/z.budgetLimit*100,1):0}};if(Q)X.iterations=z.iterations.map((W,U)=>({number:U+1,input_tokens:W.input_tokens??0,output_tokens:W.output_tokens??0,cost_usd:y(W.cost_usd??0,2),duration_seconds:W.duration_seconds??0}));let Z=JSON.stringify(X,null,2);function H(W,U){if(!U)return;let J=new RegExp(`("${W}": )(-?\\d+)(,?)$`,"m");Z=Z.replace(J,(O,T,E,B)=>`${T}${E}.0${B}`)}if(H("avg_duration_per_iteration",$>0&&Number.isInteger(X.efficiency.avg_duration_per_iteration)),H("percent",z.budgetLimit>0&&Number.isInteger(X.budget.percent)),H("cost_usd",$>0&&Number.isInteger(X.tokens.cost_usd)),Q)Z=Z.replace(/("cost_usd": )(-?\d+)(,?)$/gm,(W,U,J,O)=>`${U}${J}.0${O}`);return Z}function p0(z,Q){let $=[];if($.push("Loki Mode Session Statistics"),$.push("============================"),$.push(""),$.push("Session"),$.push(` Iterations completed: ${z.iterationCount}`),$.push(` Duration: ${J1(z.totalDuration)}`),$.push(` Current phase: ${z.phase}`),$.push(""),$.push("Token Usage"),z.iterations.length>0)$.push(` Input tokens: ${n(z.totalInput)}`),$.push(` Output tokens: ${n(z.totalOutput)}`),$.push(` Total tokens: ${n(z.totalTokens)}`),$.push(` Estimated cost: $${r(z.totalCost,2)}`);else $.push(" N/A (no iteration metrics found)");if($.push(""),$.push("Quality Gates"),z.gatesTotal>0){let X=Math.round(z.gatesPassed/z.gatesTotal*100);$.push(` Gates passed: ${z.gatesPassed}/${z.gatesTotal} (${X}%)`)}else $.push(" Gates passed: N/A");if(z.reviewsTotal>0){let X=[];if(z.reviewsApproved>0)X.push(`${z.reviewsApproved} approved`);if(z.reviewsRevision>0)X.push(`${z.reviewsRevision} revision requested`);let Z=X.length>0?X.join(", "):"N/A";$.push(` Code reviews: ${z.reviewsTotal} (${Z})`)}if(Object.keys(z.gateFailures).length>0){let X=Object.entries(z.gateFailures).filter(([,Z])=>Z>0).map(([Z,H])=>`${Z} (${H})`);if(X.length>0)$.push(` Gate failures: ${X.join(", ")}`)}if($.push(""),$.push("Efficiency"),z.iterationCount>0&&z.iterations.length>0){let X=Math.round(z.totalTokens/z.iterationCount),Z=z.totalCost/z.iterationCount,H=z.totalDuration/z.iterationCount;$.push(` Avg tokens/iteration: ${n(X)}`),$.push(` Avg cost/iteration: $${r(Z,2)}`),$.push(` Avg duration/iteration: ${J1(H)}`)}else $.push(" N/A (no iteration metrics found)");if($.push(""),$.push("Budget"),z.budgetLimit>0){let X=y(z.budgetUsed/z.budgetLimit*100,1),Z=Number.isInteger(X)?`${X}.0`:`${X}`;$.push(` Used: $${r(z.budgetUsed,2)} / $${r(z.budgetLimit,2)} (${Z}%)`)}else if(z.budgetUsed>0)$.push(` Used: $${r(z.budgetUsed,2)} (no limit set)`);else $.push(" N/A");if(Q&&z.iterations.length>0)$.push(""),$.push("Per-Iteration Breakdown"),z.iterations.forEach((X,Z)=>{let H=Z+1,W=j1(n(X.input_tokens??0),10),U=j1(n(X.output_tokens??0),10),J=X.cost_usd??0,O=J1(X.duration_seconds??0),T=j1(`${H}`,3);$.push(` #${T} input: ${W} output: ${U} cost: $${r(J,2)} time: ${O}`)});return $.join(`
199
- `)}function c1(z){let Q=!1,$=!1;for(let W of z)if(W==="--json")Q=!0;else if(W==="--efficiency")$=!0;let X=C();if(!M1(X)){if(Q)return{exitCode:0,stdout:'{"error": "No active session"}'};return{exitCode:0,stdout:`${_}No active session found.${K}
200
- Start a session with: loki start <prd>`}}let Z=m0(X);return{exitCode:0,stdout:Q?u0(Z,$):p0(Z,$)}}async function c0(z){let Q=c1(z);return console.log(Q.stdout),Q.exitCode}var d1=h(()=>{m();l()});var e1={};t(e1,{runDoctor:()=>$3,httpReachable:()=>A1,checkTool:()=>t1,checkSkills:()=>a1,checkDisk:()=>O1,buildDoctorJson:()=>i1});import{existsSync as l0,lstatSync as d0,readlinkSync as o0,statfsSync as n0}from"fs";import{homedir as n1}from"os";import{resolve as o1}from"path";function t0(z){let Q=z.match(r0);return Q?Q[1]:null}async function a0(z){try{let Q=await w([z,"--version"],{timeoutMs:5000}),$=(Q.stdout||Q.stderr||"").trim();return t0($)}catch{return null}}function r1(z,Q){let $=z.split(".").map((Z)=>parseInt(Z,10)),X=Q.split(".").map((Z)=>parseInt(Z,10));while($.length<2)$.push(0);while(X.length<2)X.push(0);for(let Z=0;Z<2;Z++){let H=$[Z]??0,W=X[Z]??0;if(Number.isNaN(H)||Number.isNaN(W))return 0;if(H!==W)return H-W}return 0}async function t1(z,Q,$,X=null){let Z=await S(Q),H=Z!==null,W=H?await a0(Q):null,U="pass";if(!H)U=$==="required"?"fail":"warn";else if(X&&W){if(r1(W,X)<0)U=$==="required"?"fail":"warn"}return{name:z,command:Q,found:H,version:W,required:$,min_version:X,status:U,path:Z}}function O1(){let z=null;try{let $=n0(n1()),X=Number($.bavail)*Number($.bsize);z=Math.round(X/1073741824*10)/10}catch{z=null}let Q="pass";if(z!==null){if(z<1)Q="fail";else if(z<5)Q="warn"}return{available_gb:z,status:Q}}async function A1(z,Q=2000){try{return(await fetch(z,{signal:AbortSignal.timeout(Q)})).ok}catch{return!1}}async function Y1(z,Q=!1){let $=`import ${z}`,X=Q?30000:5000;if(!Q)return(await i($,{timeoutMs:X})).exitCode===0;let Z=await s();if(!Z)return!1;return(await w([Z,"-c",$],{timeoutMs:X})).exitCode===0}function a1(){let z=n1();return s0.map(({name:Q,dir:$})=>{let X=o1(z,$),Z=X,H=o1(X,"SKILL.md");if(l0(H))return{name:Q,path:Z,status:"pass",detail:""};try{if(d0(X).isSymbolicLink()){let U="unknown";try{U=o0(X)}catch{}return{name:Q,path:Z,status:"fail",detail:`(broken symlink -> ${U})`}}}catch{}return{name:Q,path:Z,status:"warn",detail:"(not found - run 'loki setup-skill')"}})}async function s1(){return Promise.all(i0.map(async(z)=>{return{...await t1(z.jsonName,z.cmd,z.required,z.min??null),displayName:z.displayName}}))}async function i1(){let Q=(await s1()).map(({displayName:W,...U})=>U),$=O1(),X=0,Z=0,H=0;for(let W of Q)if(W.status==="pass")X++;else if(W.status==="fail")Z++;else H++;if($.status==="pass")X++;else if($.status==="fail")Z++;else H++;return{checks:Q,disk:$,summary:{passed:X,failed:Z,warnings:H,ok:Z===0}}}function V(z){switch(z){case"pass":return`${k}PASS${K}`;case"fail":return`${D}FAIL${K}`;case"warn":return`${_}WARN${K}`}}function K1(z){let Q=z.version?` (v${z.version})`:"",$=z.displayName;if(!z.found){let X=z.required==="required"?"not found":z.required==="recommended"?"not found (recommended)":"not found (optional)";return` ${V(z.status)} ${$} - ${X}`}if(z.min_version&&z.version&&r1(z.version,z.min_version)<0){let X=z.required==="required"?"requires":"recommended";return` ${V(z.status)} ${$}${Q} - ${X} >= ${z.min_version}`}return` ${V(z.status)} ${$}${Q}`}function H1(z,Q){if(Q==="pass")z.pass++;else if(Q==="fail")z.fail++;else z.warn++}function e0(){process.stdout.write(`${x}loki doctor${K} - Check system prerequisites
198
+ `;var o1=b(()=>{y();z1();l();v()});var a1={};o(a1,{runStats:()=>K3,computeStats:()=>t1});import{readdirSync as n1,readFileSync as $3,statSync as r1}from"fs";import{join as u}from"path";function a($){try{if(!r1($).isFile())return null;return JSON.parse($3($,"utf-8"))}catch{return null}}function O1($){try{return r1($).isDirectory()}catch{return!1}}function Q3($){if(!O1($))return[];try{let Z=n1($).filter((Q)=>Q.startsWith("iteration-")&&Q.endsWith(".json"));return Z.sort(),Z.map((Q)=>u($,Q))}catch{return[]}}function s($){return Math.trunc($).toLocaleString("en-US")}function j1($){let Z=Math.trunc($);if(Z<60)return`${Z}s`;let Q=Math.trunc(Z/3600),X=Math.trunc(Z%3600/60),z=Z%60;if(Q>0)return`${Q}h ${String(X).padStart(2,"0")}m`;return`${X}m ${String(z).padStart(2,"0")}s`}function m($,Z=0){let Q=Math.pow(10,Z);return Math.round($*Q)/Q}function i($,Z){return $.toFixed(Z)}function _1($,Z){return $.length>=Z?$:$+" ".repeat(Z-$.length)}function Z3($){let Z="N/A",Q=0,X=a(u($,"state","orchestrator.json"));if(X&&typeof X==="object"){if(typeof X.currentPhase==="string")Z=X.currentPhase;if(typeof X.currentIteration==="number")Q=X.currentIteration}let z=u($,"metrics","efficiency"),H=Q3(z),V=[];for(let G of H){let M=a(G);if(M&&typeof M==="object")V.push(M)}if(V.length>0)Q=Math.max(Q,V.length);let U=V.reduce((G,M)=>G+(M.input_tokens??0),0),B=V.reduce((G,M)=>G+(M.output_tokens??0),0),w=U+B,L=V.reduce((G,M)=>G+(M.cost_usd??0),0),N=V.reduce((G,M)=>G+(M.duration_seconds??0),0),J=0,A=0,F=a(u($,"metrics","budget.json"));if(F&&typeof F==="object"){if(typeof F.budget_limit==="number")J=F.budget_limit;if(typeof F.budget_used==="number")A=F.budget_used}let f=0,C=0,Y=a(u($,"state","quality-gates.json"));if(Y&&typeof Y==="object"){if(Array.isArray(Y)){for(let G of Y)if(C+=1,G===!0)f+=1;else if(G&&typeof G==="object"){let M=G;if(M.passed===!0||M.status==="passed")f+=1}}else for(let G of Object.values(Y))if(typeof G==="boolean"){if(C+=1,G)f+=1}else if(G&&typeof G==="object"){C+=1;let M=G;if(M.passed===!0||M.status==="passed")f+=1}}let x={},X1=a(u($,"quality","gate-failure-count.json"));if(X1&&typeof X1==="object"&&!Array.isArray(X1)){let G={};for(let[M,g]of Object.entries(X1))if(typeof g==="number")G[M]=g;x=G}let P1=0,R1=0,S1=0,J1=u($,"quality");if(O1(J1)){let G=[];try{G=n1(J1)}catch{G=[]}for(let M of G){if(!M.endsWith(".json")||M==="gate-failure-count.json")continue;let g=a(u(J1,M));if(!g||typeof g!=="object")continue;if(!(("verdict"in g)||("approved"in g)||("reviewers"in g)))continue;P1+=1;let D1=(g.verdict??"").toString().toLowerCase();if(g.approved===!0||["approved","approve","pass"].includes(D1))R1+=1;else if(["revision","revise","changes_requested","reject"].includes(D1))S1+=1}}return{phase:Z,iterationCount:Q,iterations:V,totalInput:U,totalOutput:B,totalTokens:w,totalCost:L,totalDuration:N,budgetLimit:J,budgetUsed:A,gatesPassed:f,gatesTotal:C,gateFailures:x,reviewsTotal:P1,reviewsApproved:R1,reviewsRevision:S1}}function X3($,Z){let Q=$.iterationCount,X={session:{iterations:Q,duration_seconds:$.totalDuration,phase:$.phase},tokens:{input:$.totalInput,output:$.totalOutput,total:$.totalTokens,cost_usd:m($.totalCost,2)},quality:{gates_passed:$.gatesPassed,gates_total:$.gatesTotal,reviews_total:$.reviewsTotal,reviews_approved:$.reviewsApproved,reviews_revision:$.reviewsRevision,gate_failures:$.gateFailures},efficiency:{avg_tokens_per_iteration:Q>0?m($.totalTokens/Q,0):0,avg_cost_per_iteration:Q>0?m($.totalCost/Q,2):0,avg_duration_per_iteration:Q>0?m($.totalDuration/Q,1):0},budget:{used:m($.budgetUsed,2),limit:$.budgetLimit,percent:$.budgetLimit>0?m($.budgetUsed/$.budgetLimit*100,1):0}};if(Z)X.iterations=$.iterations.map((V,U)=>({number:U+1,input_tokens:V.input_tokens??0,output_tokens:V.output_tokens??0,cost_usd:m(V.cost_usd??0,2),duration_seconds:V.duration_seconds??0}));let z=JSON.stringify(X,null,2);function H(V,U){if(!U)return;let B=new RegExp(`("${V}": )(-?\\d+)(,?)$`,"m");z=z.replace(B,(w,L,N,J)=>`${L}${N}.0${J}`)}if(H("avg_duration_per_iteration",Q>0&&Number.isInteger(X.efficiency.avg_duration_per_iteration)),H("percent",$.budgetLimit>0&&Number.isInteger(X.budget.percent)),H("cost_usd",Q>0&&Number.isInteger(X.tokens.cost_usd)),Z)z=z.replace(/("cost_usd": )(-?\d+)(,?)$/gm,(V,U,B,w)=>`${U}${B}.0${w}`);return z}function z3($,Z){let Q=[];if(Q.push("Loki Mode Session Statistics"),Q.push("============================"),Q.push(""),Q.push("Session"),Q.push(` Iterations completed: ${$.iterationCount}`),Q.push(` Duration: ${j1($.totalDuration)}`),Q.push(` Current phase: ${$.phase}`),Q.push(""),Q.push("Token Usage"),$.iterations.length>0)Q.push(` Input tokens: ${s($.totalInput)}`),Q.push(` Output tokens: ${s($.totalOutput)}`),Q.push(` Total tokens: ${s($.totalTokens)}`),Q.push(` Estimated cost: $${i($.totalCost,2)}`);else Q.push(" N/A (no iteration metrics found)");if(Q.push(""),Q.push("Quality Gates"),$.gatesTotal>0){let X=Math.round($.gatesPassed/$.gatesTotal*100);Q.push(` Gates passed: ${$.gatesPassed}/${$.gatesTotal} (${X}%)`)}else Q.push(" Gates passed: N/A");if($.reviewsTotal>0){let X=[];if($.reviewsApproved>0)X.push(`${$.reviewsApproved} approved`);if($.reviewsRevision>0)X.push(`${$.reviewsRevision} revision requested`);let z=X.length>0?X.join(", "):"N/A";Q.push(` Code reviews: ${$.reviewsTotal} (${z})`)}if(Object.keys($.gateFailures).length>0){let X=Object.entries($.gateFailures).filter(([,z])=>z>0).map(([z,H])=>`${z} (${H})`);if(X.length>0)Q.push(` Gate failures: ${X.join(", ")}`)}if(Q.push(""),Q.push("Efficiency"),$.iterationCount>0&&$.iterations.length>0){let X=Math.round($.totalTokens/$.iterationCount),z=$.totalCost/$.iterationCount,H=$.totalDuration/$.iterationCount;Q.push(` Avg tokens/iteration: ${s(X)}`),Q.push(` Avg cost/iteration: $${i(z,2)}`),Q.push(` Avg duration/iteration: ${j1(H)}`)}else Q.push(" N/A (no iteration metrics found)");if(Q.push(""),Q.push("Budget"),$.budgetLimit>0){let X=m($.budgetUsed/$.budgetLimit*100,1),z=Number.isInteger(X)?`${X}.0`:`${X}`;Q.push(` Used: $${i($.budgetUsed,2)} / $${i($.budgetLimit,2)} (${z}%)`)}else if($.budgetUsed>0)Q.push(` Used: $${i($.budgetUsed,2)} (no limit set)`);else Q.push(" N/A");if(Z&&$.iterations.length>0)Q.push(""),Q.push("Per-Iteration Breakdown"),$.iterations.forEach((X,z)=>{let H=z+1,V=_1(s(X.input_tokens??0),10),U=_1(s(X.output_tokens??0),10),B=X.cost_usd??0,w=j1(X.duration_seconds??0),L=_1(`${H}`,3);Q.push(` #${L} input: ${V} output: ${U} cost: $${i(B,2)} time: ${w}`)});return Q.join(`
199
+ `)}function t1($){let Z=!1,Q=!1;for(let V of $)if(V==="--json")Z=!0;else if(V==="--efficiency")Q=!0;let X=P();if(!O1(X)){if(Z)return{exitCode:0,stdout:'{"error": "No active session"}'};return{exitCode:0,stdout:`${O}No active session found.${K}
200
+ Start a session with: loki start <prd>`}}let z=Z3(X);return{exitCode:0,stdout:Z?X3(z,Q):z3(z,Q)}}async function K3($){let Z=t1($);return console.log(Z.stdout),Z.exitCode}var s1=b(()=>{v();l()});var K0={};o(K0,{runDoctor:()=>j3,httpReachable:()=>w1,checkTool:()=>Q0,checkSkills:()=>Z0,checkDisk:()=>I1,buildDoctorJson:()=>z0});import{existsSync as H3,lstatSync as V3,readlinkSync as U3,statfsSync as W3}from"fs";import{homedir as e1}from"os";import{resolve as i1}from"path";function J3($){let Z=$.match(q3);return Z?Z[1]:null}async function G3($){try{let Z=await T([$,"--version"],{timeoutMs:5000}),Q=(Z.stdout||Z.stderr||"").trim();return J3(Q)}catch{return null}}function $0($,Z){let Q=$.split(".").map((z)=>parseInt(z,10)),X=Z.split(".").map((z)=>parseInt(z,10));while(Q.length<2)Q.push(0);while(X.length<2)X.push(0);for(let z=0;z<2;z++){let H=Q[z]??0,V=X[z]??0;if(Number.isNaN(H)||Number.isNaN(V))return 0;if(H!==V)return H-V}return 0}async function Q0($,Z,Q,X=null){let z=await k(Z),H=z!==null,V=H?await G3(Z):null,U="pass";if(!H)U=Q==="required"?"fail":"warn";else if(X&&V){if($0(V,X)<0)U=Q==="required"?"fail":"warn"}return{name:$,command:Z,found:H,version:V,required:Q,min_version:X,status:U,path:z}}function I1(){let $=null;try{let Q=W3(e1()),X=Number(Q.bavail)*Number(Q.bsize);$=Math.round(X/1073741824*10)/10}catch{$=null}let Z="pass";if($!==null){if($<1)Z="fail";else if($<5)Z="warn"}return{available_gb:$,status:Z}}async function w1($,Z=2000){try{return(await fetch($,{signal:AbortSignal.timeout(Z)})).ok}catch{return!1}}async function T1($,Z=!1){let Q=`import ${$}`,X=Z?30000:5000;if(!Z)return(await $1(Q,{timeoutMs:X})).exitCode===0;let z=await r();if(!z)return!1;return(await T([z,"-c",Q],{timeoutMs:X})).exitCode===0}function Z0(){let $=e1();return B3.map(({name:Z,dir:Q})=>{let X=i1($,Q),z=X,H=i1(X,"SKILL.md");if(H3(H))return{name:Z,path:z,status:"pass",detail:""};try{if(V3(X).isSymbolicLink()){let U="unknown";try{U=U3(X)}catch{}return{name:Z,path:z,status:"fail",detail:`(broken symlink -> ${U})`}}}catch{}return{name:Z,path:z,status:"warn",detail:"(not found - run 'loki setup-skill')"}})}async function X0(){return Promise.all(M3.map(async($)=>{return{...await Q0($.jsonName,$.cmd,$.required,$.min??null),displayName:$.displayName}}))}async function z0(){let Z=(await X0()).map(({displayName:V,...U})=>U),Q=I1(),X=0,z=0,H=0;for(let V of Z)if(V.status==="pass")X++;else if(V.status==="fail")z++;else H++;if(Q.status==="pass")X++;else if(Q.status==="fail")z++;else H++;return{checks:Z,disk:Q,summary:{passed:X,failed:z,warnings:H,ok:z===0}}}function W($){switch($){case"pass":return`${R}PASS${K}`;case"fail":return`${j}FAIL${K}`;case"warn":return`${O}WARN${K}`}}function V1($){let Z=$.version?` (v${$.version})`:"",Q=$.displayName;if(!$.found){let X=$.required==="required"?"not found":$.required==="recommended"?"not found (recommended)":"not found (optional)";return` ${W($.status)} ${Q} - ${X}`}if($.min_version&&$.version&&$0($.version,$.min_version)<0){let X=$.required==="required"?"requires":"recommended";return` ${W($.status)} ${Q}${Z} - ${X} >= ${$.min_version}`}return` ${W($.status)} ${Q}${Z}`}function U1($,Z){if(Z==="pass")$.pass++;else if(Z==="fail")$.fail++;else $.warn++}function Y3(){process.stdout.write(`${I}loki doctor${K} - Check system prerequisites
201
201
 
202
202
  `),process.stdout.write(`Usage: loki doctor [--json]
203
203
 
@@ -206,87 +206,121 @@ Start a session with: loki start <prd>`}}let Z=m0(X);return{exitCode:0,stdout:Q?
206
206
 
207
207
  `),process.stdout.write(`Checks: node, python3, jq, git, curl, bash version,
208
208
  `),process.stdout.write(` claude/codex/gemini CLIs, and disk space.
209
- `)}async function z3(){process.stdout.write(`${x}Loki Mode Doctor${K}
209
+ `)}async function A3(){process.stdout.write(`${I}Loki Mode Doctor${K}
210
210
 
211
211
  `),process.stdout.write(`Checking system prerequisites...
212
212
 
213
- `);let z={pass:0,fail:0,warn:0},Q=await s1(),$=new Map(Q.map((Y)=>[Y.command,Y]));process.stdout.write(`${G}Required:${K}
214
- `);for(let Y of["node","python3","jq","git","curl"]){let P=$.get(Y);process.stdout.write(K1(P)+`
215
- `),H1(z,P.status)}process.stdout.write(`
216
- `),process.stdout.write(`${G}AI Providers:${K}
217
- `);let X=["claude","codex","gemini","cline","aider"],Z=!1;for(let Y of X){let P=$.get(Y);if(process.stdout.write(K1(P)+`
218
- `),H1(z,P.status),P.found)Z=!0}if(!Z)process.stdout.write(` ${V("fail")} No AI provider CLI installed -- at least one is required
219
- `),process.stdout.write(` ${_}Install: npm install -g @anthropic-ai/claude-code${K}
220
- `),z.fail++;process.stdout.write(`
221
- `),process.stdout.write(`${G}API Keys:${K}
222
- `);let H=$.get("claude").found,W=$.get("codex").found,U=$.get("gemini").found,J=process.env;if(J.ANTHROPIC_API_KEY)process.stdout.write(` ${V("pass")} ANTHROPIC_API_KEY is set
223
- `),z.pass++;else if(H)process.stdout.write(` ${A} -- ${K} ANTHROPIC_API_KEY not set (Claude CLI uses its own login)
224
- `);if(J.OPENAI_API_KEY)process.stdout.write(` ${V("pass")} OPENAI_API_KEY is set
225
- `),z.pass++;else if(W)process.stdout.write(` ${A} -- ${K} OPENAI_API_KEY not set (Codex CLI uses its own login)
226
- `);if(J.GOOGLE_API_KEY||J.GEMINI_API_KEY)process.stdout.write(` ${V("pass")} GOOGLE_API_KEY is set
227
- `),z.pass++;else if(U)process.stdout.write(` ${A} -- ${K} GOOGLE_API_KEY not set (Gemini CLI uses its own login)
213
+ `);let $={pass:0,fail:0,warn:0},Z=await X0(),Q=new Map(Z.map((Y)=>[Y.command,Y]));process.stdout.write(`${q}Required:${K}
214
+ `);for(let Y of["node","python3","jq","git","curl"]){let x=Q.get(Y);process.stdout.write(V1(x)+`
215
+ `),U1($,x.status)}process.stdout.write(`
216
+ `),process.stdout.write(`${q}AI Providers:${K}
217
+ `);let X=["claude","codex","gemini","cline","aider"],z=!1;for(let Y of X){let x=Q.get(Y);if(process.stdout.write(V1(x)+`
218
+ `),U1($,x.status),x.found)z=!0}if(!z)process.stdout.write(` ${W("fail")} No AI provider CLI installed -- at least one is required
219
+ `),process.stdout.write(` ${O}Install: npm install -g @anthropic-ai/claude-code${K}
220
+ `),$.fail++;process.stdout.write(`
221
+ `),process.stdout.write(`${q}API Keys:${K}
222
+ `);let H=Q.get("claude").found,V=Q.get("codex").found,U=Q.get("gemini").found,B=process.env;if(B.ANTHROPIC_API_KEY)process.stdout.write(` ${W("pass")} ANTHROPIC_API_KEY is set
223
+ `),$.pass++;else if(H)process.stdout.write(` ${_} -- ${K} ANTHROPIC_API_KEY not set (Claude CLI uses its own login)
224
+ `);if(B.OPENAI_API_KEY)process.stdout.write(` ${W("pass")} OPENAI_API_KEY is set
225
+ `),$.pass++;else if(V)process.stdout.write(` ${_} -- ${K} OPENAI_API_KEY not set (Codex CLI uses its own login)
226
+ `);if(B.GOOGLE_API_KEY||B.GEMINI_API_KEY)process.stdout.write(` ${W("pass")} GOOGLE_API_KEY is set
227
+ `),$.pass++;else if(U)process.stdout.write(` ${_} -- ${K} GOOGLE_API_KEY not set (Gemini CLI uses its own login)
228
228
  `);process.stdout.write(`
229
- `),process.stdout.write(`${G}Skills:${K}
230
- `);for(let Y of a1())if(Y.status==="pass")process.stdout.write(` ${V("pass")} ${Y.name} ${A}${Y.path}${K}
231
- `),z.pass++;else if(Y.status==="fail")process.stdout.write(` ${V("fail")} ${Y.name} ${A}${Y.detail}${K}
232
- `),process.stdout.write(` ${_}Fix: loki setup-skill${K}
233
- `),z.fail++;else process.stdout.write(` ${V("warn")} ${Y.name} ${A}${Y.detail}${K}
234
- `),z.warn++;if(process.stdout.write(`
235
- `),process.stdout.write(`${G}Integrations:${K}
236
- `),await Y1("mcp"))process.stdout.write(` ${V("pass")} MCP SDK (Python)
237
- `),z.pass++;else process.stdout.write(` ${V("warn")} MCP SDK - not installed (pip3 install mcp)
238
- `),z.warn++;if(await Y1("numpy",!0))process.stdout.write(` ${V("pass")} numpy (vector search)
239
- `),z.pass++;else process.stdout.write(` ${V("warn")} numpy - not installed (pip3 install numpy)
240
- `),z.warn++;if(await Y1("sentence_transformers",!0))process.stdout.write(` ${V("pass")} sentence-transformers (embeddings)
241
- `),z.pass++;else process.stdout.write(` ${V("warn")} sentence-transformers - not installed (loki memory vectors setup)
242
- `),z.warn++;if(await A1("http://localhost:8100/api/v2/heartbeat"))process.stdout.write(` ${V("pass")} ChromaDB server (port 8100)
243
- `),z.pass++;else process.stdout.write(` ${V("warn")} ChromaDB - not running (docker start loki-chroma)
244
- `),z.warn++;let O=process.env.LOKI_MIROFISH_URL;if(O)if(await A1(`${O}/health`))process.stdout.write(` ${V("pass")} MiroFish server (${O})
245
- `),z.pass++;else process.stdout.write(` ${V("warn")} MiroFish - not running (loki start --mirofish-docker <image>)
246
- `),z.warn++;if(process.env.LOKI_OTEL_ENDPOINT)process.stdout.write(` ${V("pass")} OTEL endpoint: ${process.env.LOKI_OTEL_ENDPOINT}
247
- `),z.pass++;else process.stdout.write(` ${V("warn")} OTEL - not configured (set LOKI_OTEL_ENDPOINT)
248
- `),z.warn++;process.stdout.write(`
249
- `),process.stdout.write(`${G}System:${K}
250
- `);let T=$.get("bash");process.stdout.write(K1(T)+`
251
- `),H1(z,T.status);let E=$.get("bun");if(E)process.stdout.write(K1(E)+`
252
- `),H1(z,E.status);let B=O1(),M=B.available_gb===null?null:Math.floor(B.available_gb);if(M===null)process.stdout.write(` ${V("warn")} Disk space: unable to determine
253
- `),z.warn++;else if(B.status==="fail")process.stdout.write(` ${V("fail")} Disk space: ${M}GB available (need >= 1GB)
254
- `),z.fail++;else if(B.status==="warn")process.stdout.write(` ${V("warn")} Disk space: ${M}GB available (low)
255
- `),z.warn++;else process.stdout.write(` ${V("pass")} Disk space: ${M}GB available
256
- `),z.pass++;process.stdout.write(`
257
- `),process.stdout.write(`${G}Runtime route:${K}
258
- `);let I=process.versions.bun!==void 0,N=process.argv[0]??"(unknown)";if(process.stdout.write(` ${V("pass")} Active runtime: ${I?"Bun":"Node"} (${N})
259
- `),process.env.LOKI_LEGACY_BASH==="1"||process.env.LOKI_LEGACY_BASH==="true")process.stdout.write(` ${V("warn")} LOKI_LEGACY_BASH set: shim routes every command to autonomy/loki (bash)
260
- `);if(process.env.LOKI_TS_ENTRY)process.stdout.write(` ${V("pass")} LOKI_TS_ENTRY override: ${process.env.LOKI_TS_ENTRY}
261
- `);if(process.env.BUN_FROM_SOURCE==="1"||process.env.BUN_FROM_SOURCE==="true")process.stdout.write(` ${V("pass")} BUN_FROM_SOURCE set: shim prefers loki-ts/src/ over dist/
229
+ `),process.stdout.write(`${q}Skills:${K}
230
+ `);for(let Y of Z0())if(Y.status==="pass")process.stdout.write(` ${W("pass")} ${Y.name} ${_}${Y.path}${K}
231
+ `),$.pass++;else if(Y.status==="fail")process.stdout.write(` ${W("fail")} ${Y.name} ${_}${Y.detail}${K}
232
+ `),process.stdout.write(` ${O}Fix: loki setup-skill${K}
233
+ `),$.fail++;else process.stdout.write(` ${W("warn")} ${Y.name} ${_}${Y.detail}${K}
234
+ `),$.warn++;if(process.stdout.write(`
235
+ `),process.stdout.write(`${q}Integrations:${K}
236
+ `),await T1("mcp"))process.stdout.write(` ${W("pass")} MCP SDK (Python)
237
+ `),$.pass++;else process.stdout.write(` ${W("warn")} MCP SDK - not installed (pip3 install mcp)
238
+ `),$.warn++;if(await T1("numpy",!0))process.stdout.write(` ${W("pass")} numpy (vector search)
239
+ `),$.pass++;else process.stdout.write(` ${W("warn")} numpy - not installed (pip3 install numpy)
240
+ `),$.warn++;if(await T1("sentence_transformers",!0))process.stdout.write(` ${W("pass")} sentence-transformers (embeddings)
241
+ `),$.pass++;else process.stdout.write(` ${W("warn")} sentence-transformers - not installed (loki memory vectors setup)
242
+ `),$.warn++;if(await w1("http://localhost:8100/api/v2/heartbeat"))process.stdout.write(` ${W("pass")} ChromaDB server (port 8100)
243
+ `),$.pass++;else process.stdout.write(` ${W("warn")} ChromaDB - not running (docker start loki-chroma)
244
+ `),$.warn++;let w=process.env.LOKI_MIROFISH_URL;if(w)if(await w1(`${w}/health`))process.stdout.write(` ${W("pass")} MiroFish server (${w})
245
+ `),$.pass++;else process.stdout.write(` ${W("warn")} MiroFish - not running (loki start --mirofish-docker <image>)
246
+ `),$.warn++;if(process.env.LOKI_OTEL_ENDPOINT)process.stdout.write(` ${W("pass")} OTEL endpoint: ${process.env.LOKI_OTEL_ENDPOINT}
247
+ `),$.pass++;else process.stdout.write(` ${W("warn")} OTEL - not configured (set LOKI_OTEL_ENDPOINT)
248
+ `),$.warn++;process.stdout.write(`
249
+ `),process.stdout.write(`${q}System:${K}
250
+ `);let L=Q.get("bash");process.stdout.write(V1(L)+`
251
+ `),U1($,L.status);let N=Q.get("bun");if(N)process.stdout.write(V1(N)+`
252
+ `),U1($,N.status);let J=I1(),A=J.available_gb===null?null:Math.floor(J.available_gb);if(A===null)process.stdout.write(` ${W("warn")} Disk space: unable to determine
253
+ `),$.warn++;else if(J.status==="fail")process.stdout.write(` ${W("fail")} Disk space: ${A}GB available (need >= 1GB)
254
+ `),$.fail++;else if(J.status==="warn")process.stdout.write(` ${W("warn")} Disk space: ${A}GB available (low)
255
+ `),$.warn++;else process.stdout.write(` ${W("pass")} Disk space: ${A}GB available
256
+ `),$.pass++;process.stdout.write(`
257
+ `),process.stdout.write(`${q}Runtime route:${K}
258
+ `);let F=process.versions.bun!==void 0,f=process.argv[0]??"(unknown)";if(process.stdout.write(` ${W("pass")} Active runtime: ${F?"Bun":"Node"} (${f})
259
+ `),process.env.LOKI_LEGACY_BASH==="1"||process.env.LOKI_LEGACY_BASH==="true")process.stdout.write(` ${W("warn")} LOKI_LEGACY_BASH set: shim routes every command to autonomy/loki (bash)
260
+ `);if(process.env.LOKI_TS_ENTRY)process.stdout.write(` ${W("pass")} LOKI_TS_ENTRY override: ${process.env.LOKI_TS_ENTRY}
261
+ `);if(process.env.BUN_FROM_SOURCE==="1"||process.env.BUN_FROM_SOURCE==="true")process.stdout.write(` ${W("pass")} BUN_FROM_SOURCE set: shim prefers loki-ts/src/ over dist/
262
+ `);let C=await r();if(C!==null){let x=(await T([C,"-c","import sys; print(f'{sys.version_info.major}.{sys.version_info.minor}')"],{timeoutMs:5000})).stdout.trim();if(x.startsWith("3.12"))process.stdout.write(` ${W("pass")} Python 3.12 (chromadb / sentence-transformers): ${x} at ${C}
263
+ `);else if(x)process.stdout.write(` ${W("warn")} Python 3.12 NOT found -- using ${x} at ${C}; chromadb / sentence-transformers may fail. Install python3.12 (brew install python@3.12 / apt install python3.12).
264
+ `);else process.stdout.write(` ${W("warn")} Python 3 found at ${C} but version probe failed; chromadb may not work.
265
+ `)}else process.stdout.write(` ${W("warn")} Python 3 not on PATH -- memory + MCP integrations disabled.
262
266
  `);if(process.stdout.write(`
263
- `),process.stdout.write(`${x}Summary:${K} ${k}${z.pass} passed${K}, ${D}${z.fail} failed${K}, ${_}${z.warn} warnings${K}
267
+ `),process.stdout.write(`${I}Summary:${K} ${R}${$.pass} passed${K}, ${j}${$.fail} failed${K}, ${O}${$.warn} warnings${K}
264
268
 
265
- `),z.fail>0)return process.stdout.write(`${D}Some required prerequisites are missing.${K}
269
+ `),$.fail>0)return process.stdout.write(`${j}Some required prerequisites are missing.${K}
266
270
  `),process.stdout.write(`Install missing dependencies and run 'loki doctor' again.
267
- `),1;if(z.warn>0)return process.stdout.write(`${_}All required checks passed with some warnings.${K}
268
- `),0;return process.stdout.write(`${k}All checks passed. System is ready for Loki Mode.${K}
269
- `),0}async function $3(z){let Q=!1;for(let $ of z)if($==="--json")Q=!0;else if($==="--help"||$==="-h")return e0(),0;else return process.stderr.write(`${D}Unknown option: ${$}${K}
271
+ `),1;if($.warn>0)return process.stdout.write(`${O}All required checks passed with some warnings.${K}
272
+ `),0;return process.stdout.write(`${R}All checks passed. System is ready for Loki Mode.${K}
273
+ `),0}async function j3($){let Z=!1;for(let Q of $)if(Q==="--json")Z=!0;else if(Q==="--help"||Q==="-h")return Y3(),0;else return process.stderr.write(`${j}Unknown option: ${Q}${K}
270
274
  `),process.stderr.write(`Usage: loki doctor [--json]
271
- `),1;if(Q){let $=await i1();return process.stdout.write(JSON.stringify($,null,2)+`
272
- `),0}return z3()}var r0,s0,i0;var z0=h(()=>{p();Q1();l();r0=/(\d+\.\d+(?:\.\d+)*)/;s0=[{name:"Claude Code",dir:".claude/skills/loki-mode"},{name:"Codex CLI",dir:".codex/skills/loki-mode"},{name:"Gemini CLI",dir:".gemini/skills/loki-mode"},{name:"Cline CLI",dir:".cline/skills/loki-mode"},{name:"Aider CLI",dir:".aider/skills/loki-mode"}];i0=[{displayName:"Node.js (>= 18)",jsonName:"Node.js",cmd:"node",required:"required",min:"18.0"},{displayName:"Python 3 (>= 3.8)",jsonName:"Python 3",cmd:"python3",required:"required",min:"3.8"},{displayName:"jq",jsonName:"jq",cmd:"jq",required:"required"},{displayName:"git",jsonName:"git",cmd:"git",required:"required"},{displayName:"curl",jsonName:"curl",cmd:"curl",required:"required"},{displayName:"bash (>= 4.0)",jsonName:"bash",cmd:"bash",required:"recommended",min:"4.0"},{displayName:"Bun (>= 1.3)",jsonName:"Bun",cmd:"bun",required:"recommended",min:"1.3"},{displayName:"Claude CLI",jsonName:"Claude CLI",cmd:"claude",required:"optional"},{displayName:"Codex CLI",jsonName:"Codex CLI",cmd:"codex",required:"optional"},{displayName:"Gemini CLI",jsonName:"Gemini CLI",cmd:"gemini",required:"optional"},{displayName:"Cline CLI",jsonName:"Cline CLI",cmd:"cline",required:"optional"},{displayName:"Aider CLI",jsonName:"Aider CLI",cmd:"aider",required:"optional"}]});m();import{readFileSync as V0}from"fs";import{resolve as B0,dirname as q0}from"path";import{fileURLToPath as G0}from"url";var u=null;function k1(){if(u!==null)return u;let z="7.5.1";if(typeof z==="string"&&z.length>0)return u=z,u;try{let Q=q0(G0(import.meta.url)),$=V1(Q);u=V0(B0($,"VERSION"),"utf-8").trim()}catch{u="unknown"}return u}function R1(){return process.stdout.write(`Loki Mode v${k1()}
273
- `),0}p();l();m();import{readFileSync as A0,existsSync as O0}from"fs";import{resolve as _0}from"path";var T0=["claude","codex","gemini","cline","aider"];function D1(){let z=_0(C(),"state","provider");if(!O0(z))return"";try{return A0(z,"utf-8").trim()}catch{return""}}function I0(z,Q){return z||Q||process.env.LOKI_PROVIDER||"claude"}function w0(z){let Q=D1(),$=I0(z,Q);switch(process.stdout.write(`${x}Current Provider${K}
275
+ `),1;if(Z){let Q=await z0();return process.stdout.write(JSON.stringify(Q,null,2)+`
276
+ `),0}return A3()}var q3,B3,M3;var H0=b(()=>{y();z1();l();q3=/(\d+\.\d+(?:\.\d+)*)/;B3=[{name:"Claude Code",dir:".claude/skills/loki-mode"},{name:"Codex CLI",dir:".codex/skills/loki-mode"},{name:"Gemini CLI",dir:".gemini/skills/loki-mode"},{name:"Cline CLI",dir:".cline/skills/loki-mode"},{name:"Aider CLI",dir:".aider/skills/loki-mode"}];M3=[{displayName:"Node.js (>= 18)",jsonName:"Node.js",cmd:"node",required:"required",min:"18.0"},{displayName:"Python 3 (>= 3.8)",jsonName:"Python 3",cmd:"python3",required:"required",min:"3.8"},{displayName:"jq",jsonName:"jq",cmd:"jq",required:"required"},{displayName:"git",jsonName:"git",cmd:"git",required:"required"},{displayName:"curl",jsonName:"curl",cmd:"curl",required:"required"},{displayName:"bash (>= 4.0)",jsonName:"bash",cmd:"bash",required:"recommended",min:"4.0"},{displayName:"Bun (>= 1.3)",jsonName:"Bun",cmd:"bun",required:"recommended",min:"1.3"},{displayName:"Claude CLI",jsonName:"Claude CLI",cmd:"claude",required:"optional"},{displayName:"Codex CLI",jsonName:"Codex CLI",cmd:"codex",required:"optional"},{displayName:"Gemini CLI",jsonName:"Gemini CLI",cmd:"gemini",required:"optional"},{displayName:"Cline CLI",jsonName:"Cline CLI",cmd:"cline",required:"optional"},{displayName:"Aider CLI",jsonName:"Aider CLI",cmd:"aider",required:"optional"}]});import{existsSync as W1,mkdirSync as _3,copyFileSync as O3,readFileSync as T3,readdirSync as w3,statSync as I3,writeFileSync as S6,renameSync as L3,appendFileSync as D6,rmSync as E6}from"fs";import{join as d,dirname as F3}from"path";function q1($){return d($,"state","checkpoints")}function P3($){let Z=q1($);if(!W1(Z))return[];return w3(Z).filter((Q)=>Q.startsWith("cp-")).filter((Q)=>{try{return I3(d(Z,Q)).isDirectory()}catch{return!1}})}function R3($){return[...$].sort((Z,Q)=>{let X=V0(Z),z=V0(Q);return X-z})}function V0($){let Z=$.split("-");if(Z.length<3)return 0;let Q=Z[Z.length-1],X=Number.parseInt(Q??"0",10);return Number.isFinite(X)?X:0}function F1($){let Z=$??P(),Q=R3(P3(Z)),X=[];for(let z of Q){let H=U0(Z,z);if(H)X.push(H)}return X}function U0($,Z){let Q=d(q1($),Z,"metadata.json");if(!W1(Q))return null;try{return JSON.parse(T3(Q,"utf-8"))}catch{return null}}function x1($,Z){if(!S3.test($))throw new W0($);let Q=Z??P(),X=d(q1(Q),$);if(!W1(X))throw new L1($);let z=U0(Q,$);if(!z)throw new L1($);return z}function q0($,Z){let Q=x1($,Z),X=Z??P(),z=d(q1(X),$),H=[];for(let V of D3){let U=d(z,V);if(!W1(U))continue;H.push({from:U,to:d(X,V)})}return{id:$,metadata:Q,restore:H}}function J0($){let Z=[],Q=0;for(let X of $.restore)try{_3(F3(X.to),{recursive:!0});let z=`${X.to}.tmp.${process.pid}.${++x3}`;O3(X.from,z),L3(z,X.to),Q+=1}catch(z){Z.push(`${X.from} -> ${X.to}: ${z.message}`)}return{restored:Q,errors:Z}}var C6,x3=0,S3,L1,W0,D3;var G0=b(()=>{v();y();C6=Promise.resolve();S3=/^[a-zA-Z0-9_-]+$/;L1=class L1 extends Error{id;constructor($){super(`Checkpoint not found: ${$}`);this.id=$;this.name="CheckpointNotFoundError"}};W0=class W0 extends Error{id;constructor($){super(`Invalid checkpoint ID: must be alphanumeric, hyphens, underscores only (got: ${$})`);this.id=$;this.name="InvalidCheckpointIdError"}};D3=["state/orchestrator.json","queue/pending.json","queue/completed.json","queue/in-progress.json","queue/current-task.json"]});var Y0={};o(Y0,{runRollback:()=>E3});async function E3($){let Z=$[0],Q=$.slice(1);if(Z===void 0||Z==="help"||Z==="--help"||Z==="-h")return process.stdout.write(B0),Z===void 0?1:0;switch(Z){case"list":{let X=[...F1()].reverse();if(X.length===0)return process.stdout.write(`${O}No checkpoints found.${K}
277
+ `),0;process.stdout.write(`${I}Checkpoints${K} (${X.length}, newest first):
278
+ `);for(let z of X)process.stdout.write(` ${q}${z.id}${K} iter=${z.iteration} ${z.git_branch||"(no branch)"}@${(z.git_sha||"").slice(0,7)} ${z.timestamp}
279
+ `);return 0}case"show":{let X=Q[0];if(!X)return process.stderr.write(`${j}Missing checkpoint id.${K} Use \`loki rollback list\`.
280
+ `),2;try{let z=x1(X);return process.stdout.write(`${JSON.stringify(z,null,2)}
281
+ `),0}catch(z){return process.stderr.write(`${j}Failed to read checkpoint:${K} ${z.message}
282
+ `),1}}case"to":{let X=Q[0];if(!X)return process.stderr.write(`${j}Missing checkpoint id.${K} Use \`loki rollback list\`.
283
+ `),2;return M0(X)}case"latest":{let X=F1(),z=X[X.length-1];if(!z)return process.stderr.write(`${j}No checkpoints found to roll back to.${K}
284
+ `),1;return process.stdout.write(`Rolling back to latest checkpoint: ${q}${z.id}${K}
285
+ `),M0(z.id)}default:return process.stderr.write(`Unknown subcommand: ${Z}
286
+ `),process.stderr.write(B0),2}}function M0($){let Z;try{Z=q0($)}catch(X){return process.stderr.write(`${j}Cannot plan rollback:${K} ${X.message}
287
+ `),1}if(Z.restore.length===0)return process.stdout.write(`${O}Checkpoint ${$} has no restorable state files; nothing to do.${K}
288
+ `),0;let Q=J0(Z);if(Q.errors.length>0){for(let X of Q.errors)process.stderr.write(`${j}restore error:${K} ${X}
289
+ `);return process.stderr.write(`${j}Partial rollback: ${Q.restored}/${Z.restore.length} files restored.${K}
290
+ `),1}return process.stdout.write(`${R}Rolled back ${Q.restored}/${Z.restore.length} state files from ${$}.${K}
291
+ `),process.stdout.write("Run `loki start` to resume from the restored state.\n"),0}var B0=`Usage: loki rollback <subcommand>
292
+
293
+ Subcommands:
294
+ list List checkpoints (newest first)
295
+ show <id> Print metadata for one checkpoint
296
+ to <id> Restore .loki/ state files to that checkpoint
297
+ latest Restore to the most recent checkpoint
298
+
299
+ Restored files (matches autonomy/run.sh:7028 byte-for-byte):
300
+ .loki/state/orchestrator.json
301
+ .loki/queue/{pending,completed,in-progress,current-task}.json
302
+
303
+ Note: only state files are restored. Source code, git history, and the
304
+ session's autonomy-state.json are unchanged. Re-run \`loki start\` to
305
+ resume from the restored state.
306
+ `;var A0=b(()=>{G0();l()});v();import{readFileSync as x0}from"fs";import{resolve as P0,dirname as R0}from"path";import{fileURLToPath as S0}from"url";var c=null;function f1(){if(c!==null)return c;let $="7.5.2";if(typeof $==="string"&&$.length>0)return c=$,c;try{let Z=R0(S0(import.meta.url)),Q=B1(Z);c=x0(P0(Q,"VERSION"),"utf-8").trim()}catch{c="unknown"}return c}function C1(){return process.stdout.write(`Loki Mode v${f1()}
307
+ `),0}y();l();v();import{readFileSync as f0,existsSync as C0}from"fs";import{resolve as g0}from"path";var b0=["claude","codex","gemini","cline","aider"];function b1(){let $=g0(P(),"state","provider");if(!C0($))return"";try{return f0($,"utf-8").trim()}catch{return""}}function v0($,Z){return $||Z||process.env.LOKI_PROVIDER||"claude"}function y0($){let Z=b1(),Q=v0($,Z);switch(process.stdout.write(`${I}Current Provider${K}
274
308
  `),process.stdout.write(`
275
- `),process.stdout.write(`${G}Provider:${K} ${$}
276
- `),$){case"claude":process.stdout.write(`${k}Status:${K} Full features (subagents, parallel, MCP)
277
- `);break;case"cline":process.stdout.write(`${k}Status:${K} Near-full mode (subagents, MCP, 12+ providers)
278
- `);break;case"codex":case"gemini":case"aider":process.stdout.write(`${_}Status:${K} Degraded mode (sequential only)
279
- `);break;default:break}if(Q)process.stdout.write(`${A}(saved in .loki/state/provider)${K}
280
- `);else process.stdout.write(`${A}(default - not explicitly set)${K}
309
+ `),process.stdout.write(`${q}Provider:${K} ${Q}
310
+ `),Q){case"claude":process.stdout.write(`${R}Status:${K} Full features (subagents, parallel, MCP)
311
+ `);break;case"cline":process.stdout.write(`${R}Status:${K} Near-full mode (subagents, MCP, 12+ providers)
312
+ `);break;case"codex":case"gemini":case"aider":process.stdout.write(`${O}Status:${K} Degraded mode (sequential only)
313
+ `);break;default:break}if(Z)process.stdout.write(`${_}(saved in .loki/state/provider)${K}
314
+ `);else process.stdout.write(`${_}(default - not explicitly set)${K}
281
315
  `);return process.stdout.write(`
282
- `),process.stdout.write(`Switch provider: ${G}loki provider set <name>${K}
283
- `),process.stdout.write(`Available: ${G}loki provider list${K}
284
- `),0}async function F0(){let Q=D1()||process.env.LOKI_PROVIDER||"claude";process.stdout.write(`${x}Available Providers${K}
316
+ `),process.stdout.write(`Switch provider: ${q}loki provider set <name>${K}
317
+ `),process.stdout.write(`Available: ${q}loki provider list${K}
318
+ `),0}async function h0(){let Z=b1()||process.env.LOKI_PROVIDER||"claude";process.stdout.write(`${I}Available Providers${K}
285
319
  `),process.stdout.write(`
286
- `);let $=await Promise.all(T0.map(async(H)=>[H,await S(H)!==null])),X=new Map;for(let[H,W]of $)X.set(H,W?`${k}installed${K}`:`${D}not installed${K}`);let Z=[["claude","claude - Claude Code (Anthropic) "],["codex","codex - Codex CLI (OpenAI) "],["gemini","gemini - Gemini CLI (Google) "],["cline","cline - Cline (multi-provider) "],["aider","aider - Aider (terminal pair prog) "]];for(let[H,W]of Z){let U=Q===H?` ${G}(current)${K}`:"";process.stdout.write(` ${W} ${X.get(H)}${U}
320
+ `);let Q=await Promise.all(b0.map(async(H)=>[H,await k(H)!==null])),X=new Map;for(let[H,V]of Q)X.set(H,V?`${R}installed${K}`:`${j}not installed${K}`);let z=[["claude","claude - Claude Code (Anthropic) "],["codex","codex - Codex CLI (OpenAI) "],["gemini","gemini - Gemini CLI (Google) "],["cline","cline - Cline (multi-provider) "],["aider","aider - Aider (terminal pair prog) "]];for(let[H,V]of z){let U=Z===H?` ${q}(current)${K}`:"";process.stdout.write(` ${V} ${X.get(H)}${U}
287
321
  `)}return process.stdout.write(`
288
- `),process.stdout.write(`Set provider: ${G}loki provider set <name>${K}
289
- `),0}function x0(){return process.stdout.write(`${x}Loki Mode Provider Management${K}
322
+ `),process.stdout.write(`Set provider: ${q}loki provider set <name>${K}
323
+ `),0}function u0(){return process.stdout.write(`${I}Loki Mode Provider Management${K}
290
324
  `),process.stdout.write(`
291
325
  `),process.stdout.write(`Usage: loki provider <command>
292
326
  `),process.stdout.write(`
@@ -304,17 +338,17 @@ Start a session with: loki start <prd>`}}let Z=m0(X);return{exitCode:0,stdout:Q?
304
338
  `),process.stdout.write(` loki provider list
305
339
  `),process.stdout.write(` loki provider info gemini
306
340
  `),process.stdout.write(` loki provider models
307
- `),0}async function E1(z){let Q=z[0]??"show",$=z.slice(1);switch(Q){case"show":case"current":return w0($[0]);case"list":return F0();case"set":case"info":case"models":return L0(["provider",Q,...$]);default:return x0()}}async function L0(z){let{run:Q}=await Promise.resolve().then(() => (p(),S1)),{resolve:$}=await import("path"),{REPO_ROOT:X}=await Promise.resolve().then(() => (m(),P1)),Z=$(X,"autonomy","loki"),H=await Q([Z,...z],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(H.stdout),process.stderr.write(H.stderr),H.exitCode}l();m();Q1();p();import{existsSync as N1,readFileSync as k0}from"fs";import{resolve as d}from"path";import{mkdir as R0}from"fs/promises";var e=d(B1(),"learnings");function G1(z){if(!N1(z))return 0;try{let Q=k0(z,"utf-8"),$=0;for(let X of Q.split(`
308
- `))if(X.includes('"description"'))$++;return $}catch{return 0}}async function S0(){await R0(e,{recursive:!0});let z=G1(d(e,"patterns.jsonl")),Q=G1(d(e,"mistakes.jsonl")),$=G1(d(e,"successes.jsonl"));return process.stdout.write(`${x}Cross-Project Learnings${K}
341
+ `),0}async function v1($){let Z=$[0]??"show",Q=$.slice(1);switch(Z){case"show":case"current":return y0(Q[0]);case"list":return h0();case"set":case"info":case"models":return m0(["provider",Z,...Q]);default:return u0()}}async function m0($){let{run:Z}=await Promise.resolve().then(() => (y(),g1)),{resolve:Q}=await import("path"),{REPO_ROOT:X}=await Promise.resolve().then(() => (v(),N1)),z=Q(X,"autonomy","loki"),H=await Z([z,...$],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(H.stdout),process.stderr.write(H.stderr),H.exitCode}l();v();z1();y();import{existsSync as y1,readFileSync as c0}from"fs";import{resolve as t}from"path";import{mkdir as l0}from"fs/promises";var Q1=t(M1(),"learnings");function A1($){if(!y1($))return 0;try{let Z=c0($,"utf-8"),Q=0;for(let X of Z.split(`
342
+ `))if(X.includes('"description"'))Q++;return Q}catch{return 0}}async function d0(){await l0(Q1,{recursive:!0});let $=A1(t(Q1,"patterns.jsonl")),Z=A1(t(Q1,"mistakes.jsonl")),Q=A1(t(Q1,"successes.jsonl"));return process.stdout.write(`${I}Cross-Project Learnings${K}
309
343
  `),process.stdout.write(`
310
- `),process.stdout.write(` Patterns: ${k}${z}${K}
311
- `),process.stdout.write(` Mistakes: ${_}${Q}${K}
312
- `),process.stdout.write(` Successes: ${G}${$}${K}
344
+ `),process.stdout.write(` Patterns: ${R}${$}${K}
345
+ `),process.stdout.write(` Mistakes: ${O}${Z}${K}
346
+ `),process.stdout.write(` Successes: ${q}${Q}${K}
313
347
  `),process.stdout.write(`
314
- `),process.stdout.write(`Location: ${e}
348
+ `),process.stdout.write(`Location: ${Q1}
315
349
  `),process.stdout.write(`
316
350
  `),process.stdout.write(`Use 'loki memory show <type>' to view entries
317
- `),0}async function D0(z){if(z){let X=`
351
+ `),0}async function o0($){if($){let X=`
318
352
  try:
319
353
  from memory.layers import IndexLayer
320
354
  layer = IndexLayer('.loki/memory')
@@ -324,9 +358,9 @@ except ImportError:
324
358
  print('Error: memory.layers module not found')
325
359
  except Exception as e:
326
360
  print(f'Error: {e}')
327
- `.trim(),Z=await i(X,{cwd:v});return process.stdout.write(Z.stdout),0}let Q=d(C(),"memory","index.json");if(!N1(Q))return process.stdout.write(`No index found
328
- `),0;let $=await i(`import json, sys; sys.stdout.write(json.dumps(json.load(open(${JSON.stringify(Q)})), indent=4) + "\\n")`);if($.exitCode!==0)return process.stdout.write(`No index found
329
- `),0;return process.stdout.write($.stdout),0}async function b1(z){switch(z[0]??"list"){case"list":case"ls":return S0();case"index":return D0(z[1]==="rebuild");default:{let $=d(v,"autonomy","loki"),X=await w([$,"memory",...z],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(X.stdout),process.stderr.write(X.stderr),X.exitCode}}}var $0=`Loki Mode (TypeScript port, Phase 2 of bash->Bun migration)
361
+ `.trim(),z=await $1(X,{cwd:p});return process.stdout.write(z.stdout),0}let Z=t(P(),"memory","index.json");if(!y1(Z))return process.stdout.write(`No index found
362
+ `),0;let Q=await $1(`import json, sys; sys.stdout.write(json.dumps(json.load(open(${JSON.stringify(Z)})), indent=4) + "\\n")`);if(Q.exitCode!==0)return process.stdout.write(`No index found
363
+ `),0;return process.stdout.write(Q.stdout),0}async function h1($){switch($[0]??"list"){case"list":case"ls":return d0();case"index":return o0($[1]==="rebuild");default:{let Q=t(p,"autonomy","loki"),X=await T([Q,"memory",...$],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(X.stdout),process.stderr.write(X.stderr),X.exitCode}}}var j0=`Loki Mode (TypeScript port, Phase 2 of bash->Bun migration)
330
364
 
331
365
  Usage: loki <command> [args...]
332
366
 
@@ -339,10 +373,12 @@ Phase 2 ported (Bun-native, fast):
339
373
  memory list Cross-project learnings counts
340
374
  memory index [rebuild] Show or rebuild memory index
341
375
  doctor [--json] System prerequisites health check
376
+ rollback <subcmd> Restore .loki/ state from a checkpoint
377
+ (subcmds: list | show <id> | to <id> | latest)
342
378
 
343
379
  All other commands fall through to the bash CLI (autonomy/loki).
344
380
  Set LOKI_LEGACY_BASH=1 to force the bash CLI for every command.
345
- `;async function Q3(z){let Q=z[0],$=z.slice(1);switch(Q){case void 0:case"help":case"--help":case"-h":return process.stdout.write($0),0;case"version":case"--version":case"-v":return R1();case"provider":return E1($);case"memory":return b1($);case"status":{let{runStatus:X}=await Promise.resolve().then(() => (m1(),v1));return X($)}case"stats":{let{runStats:X}=await Promise.resolve().then(() => (d1(),l1));return X($)}case"doctor":{let{runDoctor:X}=await Promise.resolve().then(() => (z0(),e1));return X($)}default:return process.stderr.write(`Unknown command: ${Q}
346
- `),process.stderr.write($0),2}}process.on("SIGINT",()=>process.exit(130));process.on("SIGTERM",()=>process.exit(143));var X3=await Q3(Bun.argv.slice(2));process.exit(X3);
381
+ `;async function k3($){let Z=$[0],Q=$.slice(1);switch(Z){case void 0:case"help":case"--help":case"-h":return process.stdout.write(j0),0;case"version":case"--version":case"-v":return C1();case"provider":return v1(Q);case"memory":return h1(Q);case"status":{let{runStatus:X}=await Promise.resolve().then(() => (o1(),d1));return X(Q)}case"stats":{let{runStats:X}=await Promise.resolve().then(() => (s1(),a1));return X(Q)}case"doctor":{let{runDoctor:X}=await Promise.resolve().then(() => (H0(),K0));return X(Q)}case"rollback":{let{runRollback:X}=await Promise.resolve().then(() => (A0(),Y0));return X(Q)}default:return process.stderr.write(`Unknown command: ${Z}
382
+ `),process.stderr.write(j0),2}}process.on("SIGINT",()=>process.exit(130));process.on("SIGTERM",()=>process.exit(143));var N3=await k3(Bun.argv.slice(2));process.exit(N3);
347
383
 
348
- //# debugId=CB3CC6615374C70C64756E2164756E21
384
+ //# debugId=F6986F139DFEE70364756E2164756E21
package/mcp/__init__.py CHANGED
@@ -57,4 +57,4 @@ try:
57
57
  except ImportError:
58
58
  __all__ = ['mcp']
59
59
 
60
- __version__ = '7.5.1'
60
+ __version__ = '7.5.2'
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "loki-mode",
3
- "version": "7.5.1",
3
+ "version": "7.5.2",
4
4
  "description": "Loki Mode by Autonomi - Multi-agent autonomous startup system for Claude Code, Codex CLI, and Gemini CLI",
5
5
  "keywords": [
6
6
  "agent",