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