loki-mode 7.5.4 → 7.5.5
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 +3 -3
- package/VERSION +1 -1
- package/autonomy/loki +60 -0
- package/dashboard/__init__.py +1 -1
- package/docs/INSTALLATION.md +1 -1
- package/loki-ts/dist/loki.js +173 -109
- package/mcp/__init__.py +1 -1
- package/package.json +1 -1
package/SKILL.md
CHANGED
|
@@ -3,11 +3,11 @@ 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.5
|
|
7
7
|
|
|
8
8
|
**You are an autonomous agent. You make decisions. You do not ask questions. You do not stop.**
|
|
9
9
|
|
|
10
|
-
**
|
|
10
|
+
**Multi-provider (stable since v5.0.0):** Claude/Codex/Gemini/Cline/Aider with abstract model tiers and degraded mode for non-Claude providers. See `skills/providers.md`. **Current track (v7.5.x):** Phase 1 RARV-C closure -- real provider judges, gate-failure flock, synthetic PRD e2e, status `--json`, dead code cleanup.
|
|
11
11
|
|
|
12
12
|
**Runtime migration in progress:** A bash-to-Bun migration is underway on the `feat/bun-migration` branch. The first phase (shipped in v7.3.0) routes a small set of read-only commands -- `version`, `status`, `stats`, `doctor`, `provider show/list`, `memory list/index` -- through a Bun runtime via `bin/loki`. Every other command remains on the Bash runtime (`autonomy/loki`). Rollback is available with `LOKI_LEGACY_BASH=1`. See `UPGRADING.md` and `docs/architecture/ADR-001-runtime-migration.md` for the full plan.
|
|
13
13
|
|
|
@@ -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.5 | [Autonomi](https://www.autonomi.dev/) flagship product | ~260 lines core**
|
package/VERSION
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
7.5.
|
|
1
|
+
7.5.5
|
package/autonomy/loki
CHANGED
|
@@ -2174,6 +2174,13 @@ if not os.path.isdir(loki_dir):
|
|
|
2174
2174
|
result['pid'] = None
|
|
2175
2175
|
result['elapsed_time'] = 0
|
|
2176
2176
|
result['task_counts'] = {'total': 0, 'completed': 0, 'failed': 0, 'pending': 0}
|
|
2177
|
+
result['phase1'] = {
|
|
2178
|
+
'findings_iters': 0,
|
|
2179
|
+
'learnings_count': 0,
|
|
2180
|
+
'escalations_count': 0,
|
|
2181
|
+
'pause_signal': False,
|
|
2182
|
+
'gate_failure_counts': {},
|
|
2183
|
+
}
|
|
2177
2184
|
print(json.dumps(result, indent=2))
|
|
2178
2185
|
sys.exit(0)
|
|
2179
2186
|
|
|
@@ -2292,6 +2299,59 @@ if os.path.isdir(queue_dir):
|
|
|
2292
2299
|
task_counts['total'] = task_counts['pending'] + task_counts['completed'] + task_counts['failed']
|
|
2293
2300
|
result['task_counts'] = task_counts
|
|
2294
2301
|
|
|
2302
|
+
# v7.5.5 (#204): Phase 1 (RARV-C closure) artifact summary -- mirror of the
|
|
2303
|
+
# Bun route in loki-ts/src/commands/status.ts STATUS_JSON_PY. Keep in sync.
|
|
2304
|
+
phase1 = {
|
|
2305
|
+
'findings_iters': 0,
|
|
2306
|
+
'learnings_count': 0,
|
|
2307
|
+
'escalations_count': 0,
|
|
2308
|
+
'pause_signal': False,
|
|
2309
|
+
'gate_failure_counts': {},
|
|
2310
|
+
}
|
|
2311
|
+
state_dir = os.path.join(loki_dir, 'state')
|
|
2312
|
+
if os.path.isdir(state_dir):
|
|
2313
|
+
try:
|
|
2314
|
+
phase1['findings_iters'] = sum(
|
|
2315
|
+
1 for n in os.listdir(state_dir)
|
|
2316
|
+
if n.startswith('findings-') and n.endswith('.json')
|
|
2317
|
+
)
|
|
2318
|
+
except Exception:
|
|
2319
|
+
pass
|
|
2320
|
+
learnings_file = os.path.join(state_dir, 'relevant-learnings.json')
|
|
2321
|
+
if os.path.isfile(learnings_file):
|
|
2322
|
+
try:
|
|
2323
|
+
with open(learnings_file) as f:
|
|
2324
|
+
learnings = json.load(f)
|
|
2325
|
+
if isinstance(learnings, list):
|
|
2326
|
+
phase1['learnings_count'] = len(learnings)
|
|
2327
|
+
elif isinstance(learnings, dict):
|
|
2328
|
+
entries = learnings.get('entries')
|
|
2329
|
+
if isinstance(entries, list):
|
|
2330
|
+
phase1['learnings_count'] = len(entries)
|
|
2331
|
+
except Exception:
|
|
2332
|
+
pass
|
|
2333
|
+
escalations_dir = os.path.join(loki_dir, 'escalations')
|
|
2334
|
+
if os.path.isdir(escalations_dir):
|
|
2335
|
+
try:
|
|
2336
|
+
phase1['escalations_count'] = sum(
|
|
2337
|
+
1 for n in os.listdir(escalations_dir) if n.endswith('.md')
|
|
2338
|
+
)
|
|
2339
|
+
except Exception:
|
|
2340
|
+
pass
|
|
2341
|
+
phase1['pause_signal'] = os.path.isfile(os.path.join(loki_dir, 'PAUSE'))
|
|
2342
|
+
gate_count_file = os.path.join(loki_dir, 'quality', 'gate-failure-count.json')
|
|
2343
|
+
if os.path.isfile(gate_count_file):
|
|
2344
|
+
try:
|
|
2345
|
+
with open(gate_count_file) as f:
|
|
2346
|
+
gc = json.load(f)
|
|
2347
|
+
if isinstance(gc, dict):
|
|
2348
|
+
phase1['gate_failure_counts'] = {
|
|
2349
|
+
k: v for k, v in gc.items() if isinstance(v, (int, float))
|
|
2350
|
+
}
|
|
2351
|
+
except Exception:
|
|
2352
|
+
pass
|
|
2353
|
+
result['phase1'] = phase1
|
|
2354
|
+
|
|
2295
2355
|
print(json.dumps(result, indent=2))
|
|
2296
2356
|
" "$skill_dir" "$loki_dir" "$dashboard_port" "$env_provider"
|
|
2297
2357
|
|
package/dashboard/__init__.py
CHANGED
package/docs/INSTALLATION.md
CHANGED
package/loki-ts/dist/loki.js
CHANGED
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
// @bun
|
|
2
|
-
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 U1=import.meta.require;var n1={};b(n1,{lokiDir:()=>x,homeLokiDir:()=>L1,findRepoRootForVersion:()=>P1,REPO_ROOT:()=>g});import{resolve as f,dirname as I1}from"path";import{fileURLToPath as z6}from"url";import{existsSync as q1}from"fs";import{homedir as K6}from"os";function V6(){let $=o1;for(let Z=0;Z<6;Z++){if(q1(f($,"VERSION"))&&q1(f($,"autonomy/run.sh")))return $;let Q=I1($);if(Q===$)break;$=Q}return f(o1,"..","..","..")}function P1($){let Z=$;for(let Q=0;Q<6;Q++){if(q1(f(Z,"VERSION"))&&q1(f(Z,"autonomy/run.sh")))return Z;let X=I1(Z);if(X===Z)break;Z=X}return f($,"..","..","..")}function x(){return process.env.LOKI_DIR??f(process.cwd(),".loki")}function L1(){return f(K6(),".loki")}var o1,g;var h=E(()=>{o1=I1(z6(import.meta.url));g=V6()});var r1={};b(r1,{runOrThrow:()=>H6,run:()=>N,commandVersion:()=>B6,commandExists:()=>y,ShellError:()=>x1});async function N($,Z={}){let Q=Bun.spawn({cmd:[...$],stdout:"pipe",stderr:"pipe",env:Z.env?{...process.env,...Z.env}:process.env,cwd:Z.cwd}),X,z;if(Z.timeoutMs&&Z.timeoutMs>0)X=setTimeout(()=>{try{Q.kill("SIGTERM")}catch{}z=setTimeout(()=>{try{Q.kill("SIGKILL")}catch{}},2000)},Z.timeoutMs);try{let[K,V,q]=await Promise.all([new Response(Q.stdout).text(),new Response(Q.stderr).text(),Q.exited]);return{stdout:K,stderr:V,exitCode:q}}finally{if(X)clearTimeout(X);if(z)clearTimeout(z)}}async function H6($,Z={}){let Q=await N($,Z);if(Q.exitCode!==0)throw new x1(`command failed (${Q.exitCode}): ${$.join(" ")}`,Q.exitCode,Q.stdout,Q.stderr);return Q}async function 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 x1;var m=E(()=>{x1=class x1 extends Error{message;exitCode;stdout;stderr;constructor($,Z,Q,X){super($);this.message=$;this.exitCode=Z;this.stdout=Q;this.stderr=X;this.name="ShellError"}}});function u($){return Y6?"":$}var Y6,P,k,w,F5,A,D,L,U;var d=E(()=>{Y6=(process.env.NO_COLOR??"").length>0;P=u("\x1B[0;31m"),k=u("\x1B[0;32m"),w=u("\x1B[1;33m"),F5=u("\x1B[0;34m"),A=u("\x1B[0;36m"),D=u("\x1B[1m"),L=u("\x1B[2m"),U=u("\x1B[0m")});import{existsSync as x6}from"fs";async function a(){if(e!==void 0)return e;let $="/opt/homebrew/bin/python3.12";if(x6($))return e=$,$;let Z=await 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 U0={};b(U0,{runStatus:()=>h6});import{existsSync as S,readFileSync as Z1,readdirSync as Q0,statSync as Z0}from"fs";import{resolve as R,basename as N6}from"path";async function D6(){if(await y("jq"))return!0;return process.stdout.write(`${P}Error: jq is required but not installed.${U}
|
|
3
3
|
`),process.stdout.write(`Install with:
|
|
4
4
|
`),process.stdout.write(` brew install jq (macOS)
|
|
5
5
|
`),process.stdout.write(` apt install jq (Debian/Ubuntu)
|
|
6
6
|
`),process.stdout.write(` yum install jq (RHEL/CentOS)
|
|
7
|
-
`),!1}function
|
|
7
|
+
`),!1}function W1($){if(!Number.isFinite($)||$<=0)return!1;try{return process.kill($,0),!0}catch{return!1}}function G1($){if(!S($))return null;try{let Z=Z1($,"utf-8").trim();if(!Z)return null;let Q=Number.parseInt(Z,10);return Number.isFinite(Q)?Q:null}catch{return null}}function C6($){let Z=[],Q=G1(R($,"loki.pid"));if(Q!==null&&W1(Q))Z.push(`global:${Q}`);let X=R($,"sessions");if(S(X)){let z=[];try{z=Q0(X)}catch{z=[]}for(let K of z){let V=R(X,K);try{if(!Z0(V).isDirectory())continue}catch{continue}let q=R(V,"loki.pid"),W=G1(q);if(W!==null&&W1(W))Z.push(`${K}:${W}`)}}if(S($)){let z=[];try{z=Q0($)}catch{z=[]}for(let K of z){if(!K.startsWith("run-")||!K.endsWith(".pid"))continue;let V=R($,K);try{if(!Z0(V).isFile())continue}catch{continue}let q=N6(K,".pid").slice(4),W=G1(V);if(W!==null&&W1(W)){if(!Z.some((H)=>H.startsWith(`${q}:`)))Z.push(`${q}:${W}`)}}}return Z}async function X0($,Z){let Q=await N(["jq","-r",$,Z]);if(Q.exitCode!==0)return null;return Q.stdout.trim()}function z0($,Z){try{let Q=Z1($,"utf-8"),z=JSON.parse(Q)[Z];if(typeof z==="number"){if(Z==="budget_used"){let K=Math.round(z*100)/100;if(Number.isInteger(K))return String(K);return String(K)}return String(z)}if(z===void 0||z===null)return"0";return String(z)}catch{return"0"}}function K0($,Z,Q){try{let X=Z1($,"utf-8"),K=JSON.parse(X)[Z];if(typeof K==="number"&&Number.isFinite(K))return K;return Q}catch{return Q}}async function k6(){let $=x();if(!await D6())return 1;if(!S($))return process.stdout.write(`${D}Loki Mode Status${U}
|
|
8
8
|
`),process.stdout.write(`
|
|
9
9
|
`),process.stdout.write(`${w}No active session found.${U}
|
|
10
10
|
`),process.stdout.write(`Loki Mode has not been initialized in this directory.
|
|
@@ -16,45 +16,45 @@ var Z6=Object.defineProperty;var X6=($)=>$;function z6($,Z){this[$]=X6.bind(null
|
|
|
16
16
|
`),process.stdout.write(`${L}Current directory: ${process.cwd()}${U}
|
|
17
17
|
`),0;process.stdout.write(`${D}Loki Mode Status${U}
|
|
18
18
|
`),process.stdout.write(`
|
|
19
|
-
`);let Z="",Q=
|
|
19
|
+
`);let Z="",Q=R($,"state","provider");if(S(Q))try{Z=Z1(Q,"utf-8").trim()}catch{Z=""}let X=Z||process.env.LOKI_PROVIDER||"claude",z="full features";switch(X){case"codex":case"gemini":case"aider":z="degraded mode";break;case"cline":z="near-full mode";break;default:z="full features";break}process.stdout.write(`${A}Provider:${U} ${X} (${z})
|
|
20
20
|
`),process.stdout.write(`${L} Switch with: loki provider set <claude|codex|gemini|cline|aider>${U}
|
|
21
21
|
`),process.stdout.write(`
|
|
22
|
-
`);let K=C6($);if(K.length>0){process.stdout.write(`${
|
|
23
|
-
`);for(let J of K){let
|
|
24
|
-
`);else process.stdout.write(` ${A}[#${
|
|
22
|
+
`);let K=C6($);if(K.length>0){process.stdout.write(`${k}Active Sessions: ${K.length}${U}
|
|
23
|
+
`);for(let J of K){let M=J.indexOf(":"),j=M>=0?J.slice(0,M):J,I=M>=0?J.slice(M+1):"";if(j==="global")process.stdout.write(` ${A}[global]${U} PID ${I}
|
|
24
|
+
`);else process.stdout.write(` ${A}[#${j}]${U} PID ${I}
|
|
25
25
|
`)}process.stdout.write(`
|
|
26
26
|
`),process.stdout.write(`${L} Stop specific: loki stop <session-id>${U}
|
|
27
27
|
`),process.stdout.write(`${L} Stop all: loki stop${U}
|
|
28
28
|
`),process.stdout.write(`
|
|
29
|
-
`)}if(
|
|
29
|
+
`)}if(S(R($,"PAUSE")))process.stdout.write(`${w}Status: PAUSED${U}
|
|
30
30
|
`),process.stdout.write(`${L} Resume with: loki resume${U}
|
|
31
31
|
`),process.stdout.write(`
|
|
32
|
-
`);else if(
|
|
32
|
+
`);else if(S(R($,"STOP")))process.stdout.write(`${P}Status: STOPPED${U}
|
|
33
33
|
`),process.stdout.write(`${L} Clear with: loki resume${U}
|
|
34
34
|
`),process.stdout.write(`
|
|
35
|
-
`);let
|
|
36
|
-
`);try{process.stdout.write(
|
|
37
|
-
`)}let
|
|
38
|
-
`);let J=await
|
|
39
|
-
`)}let
|
|
40
|
-
`)}let
|
|
41
|
-
`);else process.stdout.write(`${A}Cost:${U} $${
|
|
42
|
-
`)}let
|
|
43
|
-
`)}let B=
|
|
44
|
-
`)}}return await
|
|
35
|
+
`);let V=R($,"STATUS.txt");if(S(V)){process.stdout.write(`${A}Session Info:${U}
|
|
36
|
+
`);try{process.stdout.write(Z1(V,"utf-8"))}catch{}process.stdout.write(`
|
|
37
|
+
`)}let q=R($,"state","orchestrator.json");if(S(q)){process.stdout.write(`${A}Orchestrator State:${U}
|
|
38
|
+
`);let J=await X0('.currentPhase // "unknown"',q);process.stdout.write(`${J??"unknown"}
|
|
39
|
+
`)}let W=R($,"queue","pending.json");if(S(W)){let J=await X0('if type == "array" then length elif .tasks then .tasks | length else 0 end',W);process.stdout.write(`${A}Pending Tasks:${U} ${J??"0"}
|
|
40
|
+
`)}let G=R($,"metrics","budget.json");if(S(G)){let J=z0(G,"budget_limit"),M=z0(G,"budget_used");if(J!=="0")process.stdout.write(`${A}Budget:${U} $${M} / $${J}
|
|
41
|
+
`);else process.stdout.write(`${A}Cost:${U} $${M} (no limit)
|
|
42
|
+
`)}let H=R($,"state","context-usage.json");if(S(H)){let J=K0(H,"window_size",200000),M=K0(H,"used_tokens",0),j=0;if(J>0)j=Math.floor(M*100/J);process.stdout.write(`${A}Context:${U} ${j}% (${M} / ${J} tokens)
|
|
43
|
+
`)}let B=R($,"dashboard","dashboard.pid");if(S(B)){let J=G1(B);if(J!==null&&W1(J)){let M=process.env.LOKI_DASHBOARD_PORT||"57374";process.stdout.write(`${A}Dashboard:${U} http://127.0.0.1:${M}/
|
|
44
|
+
`)}}return await b6($),process.stdout.write(`
|
|
45
45
|
`),process.stdout.write(`${L} Tip: loki context show - detailed token breakdown${U}
|
|
46
46
|
`),process.stdout.write(`${L} Tip: loki code overview - codebase intelligence${U}
|
|
47
|
-
`),0}async function
|
|
47
|
+
`),0}async function b6($){let Z=R($,"state"),Q=y6(Z),X=R(Z,"relevant-learnings.json"),z=R($,"escalations"),K=Q.length>0,V=S(X),q=S(z);if(!K&&!V&&!q)return;if(process.stdout.write(`
|
|
48
48
|
${A}Phase 1 artifacts:${U}
|
|
49
|
-
`),K){let
|
|
50
|
-
`)}}if(
|
|
51
|
-
`)}}if(
|
|
52
|
-
`)}}function
|
|
53
|
-
`),1;let Z=g,Q=
|
|
54
|
-
`),1;return process.stdout.write(K.stdout),0}async function
|
|
49
|
+
`),K){let W=Q[Q.length-1],G=V0(W);if(G&&Array.isArray(G.findings)){let H={Critical:0,High:0,Medium:0,Low:0};for(let J of G.findings){let M=String(J.severity??"");if(M in H)H[M]=(H[M]??0)+1}let B=Object.entries(H).filter(([,J])=>J>0).map(([J,M])=>`${M} ${J.toLowerCase()}`).join(", ");process.stdout.write(` Findings (iter ${G.iteration??"?"}): ${B||"none"} -- ${G.findings.length} total
|
|
50
|
+
`)}}if(V){let W=V0(X);if(W&&Array.isArray(W.learnings)&&W.learnings.length>0){let G=new Map;for(let B of W.learnings){let J=String(B.trigger??"unknown");G.set(J,(G.get(J)??0)+1)}let H=[...G.entries()].sort((B,J)=>J[1]-B[1]).slice(0,3).map(([B,J])=>`${J} ${B}`).join(", ");process.stdout.write(` Learnings: ${W.learnings.length} total (${H})
|
|
51
|
+
`)}}if(q){let W=0,G="";try{let B=(await import("fs")).readdirSync(z).filter((J)=>J.endsWith(".md"));if(W=B.length,B.length>0)B.sort(),G=B[B.length-1]??""}catch{}if(W>0)process.stdout.write(` Escalations: ${W} handoff doc${W===1?"":"s"} (latest: ${G})
|
|
52
|
+
`)}}function y6($){if(!S($))return[];try{return U1("fs").readdirSync($).filter((X)=>/^findings-\d+\.json$/.test(X)).sort((X,z)=>{let K=Number.parseInt(X.replace(/[^0-9]/g,""),10)||0,V=Number.parseInt(z.replace(/[^0-9]/g,""),10)||0;return K-V}).map((X)=>R($,X))}catch{return[]}}function V0($){try{let Z=U1("fs");return JSON.parse(Z.readFileSync($,"utf-8"))}catch{return null}}async function v6(){let $=await a();if(!$)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
|
|
53
|
+
`),1;let Z=g,Q=x(),X=process.env.LOKI_DASHBOARD_PORT||"57374",z=process.env.LOKI_PROVIDER||"claude",K=await N([$,"-c",S6,Z,Q,X,z],{timeoutMs:30000});if(K.exitCode!==0)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
|
|
54
|
+
`),1;return process.stdout.write(K.stdout),0}async function h6($){let Z=[...$];while(Z.length>0){let Q=Z[0];if(Q==="--json")return v6();if(Q==="--help"||Q==="-h")return process.stdout.write(`Usage: loki status [--json]
|
|
55
55
|
`),0;return process.stdout.write(`${P}Unknown flag: ${Q}${U}
|
|
56
56
|
`),process.stdout.write(`Usage: loki status [--json]
|
|
57
|
-
`),1}return
|
|
57
|
+
`),1}return k6()}var S6=`
|
|
58
58
|
import json, os, sys, time
|
|
59
59
|
|
|
60
60
|
skill_dir = sys.argv[1]
|
|
@@ -81,6 +81,13 @@ if not os.path.isdir(loki_dir):
|
|
|
81
81
|
result['pid'] = None
|
|
82
82
|
result['elapsed_time'] = 0
|
|
83
83
|
result['task_counts'] = {'total': 0, 'completed': 0, 'failed': 0, 'pending': 0}
|
|
84
|
+
result['phase1'] = {
|
|
85
|
+
'findings_iters': 0,
|
|
86
|
+
'learnings_count': 0,
|
|
87
|
+
'escalations_count': 0,
|
|
88
|
+
'pause_signal': False,
|
|
89
|
+
'gate_failure_counts': {},
|
|
90
|
+
}
|
|
84
91
|
print(json.dumps(result, indent=2))
|
|
85
92
|
sys.exit(0)
|
|
86
93
|
|
|
@@ -199,10 +206,64 @@ if os.path.isdir(queue_dir):
|
|
|
199
206
|
task_counts['total'] = task_counts['pending'] + task_counts['completed'] + task_counts['failed']
|
|
200
207
|
result['task_counts'] = task_counts
|
|
201
208
|
|
|
209
|
+
# v7.5.5 (#204): Phase 1 (RARV-C closure) artifact summary so dashboard /
|
|
210
|
+
# CI / operators can confirm the embedded-by-default flow is wired without
|
|
211
|
+
# tailing files. All counts are read-only and degrade silently to zero.
|
|
212
|
+
phase1 = {
|
|
213
|
+
'findings_iters': 0,
|
|
214
|
+
'learnings_count': 0,
|
|
215
|
+
'escalations_count': 0,
|
|
216
|
+
'pause_signal': False,
|
|
217
|
+
'gate_failure_counts': {},
|
|
218
|
+
}
|
|
219
|
+
state_dir = os.path.join(loki_dir, 'state')
|
|
220
|
+
if os.path.isdir(state_dir):
|
|
221
|
+
try:
|
|
222
|
+
phase1['findings_iters'] = sum(
|
|
223
|
+
1 for n in os.listdir(state_dir)
|
|
224
|
+
if n.startswith('findings-') and n.endswith('.json')
|
|
225
|
+
)
|
|
226
|
+
except Exception:
|
|
227
|
+
pass
|
|
228
|
+
learnings_file = os.path.join(state_dir, 'relevant-learnings.json')
|
|
229
|
+
if os.path.isfile(learnings_file):
|
|
230
|
+
try:
|
|
231
|
+
with open(learnings_file) as f:
|
|
232
|
+
learnings = json.load(f)
|
|
233
|
+
if isinstance(learnings, list):
|
|
234
|
+
phase1['learnings_count'] = len(learnings)
|
|
235
|
+
elif isinstance(learnings, dict):
|
|
236
|
+
entries = learnings.get('entries')
|
|
237
|
+
if isinstance(entries, list):
|
|
238
|
+
phase1['learnings_count'] = len(entries)
|
|
239
|
+
except Exception:
|
|
240
|
+
pass
|
|
241
|
+
escalations_dir = os.path.join(loki_dir, 'escalations')
|
|
242
|
+
if os.path.isdir(escalations_dir):
|
|
243
|
+
try:
|
|
244
|
+
phase1['escalations_count'] = sum(
|
|
245
|
+
1 for n in os.listdir(escalations_dir) if n.endswith('.md')
|
|
246
|
+
)
|
|
247
|
+
except Exception:
|
|
248
|
+
pass
|
|
249
|
+
phase1['pause_signal'] = os.path.isfile(os.path.join(loki_dir, 'PAUSE'))
|
|
250
|
+
gate_count_file = os.path.join(loki_dir, 'quality', 'gate-failure-count.json')
|
|
251
|
+
if os.path.isfile(gate_count_file):
|
|
252
|
+
try:
|
|
253
|
+
with open(gate_count_file) as f:
|
|
254
|
+
gc = json.load(f)
|
|
255
|
+
if isinstance(gc, dict):
|
|
256
|
+
phase1['gate_failure_counts'] = {
|
|
257
|
+
k: v for k, v in gc.items() if isinstance(v, (int, float))
|
|
258
|
+
}
|
|
259
|
+
except Exception:
|
|
260
|
+
pass
|
|
261
|
+
result['phase1'] = phase1
|
|
262
|
+
|
|
202
263
|
print(json.dumps(result, indent=2))
|
|
203
|
-
`;var
|
|
204
|
-
`)}function
|
|
205
|
-
Start a session with: loki start <prd>`}}let z=
|
|
264
|
+
`;var q0=E(()=>{m();$1();d();h()});var J0={};b(J0,{runStats:()=>c6,computeStats:()=>H0});import{readdirSync as W0,readFileSync as f6,statSync as G0}from"fs";import{join as p}from"path";function r($){try{if(!G0($).isFile())return null;return JSON.parse(f6($,"utf-8"))}catch{return null}}function E1($){try{return G0($).isDirectory()}catch{return!1}}function g6($){if(!E1($))return[];try{let Z=W0($).filter((Q)=>Q.startsWith("iteration-")&&Q.endsWith(".json"));return Z.sort(),Z.map((Q)=>p($,Q))}catch{return[]}}function t($){return Math.trunc($).toLocaleString("en-US")}function R1($){let Z=Math.trunc($);if(Z<60)return`${Z}s`;let Q=Math.trunc(Z/3600),X=Math.trunc(Z%3600/60),z=Z%60;if(Q>0)return`${Q}h ${String(X).padStart(2,"0")}m`;return`${X}m ${String(z).padStart(2,"0")}s`}function c($,Z=0){let Q=Math.pow(10,Z);return Math.round($*Q)/Q}function i($,Z){return $.toFixed(Z)}function F1($,Z){return $.length>=Z?$:$+" ".repeat(Z-$.length)}function m6($){let Z="N/A",Q=0,X=r(p($,"state","orchestrator.json"));if(X&&typeof X==="object"){if(typeof X.currentPhase==="string")Z=X.currentPhase;if(typeof X.currentIteration==="number")Q=X.currentIteration}let z=p($,"metrics","efficiency"),K=g6(z),V=[];for(let T of K){let _=r(T);if(_&&typeof _==="object")V.push(_)}if(V.length>0)Q=Math.max(Q,V.length);let q=V.reduce((T,_)=>T+(_.input_tokens??0),0),W=V.reduce((T,_)=>T+(_.output_tokens??0),0),G=q+W,H=V.reduce((T,_)=>T+(_.cost_usd??0),0),B=V.reduce((T,_)=>T+(_.duration_seconds??0),0),J=0,M=0,j=r(p($,"metrics","budget.json"));if(j&&typeof j==="object"){if(typeof j.budget_limit==="number")J=j.budget_limit;if(typeof j.budget_used==="number")M=j.budget_used}let I=0,F=0,O=r(p($,"state","quality-gates.json"));if(O&&typeof O==="object"){if(Array.isArray(O)){for(let T of O)if(F+=1,T===!0)I+=1;else if(T&&typeof T==="object"){let _=T;if(_.passed===!0||_.status==="passed")I+=1}}else for(let T of Object.values(O))if(typeof T==="boolean"){if(F+=1,T)I+=1}else if(T&&typeof T==="object"){F+=1;let _=T;if(_.passed===!0||_.status==="passed")I+=1}}let C={},V1=r(p($,"quality","gate-failure-count.json"));if(V1&&typeof V1==="object"&&!Array.isArray(V1)){let T={};for(let[_,v]of Object.entries(V1))if(typeof v==="number")T[_]=v;C=T}let p1=0,c1=0,l1=0,j1=p($,"quality");if(E1(j1)){let T=[];try{T=W0(j1)}catch{T=[]}for(let _ of T){if(!_.endsWith(".json")||_==="gate-failure-count.json")continue;let v=r(p(j1,_));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:q,totalOutput:W,totalTokens:G,totalCost:H,totalDuration:B,budgetLimit:J,budgetUsed:M,gatesPassed:I,gatesTotal:F,gateFailures:C,reviewsTotal:p1,reviewsApproved:c1,reviewsRevision:l1}}function u6($,Z){let Q=$.iterationCount,X={session:{iterations:Q,duration_seconds:$.totalDuration,phase:$.phase},tokens:{input:$.totalInput,output:$.totalOutput,total:$.totalTokens,cost_usd:c($.totalCost,2)},quality:{gates_passed:$.gatesPassed,gates_total:$.gatesTotal,reviews_total:$.reviewsTotal,reviews_approved:$.reviewsApproved,reviews_revision:$.reviewsRevision,gate_failures:$.gateFailures},efficiency:{avg_tokens_per_iteration:Q>0?c($.totalTokens/Q,0):0,avg_cost_per_iteration:Q>0?c($.totalCost/Q,2):0,avg_duration_per_iteration:Q>0?c($.totalDuration/Q,1):0},budget:{used:c($.budgetUsed,2),limit:$.budgetLimit,percent:$.budgetLimit>0?c($.budgetUsed/$.budgetLimit*100,1):0}};if(Z)X.iterations=$.iterations.map((V,q)=>({number:q+1,input_tokens:V.input_tokens??0,output_tokens:V.output_tokens??0,cost_usd:c(V.cost_usd??0,2),duration_seconds:V.duration_seconds??0}));let z=JSON.stringify(X,null,2);function K(V,q){if(!q)return;let W=new RegExp(`("${V}": )(-?\\d+)(,?)$`,"m");z=z.replace(W,(G,H,B,J)=>`${H}${B}.0${J}`)}if(K("avg_duration_per_iteration",Q>0&&Number.isInteger(X.efficiency.avg_duration_per_iteration)),K("percent",$.budgetLimit>0&&Number.isInteger(X.budget.percent)),K("cost_usd",Q>0&&Number.isInteger(X.tokens.cost_usd)),Z)z=z.replace(/("cost_usd": )(-?\d+)(,?)$/gm,(V,q,W,G)=>`${q}${W}.0${G}`);return z}function p6($,Z){let Q=[];if(Q.push("Loki Mode Session Statistics"),Q.push("============================"),Q.push(""),Q.push("Session"),Q.push(` Iterations completed: ${$.iterationCount}`),Q.push(` Duration: ${R1($.totalDuration)}`),Q.push(` Current phase: ${$.phase}`),Q.push(""),Q.push("Token Usage"),$.iterations.length>0)Q.push(` Input tokens: ${t($.totalInput)}`),Q.push(` Output tokens: ${t($.totalOutput)}`),Q.push(` Total tokens: ${t($.totalTokens)}`),Q.push(` Estimated cost: $${i($.totalCost,2)}`);else Q.push(" N/A (no iteration metrics found)");if(Q.push(""),Q.push("Quality Gates"),$.gatesTotal>0){let X=Math.round($.gatesPassed/$.gatesTotal*100);Q.push(` Gates passed: ${$.gatesPassed}/${$.gatesTotal} (${X}%)`)}else Q.push(" Gates passed: N/A");if($.reviewsTotal>0){let X=[];if($.reviewsApproved>0)X.push(`${$.reviewsApproved} approved`);if($.reviewsRevision>0)X.push(`${$.reviewsRevision} revision requested`);let z=X.length>0?X.join(", "):"N/A";Q.push(` Code reviews: ${$.reviewsTotal} (${z})`)}if(Object.keys($.gateFailures).length>0){let X=Object.entries($.gateFailures).filter(([,z])=>z>0).map(([z,K])=>`${z} (${K})`);if(X.length>0)Q.push(` Gate failures: ${X.join(", ")}`)}if(Q.push(""),Q.push("Efficiency"),$.iterationCount>0&&$.iterations.length>0){let X=Math.round($.totalTokens/$.iterationCount),z=$.totalCost/$.iterationCount,K=$.totalDuration/$.iterationCount;Q.push(` Avg tokens/iteration: ${t(X)}`),Q.push(` Avg cost/iteration: $${i(z,2)}`),Q.push(` Avg duration/iteration: ${R1(K)}`)}else Q.push(" N/A (no iteration metrics found)");if(Q.push(""),Q.push("Budget"),$.budgetLimit>0){let X=c($.budgetUsed/$.budgetLimit*100,1),z=Number.isInteger(X)?`${X}.0`:`${X}`;Q.push(` Used: $${i($.budgetUsed,2)} / $${i($.budgetLimit,2)} (${z}%)`)}else if($.budgetUsed>0)Q.push(` Used: $${i($.budgetUsed,2)} (no limit set)`);else Q.push(" N/A");if(Z&&$.iterations.length>0)Q.push(""),Q.push("Per-Iteration Breakdown"),$.iterations.forEach((X,z)=>{let K=z+1,V=F1(t(X.input_tokens??0),10),q=F1(t(X.output_tokens??0),10),W=X.cost_usd??0,G=R1(X.duration_seconds??0),H=F1(`${K}`,3);Q.push(` #${H} input: ${V} output: ${q} cost: $${i(W,2)} time: ${G}`)});return Q.join(`
|
|
265
|
+
`)}function H0($){let Z=!1,Q=!1;for(let V of $)if(V==="--json")Z=!0;else if(V==="--efficiency")Q=!0;let X=x();if(!E1(X)){if(Z)return{exitCode:0,stdout:'{"error": "No active session"}'};return{exitCode:0,stdout:`${w}No active session found.${U}
|
|
266
|
+
Start a session with: loki start <prd>`}}let z=m6(X);return{exitCode:0,stdout:Z?u6(z,Q):p6(z,Q)}}async function c6($){let Z=H0($);return console.log(Z.stdout),Z.exitCode}var B0=E(()=>{h();d()});var I0={};b(I0,{runDoctor:()=>Q3,httpReachable:()=>S1,checkTool:()=>T0,checkSkills:()=>O0,checkDisk:()=>D1,buildDoctorJson:()=>j0});import{existsSync as l6,lstatSync as d6,readlinkSync as o6,statfsSync as n6}from"fs";import{homedir as M0}from"os";import{resolve as Y0}from"path";function s6($){let Z=$.match(a6);return Z?Z[1]:null}async function r6($){try{let Z=await N([$,"--version"],{timeoutMs:5000}),Q=(Z.stdout||Z.stderr||"").trim();return s6(Q)}catch{return null}}function A0($,Z){let Q=$.split(".").map((z)=>parseInt(z,10)),X=Z.split(".").map((z)=>parseInt(z,10));while(Q.length<2)Q.push(0);while(X.length<2)X.push(0);for(let z=0;z<2;z++){let K=Q[z]??0,V=X[z]??0;if(Number.isNaN(K)||Number.isNaN(V))return 0;if(K!==V)return K-V}return 0}async function T0($,Z,Q,X=null){let z=await y(Z),K=z!==null,V=K?await r6(Z):null,q="pass";if(!K)q=Q==="required"?"fail":"warn";else if(X&&V){if(A0(V,X)<0)q=Q==="required"?"fail":"warn"}return{name:$,command:Z,found:K,version:V,required:Q,min_version:X,status:q,path:z}}function D1(){let $=null;try{let Q=n6(M0()),X=Number(Q.bavail)*Number(Q.bsize);$=Math.round(X/1073741824*10)/10}catch{$=null}let Z="pass";if($!==null){if($<1)Z="fail";else if($<5)Z="warn"}return{available_gb:$,status:Z}}async function S1($,Z=2000){try{return(await fetch($,{signal:AbortSignal.timeout(Z)})).ok}catch{return!1}}async function N1($,Z=!1){let Q=`import ${$}`,X=Z?30000:5000;if(!Z)return(await o(Q,{timeoutMs:X})).exitCode===0;let z=await a();if(!z)return!1;return(await N([z,"-c",Q],{timeoutMs:X})).exitCode===0}function O0(){let $=M0();return t6.map(({name:Z,dir:Q})=>{let X=Y0($,Q),z=X,K=Y0(X,"SKILL.md");if(l6(K))return{name:Z,path:z,status:"pass",detail:""};try{if(d6(X).isSymbolicLink()){let q="unknown";try{q=o6(X)}catch{}return{name:Z,path:z,status:"fail",detail:`(broken symlink -> ${q})`}}}catch{}return{name:Z,path:z,status:"warn",detail:"(not found - run 'loki setup-skill')"}})}async function _0(){return Promise.all(i6.map(async($)=>{return{...await T0($.jsonName,$.cmd,$.required,$.min??null),displayName:$.displayName}}))}async function j0(){let Z=(await _0()).map(({displayName:V,...q})=>q),Q=D1(),X=0,z=0,K=0;for(let V of Z)if(V.status==="pass")X++;else if(V.status==="fail")z++;else K++;if(Q.status==="pass")X++;else if(Q.status==="fail")z++;else K++;return{checks:Z,disk:Q,summary:{passed:X,failed:z,warnings:K,ok:z===0}}}function Y($){switch($){case"pass":return`${k}PASS${U}`;case"fail":return`${P}FAIL${U}`;case"warn":return`${w}WARN${U}`}}function H1($){let Z=$.version?` (v${$.version})`:"",Q=$.displayName;if(!$.found){let X=$.required==="required"?"not found":$.required==="recommended"?"not found (recommended)":"not found (optional)";return` ${Y($.status)} ${Q} - ${X}`}if($.min_version&&$.version&&A0($.version,$.min_version)<0){let X=$.required==="required"?"requires":"recommended";return` ${Y($.status)} ${Q}${Z} - ${X} >= ${$.min_version}`}return` ${Y($.status)} ${Q}${Z}`}function J1($,Z){if(Z==="pass")$.pass++;else if(Z==="fail")$.fail++;else $.warn++}function e6(){process.stdout.write(`${D}loki doctor${U} - Check system prerequisites
|
|
206
267
|
|
|
207
268
|
`),process.stdout.write(`Usage: loki doctor [--json]
|
|
208
269
|
|
|
@@ -211,88 +272,88 @@ Start a session with: loki start <prd>`}}let z=u6(X);return{exitCode:0,stdout:Z?
|
|
|
211
272
|
|
|
212
273
|
`),process.stdout.write(`Checks: node, python3, jq, git, curl, bash version,
|
|
213
274
|
`),process.stdout.write(` claude/codex/gemini CLIs, and disk space.
|
|
214
|
-
`)}async function
|
|
275
|
+
`)}async function $3(){process.stdout.write(`${D}Loki Mode Doctor${U}
|
|
215
276
|
|
|
216
277
|
`),process.stdout.write(`Checking system prerequisites...
|
|
217
278
|
|
|
218
279
|
`);let $={pass:0,fail:0,warn:0},Z=await _0(),Q=new Map(Z.map((O)=>[O.command,O]));process.stdout.write(`${A}Required:${U}
|
|
219
|
-
`);for(let O of["node","python3","jq","git","curl"]){let
|
|
220
|
-
`),J1($,
|
|
280
|
+
`);for(let O of["node","python3","jq","git","curl"]){let C=Q.get(O);process.stdout.write(H1(C)+`
|
|
281
|
+
`),J1($,C.status)}process.stdout.write(`
|
|
221
282
|
`),process.stdout.write(`${A}AI Providers:${U}
|
|
222
|
-
`);let X=["claude","codex","gemini","cline","aider"],z=!1;for(let O of X){let
|
|
223
|
-
`),J1($,
|
|
283
|
+
`);let X=["claude","codex","gemini","cline","aider"],z=!1;for(let O of X){let C=Q.get(O);if(process.stdout.write(H1(C)+`
|
|
284
|
+
`),J1($,C.status),C.found)z=!0}if(!z)process.stdout.write(` ${Y("fail")} No AI provider CLI installed -- at least one is required
|
|
224
285
|
`),process.stdout.write(` ${w}Install: npm install -g @anthropic-ai/claude-code${U}
|
|
225
286
|
`),$.fail++;process.stdout.write(`
|
|
226
287
|
`),process.stdout.write(`${A}API Keys:${U}
|
|
227
|
-
`);let K=Q.get("claude").found,
|
|
288
|
+
`);let K=Q.get("claude").found,V=Q.get("codex").found,q=Q.get("gemini").found,W=process.env;if(W.ANTHROPIC_API_KEY)process.stdout.write(` ${Y("pass")} ANTHROPIC_API_KEY is set
|
|
228
289
|
`),$.pass++;else if(K)process.stdout.write(` ${L} -- ${U} ANTHROPIC_API_KEY not set (Claude CLI uses its own login)
|
|
229
|
-
`);if(
|
|
230
|
-
`),$.pass++;else if(
|
|
231
|
-
`);if(
|
|
232
|
-
`),$.pass++;else if(
|
|
290
|
+
`);if(W.OPENAI_API_KEY)process.stdout.write(` ${Y("pass")} OPENAI_API_KEY is set
|
|
291
|
+
`),$.pass++;else if(V)process.stdout.write(` ${L} -- ${U} OPENAI_API_KEY not set (Codex CLI uses its own login)
|
|
292
|
+
`);if(W.GOOGLE_API_KEY||W.GEMINI_API_KEY)process.stdout.write(` ${Y("pass")} GOOGLE_API_KEY is set
|
|
293
|
+
`),$.pass++;else if(q)process.stdout.write(` ${L} -- ${U} GOOGLE_API_KEY not set (Gemini CLI uses its own login)
|
|
233
294
|
`);process.stdout.write(`
|
|
234
295
|
`),process.stdout.write(`${A}Skills:${U}
|
|
235
|
-
`);for(let O of
|
|
236
|
-
`),$.pass++;else if(O.status==="fail")process.stdout.write(` ${
|
|
296
|
+
`);for(let O of O0())if(O.status==="pass")process.stdout.write(` ${Y("pass")} ${O.name} ${L}${O.path}${U}
|
|
297
|
+
`),$.pass++;else if(O.status==="fail")process.stdout.write(` ${Y("fail")} ${O.name} ${L}${O.detail}${U}
|
|
237
298
|
`),process.stdout.write(` ${w}Fix: loki setup-skill${U}
|
|
238
|
-
`),$.fail++;else process.stdout.write(` ${
|
|
299
|
+
`),$.fail++;else process.stdout.write(` ${Y("warn")} ${O.name} ${L}${O.detail}${U}
|
|
239
300
|
`),$.warn++;if(process.stdout.write(`
|
|
240
301
|
`),process.stdout.write(`${A}Integrations:${U}
|
|
241
|
-
`),await
|
|
242
|
-
`),$.pass++;else process.stdout.write(` ${
|
|
243
|
-
`),$.warn++;if(await
|
|
244
|
-
`),$.pass++;else process.stdout.write(` ${
|
|
245
|
-
`),$.warn++;if(await
|
|
246
|
-
`),$.pass++;else process.stdout.write(` ${
|
|
247
|
-
`),$.warn++;if(await
|
|
248
|
-
`),$.pass++;else process.stdout.write(` ${
|
|
249
|
-
`),$.warn++;let
|
|
250
|
-
`),$.pass++;else process.stdout.write(` ${
|
|
251
|
-
`),$.warn++;if(process.env.LOKI_OTEL_ENDPOINT)process.stdout.write(` ${
|
|
252
|
-
`),$.pass++;else process.stdout.write(` ${
|
|
302
|
+
`),await N1("mcp"))process.stdout.write(` ${Y("pass")} MCP SDK (Python)
|
|
303
|
+
`),$.pass++;else process.stdout.write(` ${Y("warn")} MCP SDK - not installed (pip3 install mcp)
|
|
304
|
+
`),$.warn++;if(await N1("numpy",!0))process.stdout.write(` ${Y("pass")} numpy (vector search)
|
|
305
|
+
`),$.pass++;else process.stdout.write(` ${Y("warn")} numpy - not installed (pip3 install numpy)
|
|
306
|
+
`),$.warn++;if(await N1("sentence_transformers",!0))process.stdout.write(` ${Y("pass")} sentence-transformers (embeddings)
|
|
307
|
+
`),$.pass++;else process.stdout.write(` ${Y("warn")} sentence-transformers - not installed (loki memory vectors setup)
|
|
308
|
+
`),$.warn++;if(await S1("http://localhost:8100/api/v2/heartbeat"))process.stdout.write(` ${Y("pass")} ChromaDB server (port 8100)
|
|
309
|
+
`),$.pass++;else process.stdout.write(` ${Y("warn")} ChromaDB - not running (docker start loki-chroma)
|
|
310
|
+
`),$.warn++;let G=process.env.LOKI_MIROFISH_URL;if(G)if(await S1(`${G}/health`))process.stdout.write(` ${Y("pass")} MiroFish server (${G})
|
|
311
|
+
`),$.pass++;else process.stdout.write(` ${Y("warn")} MiroFish - not running (loki start --mirofish-docker <image>)
|
|
312
|
+
`),$.warn++;if(process.env.LOKI_OTEL_ENDPOINT)process.stdout.write(` ${Y("pass")} OTEL endpoint: ${process.env.LOKI_OTEL_ENDPOINT}
|
|
313
|
+
`),$.pass++;else process.stdout.write(` ${Y("warn")} OTEL - not configured (set LOKI_OTEL_ENDPOINT)
|
|
253
314
|
`),$.warn++;process.stdout.write(`
|
|
254
315
|
`),process.stdout.write(`${A}System:${U}
|
|
255
|
-
`);let
|
|
256
|
-
`),J1($,
|
|
257
|
-
`),J1($,B.status);let J=D1(),
|
|
258
|
-
`),$.warn++;else if(J.status==="fail")process.stdout.write(` ${
|
|
259
|
-
`),$.fail++;else if(J.status==="warn")process.stdout.write(` ${
|
|
260
|
-
`),$.warn++;else process.stdout.write(` ${
|
|
316
|
+
`);let H=Q.get("bash");process.stdout.write(H1(H)+`
|
|
317
|
+
`),J1($,H.status);let B=Q.get("bun");if(B)process.stdout.write(H1(B)+`
|
|
318
|
+
`),J1($,B.status);let J=D1(),M=J.available_gb===null?null:Math.floor(J.available_gb);if(M===null)process.stdout.write(` ${Y("warn")} Disk space: unable to determine
|
|
319
|
+
`),$.warn++;else if(J.status==="fail")process.stdout.write(` ${Y("fail")} Disk space: ${M}GB available (need >= 1GB)
|
|
320
|
+
`),$.fail++;else if(J.status==="warn")process.stdout.write(` ${Y("warn")} Disk space: ${M}GB available (low)
|
|
321
|
+
`),$.warn++;else process.stdout.write(` ${Y("pass")} Disk space: ${M}GB available
|
|
261
322
|
`),$.pass++;process.stdout.write(`
|
|
262
323
|
`),process.stdout.write(`${A}Runtime route:${U}
|
|
263
|
-
`);let
|
|
264
|
-
`),process.env.LOKI_LEGACY_BASH==="1"||process.env.LOKI_LEGACY_BASH==="true")process.stdout.write(` ${
|
|
265
|
-
`);if(process.env.LOKI_TS_ENTRY)process.stdout.write(` ${
|
|
266
|
-
`);if(process.env.BUN_FROM_SOURCE==="1"||process.env.BUN_FROM_SOURCE==="true")process.stdout.write(` ${
|
|
267
|
-
`);let
|
|
268
|
-
`);else if(
|
|
269
|
-
`);else process.stdout.write(` ${
|
|
270
|
-
`)}else process.stdout.write(` ${
|
|
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
|
+
`),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
|
+
`);if(process.env.LOKI_TS_ENTRY)process.stdout.write(` ${Y("pass")} LOKI_TS_ENTRY override: ${process.env.LOKI_TS_ENTRY}
|
|
327
|
+
`);if(process.env.BUN_FROM_SOURCE==="1"||process.env.BUN_FROM_SOURCE==="true")process.stdout.write(` ${Y("pass")} BUN_FROM_SOURCE set: shim prefers loki-ts/src/ over dist/
|
|
328
|
+
`);let F=await a();if(F!==null){let C=(await N([F,"-c","import sys; print(f'{sys.version_info.major}.{sys.version_info.minor}')"],{timeoutMs:5000})).stdout.trim();if(C.startsWith("3.12"))process.stdout.write(` ${Y("pass")} Python 3.12 (chromadb / sentence-transformers): ${C} at ${F}
|
|
329
|
+
`);else if(C)process.stdout.write(` ${Y("warn")} Python 3.12 NOT found -- using ${C} at ${F}; chromadb / sentence-transformers may fail. Install python3.12 (brew install python@3.12 / apt install python3.12).
|
|
330
|
+
`);else process.stdout.write(` ${Y("warn")} Python 3 found at ${F} but version probe failed; chromadb may not work.
|
|
331
|
+
`)}else process.stdout.write(` ${Y("warn")} Python 3 not on PATH -- memory + MCP integrations disabled.
|
|
271
332
|
`);if(process.stdout.write(`
|
|
272
|
-
`),process.stdout.write(`${D}Summary:${U} ${
|
|
333
|
+
`),process.stdout.write(`${D}Summary:${U} ${k}${$.pass} passed${U}, ${P}${$.fail} failed${U}, ${w}${$.warn} warnings${U}
|
|
273
334
|
|
|
274
335
|
`),$.fail>0)return process.stdout.write(`${P}Some required prerequisites are missing.${U}
|
|
275
336
|
`),process.stdout.write(`Install missing dependencies and run 'loki doctor' again.
|
|
276
337
|
`),1;if($.warn>0)return process.stdout.write(`${w}All required checks passed with some warnings.${U}
|
|
277
|
-
`),0;return process.stdout.write(`${
|
|
278
|
-
`),0}async function
|
|
338
|
+
`),0;return process.stdout.write(`${k}All checks passed. System is ready for Loki Mode.${U}
|
|
339
|
+
`),0}async function Q3($){let Z=!1;for(let Q of $)if(Q==="--json")Z=!0;else if(Q==="--help"||Q==="-h")return e6(),0;else return process.stderr.write(`${P}Unknown option: ${Q}${U}
|
|
279
340
|
`),process.stderr.write(`Usage: loki doctor [--json]
|
|
280
|
-
`),1;if(Z){let Q=await
|
|
281
|
-
`),0}return
|
|
341
|
+
`),1;if(Z){let Q=await j0();return process.stdout.write(JSON.stringify(Q,null,2)+`
|
|
342
|
+
`),0}return $3()}var a6,t6,i6;var P0=E(()=>{m();$1();d();a6=/(\d+\.\d+(?:\.\d+)*)/;t6=[{name:"Claude Code",dir:".claude/skills/loki-mode"},{name:"Codex CLI",dir:".codex/skills/loki-mode"},{name:"Gemini CLI",dir:".gemini/skills/loki-mode"},{name:"Cline CLI",dir:".cline/skills/loki-mode"},{name:"Aider CLI",dir:".aider/skills/loki-mode"}];i6=[{displayName:"Node.js (>= 18)",jsonName:"Node.js",cmd:"node",required:"required",min:"18.0"},{displayName:"Python 3 (>= 3.8)",jsonName:"Python 3",cmd:"python3",required:"required",min:"3.8"},{displayName:"jq",jsonName:"jq",cmd:"jq",required:"required"},{displayName:"git",jsonName:"git",cmd:"git",required:"required"},{displayName:"curl",jsonName:"curl",cmd:"curl",required:"required"},{displayName:"bash (>= 4.0)",jsonName:"bash",cmd:"bash",required:"recommended",min:"4.0"},{displayName:"Bun (>= 1.3)",jsonName:"Bun",cmd:"bun",required:"recommended",min:"1.3"},{displayName:"Claude CLI",jsonName:"Claude CLI",cmd:"claude",required:"optional"},{displayName:"Codex CLI",jsonName:"Codex CLI",cmd:"codex",required:"optional"},{displayName:"Gemini CLI",jsonName:"Gemini CLI",cmd:"gemini",required:"optional"},{displayName:"Cline CLI",jsonName:"Cline CLI",cmd:"cline",required:"optional"},{displayName:"Aider CLI",jsonName:"Aider CLI",cmd:"aider",required:"optional"}]});import{existsSync as B1,mkdirSync as Z3,copyFileSync as X3,readFileSync as z3,readdirSync as K3,statSync as V3,writeFileSync as W7,renameSync as U3,appendFileSync as G7,rmSync as H7}from"fs";import{join as n,dirname as q3}from"path";function Y1($){return n($,"state","checkpoints")}function G3($){let Z=Y1($);if(!B1(Z))return[];return K3(Z).filter((Q)=>Q.startsWith("cp-")).filter((Q)=>{try{return V3(n(Z,Q)).isDirectory()}catch{return!1}})}function H3($){return[...$].sort((Z,Q)=>{let X=L0(Z),z=L0(Q);return X-z})}function L0($){let Z=$.split("-");if(Z.length<3)return 0;let Q=Z[Z.length-1],X=Number.parseInt(Q??"0",10);return Number.isFinite(X)?X:0}function k1($){let Z=$??x(),Q=H3(G3(Z)),X=[];for(let z of Q){let K=x0(Z,z);if(K)X.push(K)}return X}function x0($,Z){let Q=n(Y1($),Z,"metadata.json");if(!B1(Q))return null;try{return JSON.parse(z3(Q,"utf-8"))}catch{return null}}function b1($,Z){if(!J3.test($))throw new w0($);let Q=Z??x(),X=n(Y1(Q),$);if(!B1(X))throw new C1($);let z=x0(Q,$);if(!z)throw new C1($);return z}function R0($,Z){let Q=b1($,Z),X=Z??x(),z=n(Y1(X),$),K=[];for(let V of B3){let q=n(z,V);if(!B1(q))continue;K.push({from:q,to:n(X,V)})}return{id:$,metadata:Q,restore:K}}function F0($){let Z=[],Q=0;for(let X of $.restore)try{Z3(q3(X.to),{recursive:!0});let z=`${X.to}.tmp.${process.pid}.${++W3}`;X3(X.from,z),U3(z,X.to),Q+=1}catch(z){Z.push(`${X.from} -> ${X.to}: ${z.message}`)}return{restored:Q,errors:Z}}var M7,W3=0,J3,C1,w0,B3;var E0=E(()=>{h();m();M7=Promise.resolve();J3=/^[a-zA-Z0-9_-]+$/;C1=class C1 extends Error{id;constructor($){super(`Checkpoint not found: ${$}`);this.id=$;this.name="CheckpointNotFoundError"}};w0=class w0 extends Error{id;constructor($){super(`Invalid checkpoint ID: must be alphanumeric, hyphens, underscores only (got: ${$})`);this.id=$;this.name="InvalidCheckpointIdError"}};B3=["state/orchestrator.json","queue/pending.json","queue/completed.json","queue/in-progress.json","queue/current-task.json"]});var D0={};b(D0,{runRollback:()=>Y3});async function Y3($){let Z=$[0],Q=$.slice(1);if(Z===void 0||Z==="help"||Z==="--help"||Z==="-h")return process.stdout.write(N0),Z===void 0?1:0;switch(Z){case"list":{let X=[...k1()].reverse();if(X.length===0)return process.stdout.write(`${w}No checkpoints found.${U}
|
|
282
343
|
`),0;process.stdout.write(`${D}Checkpoints${U} (${X.length}, newest first):
|
|
283
344
|
`);for(let z of X)process.stdout.write(` ${A}${z.id}${U} iter=${z.iteration} ${z.git_branch||"(no branch)"}@${(z.git_sha||"").slice(0,7)} ${z.timestamp}
|
|
284
345
|
`);return 0}case"show":{let X=Q[0];if(!X)return process.stderr.write(`${P}Missing checkpoint id.${U} Use \`loki rollback list\`.
|
|
285
346
|
`),2;try{let z=b1(X);return process.stdout.write(`${JSON.stringify(z,null,2)}
|
|
286
347
|
`),0}catch(z){return process.stderr.write(`${P}Failed to read checkpoint:${U} ${z.message}
|
|
287
348
|
`),1}}case"to":{let X=Q[0];if(!X)return process.stderr.write(`${P}Missing checkpoint id.${U} Use \`loki rollback list\`.
|
|
288
|
-
`),2;return
|
|
349
|
+
`),2;return S0(X)}case"latest":{let X=k1(),z=X[X.length-1];if(!z)return process.stderr.write(`${P}No checkpoints found to roll back to.${U}
|
|
289
350
|
`),1;return process.stdout.write(`Rolling back to latest checkpoint: ${A}${z.id}${U}
|
|
290
|
-
`),
|
|
291
|
-
`),process.stderr.write(N0),2}}function
|
|
351
|
+
`),S0(z.id)}default:return process.stderr.write(`Unknown subcommand: ${Z}
|
|
352
|
+
`),process.stderr.write(N0),2}}function S0($){let Z;try{Z=R0($)}catch(X){return process.stderr.write(`${P}Cannot plan rollback:${U} ${X.message}
|
|
292
353
|
`),1}if(Z.restore.length===0)return process.stdout.write(`${w}Checkpoint ${$} has no restorable state files; nothing to do.${U}
|
|
293
|
-
`),0;let Q=
|
|
354
|
+
`),0;let Q=F0(Z);if(Q.errors.length>0){for(let X of Q.errors)process.stderr.write(`${P}restore error:${U} ${X}
|
|
294
355
|
`);return process.stderr.write(`${P}Partial rollback: ${Q.restored}/${Z.restore.length} files restored.${U}
|
|
295
|
-
`),1}return process.stdout.write(`${
|
|
356
|
+
`),1}return process.stdout.write(`${k}Rolled back ${Q.restored}/${Z.restore.length} state files from ${$}.${U}
|
|
296
357
|
`),process.stdout.write("Run `loki start` to resume from the restored state.\n"),0}var N0=`Usage: loki rollback <subcommand>
|
|
297
358
|
|
|
298
359
|
Subcommands:
|
|
@@ -308,9 +369,9 @@ Restored files (matches autonomy/run.sh:7028 byte-for-byte):
|
|
|
308
369
|
Note: only state files are restored. Source code, git history, and the
|
|
309
370
|
session's autonomy-state.json are unchanged. Re-run \`loki start\` to
|
|
310
371
|
resume from the restored state.
|
|
311
|
-
`;var C0=E(()=>{
|
|
312
|
-
`),
|
|
313
|
-
`)}function
|
|
372
|
+
`;var C0=E(()=>{E0();d()});import{closeSync as j7,mkdirSync as M3,openSync as I7,readFileSync as P7,renameSync as A3,rmSync as L7,statSync as x7,writeFileSync as T3,writeSync as w7}from"fs";import{dirname as O3}from"path";function X1($,Z){M3(O3($),{recursive:!0});let Q=`${$}.tmp.${process.pid}.${++_3}`;T3(Q,`${JSON.stringify(Z,null,2)}
|
|
373
|
+
`),A3(Q,$)}async function k0($,Z){let Q=M1.get($)??Promise.resolve(),X=()=>{},z=new Promise((V)=>{X=V}),K=Q.catch(()=>{}).then(()=>z);M1.set($,K);try{return await Q.catch(()=>{}),await Z()}finally{if(X(),M1.get($)===K)M1.delete($)}}var _3=0,M1;var y1=E(()=>{M1=new Map});var h1={};b(h1,{renderFindingsForPrompt:()=>x3,loadPreviousFindings:()=>v1,findLatestReviewDir:()=>f0,_parseReviewerOutputForTests:()=>w3});import{existsSync as y0,readFileSync as b0,readdirSync as v0,statSync as j3}from"fs";import{join as A1}from"path";function L3($){let Z=$.toLowerCase();if(Z==="critical")return"Critical";if(Z==="high")return"High";if(Z==="medium")return"Medium";return"Low"}function h0($,Z,Q,X){let z=[],K=$.split(/\r?\n/);for(let V of K){let q=V.trim();if(q.length===0)continue;let W=q.replace(/^[-*]\s*/,""),G=I3.exec(W);if(!G)continue;let H=L3(G[1]),B=G[2].trim(),J=P3.exec(B),M=J?J[1]:null,j=J?Number.parseInt(J[2],10):null;z.push({reviewId:Q,iteration:X,reviewer:Z,severity:H,description:B,file:M,line:Number.isFinite(j)?j:null,raw:q})}return z}function f0($,Z){let Q=A1($,"quality","reviews");if(!y0(Q))return null;let X;try{X=v0(Q)}catch{return null}let z=Z===void 0?X.filter((q)=>q.startsWith("review-")):X.filter((q)=>q.endsWith(`-${Z}`)&&q.startsWith("review-"));if(z.length===0)return null;z.sort();let K=z[z.length-1];if(!K)return null;let V=A1(Q,K);try{if(!j3(V).isDirectory())return null}catch{return null}return V}function v1($,Z){let Q=f0($,Z);if(Q===null)return{reviewDir:null,reviewId:null,iteration:null,findings:[]};let X=null,z=null,K=A1(Q,"aggregate.json");if(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 q=new Set(["diff.txt","files.txt","anti-sycophancy.txt"]),W=[];for(let G of V){if(!G.endsWith(".txt"))continue;if(q.has(G))continue;if(G.endsWith("-prompt.txt"))continue;let H=G.replace(/\.txt$/,""),B;try{B=b0(A1(Q,G),"utf-8")}catch{continue}W.push(...h0(B,H,X??"",z??-1))}return{reviewDir:Q,reviewId:X,iteration:z,findings:W}}function x3($){if($.length===0)return"";let Z=["Critical","High","Medium","Low"],Q=new Map;for(let z of Z)Q.set(z,[]);for(let z of $){let K=Q.get(z.severity);if(K)K.push(z)}let X=[];X.push("PREVIOUS REVIEWER FINDINGS (must address each, or supply counter-evidence in .loki/state/counter-evidence-<iter>.json):");for(let z of Z){let K=Q.get(z)??[];if(K.length===0)continue;X.push(` [${z}] (${K.length}):`);for(let V of K){let q=V.file?` (${V.file}${V.line!==null?":"+V.line:""})`:"";X.push(` - ${V.description}${q} -- via ${V.reviewer}`)}}return X.join(`
|
|
374
|
+
`)}function w3($,Z,Q="review-test",X=0){return h0($,Z,Q,X)}var I3,P3;var T1=E(()=>{I3=/\[(Critical|High|Medium|Low)\]\s*(.+)/i,P3=/([\w.\-/]+\.[a-zA-Z]+):(\d+)/});import{existsSync as R3}from"fs";import{join as F3}from"path";async function g0($,Z){let Q=F3($,"memory");if(!R3(Q))return{stored:!1,reason:"memory dir not initialized"};let X=Math.max(0,Math.floor(Z.durationSeconds??0)),z={_LOKI_PROJECT_DIR:g,_LOKI_TARGET_DIR:process.cwd(),_LOKI_TASK_ID:Z.taskId,_LOKI_OUTCOME:Z.outcome,_LOKI_PHASE:Z.phase,_LOKI_GOAL:Z.goal,_LOKI_DURATION:String(X),_LOKI_LOKI_DIR:$},V=await o(`
|
|
314
375
|
import os, sys
|
|
315
376
|
project = os.environ.get('_LOKI_PROJECT_DIR', '')
|
|
316
377
|
loki = os.environ.get('_LOKI_LOKI_DIR', '.loki')
|
|
@@ -337,21 +398,21 @@ try:
|
|
|
337
398
|
print('OK')
|
|
338
399
|
except Exception as e:
|
|
339
400
|
print('ERR:' + str(e))
|
|
340
|
-
`,{env:z,timeoutMs:15000});if(
|
|
341
|
-
`)}async function
|
|
342
|
-
`)}function
|
|
343
|
-
`),process.stderr.write(m1),2}}async function
|
|
344
|
-
`),2;let Q=
|
|
345
|
-
`),0;let K=
|
|
401
|
+
`,{env:z,timeoutMs:15000});if(V.exitCode===127)return{stored:!1,reason:"python3 not found"};let q=V.stdout.trim();if(q==="OK")return{stored:!0,reason:"stored"};if(q.startsWith("ERR:"))return{stored:!1,reason:q.replace(/^ERR:/,"")};return{stored:!1,reason:V.stderr.trim()||"unknown"}}var m0=E(()=>{$1();h()});var l0={};b(l0,{loadLearnings:()=>f1,appendLearning:()=>z1,appendFromGateFailure:()=>y3});import{existsSync as E3,readFileSync as N3}from"fs";import{join as u0}from"path";import{createHash as S3}from"crypto";function p0($){return u0($,D3)}function C3($){if($===null||typeof $!=="object")return!1;let Z=$;return typeof Z.id==="string"&&typeof Z.timestamp==="string"&&typeof Z.iteration==="number"&&typeof Z.trigger==="string"&&typeof Z.rootCause==="string"&&typeof Z.fix==="string"&&typeof Z.preventInFuture==="string"&&typeof Z.evidence==="object"&&Z.evidence!==null}function c0($){if(!E3($))return{version:1,learnings:[]};try{let Z=N3($,"utf-8"),Q=JSON.parse(Z);if(Q.version===1&&Array.isArray(Q.learnings))return{version:1,learnings:Q.learnings.filter(C3)}}catch{}return{version:1,learnings:[]}}function k3($,Z){return S3("sha256").update(`${$}\x00${Z}`).digest("hex").slice(0,16)}async function z1($,Z,Q={}){let X=k3(Z.trigger,Z.rootCause),z=new Date().toISOString(),K={id:X,timestamp:z,...Z},V=p0($);if(await k0(V,()=>{let W=c0(V),G=W.learnings.findIndex((H)=>H.id===X);if(G>=0){let H=W.learnings[G];W.learnings[G]={...H,timestamp:z,iteration:K.iteration}}else W.learnings.push(K);X1(V,W)}),Q.episodeBridge!==null&&(Q.episodeBridge!==void 0||process.env.LOKI_AUTO_LEARNINGS_EPISODE==="1")){let W=Q.episodeBridge??g0,G=Q.bridgeFailureLog??b3;try{let H=await W($,{taskId:`learning-${X}`,outcome:"failure",phase:"VERIFY",goal:`${Z.trigger}: ${Z.rootCause}`});if(H&&!H.stored){if(!new Set(["memory dir not initialized","stub"]).has(H.reason))G(`episode_bridge skipped: ${H.reason}`)}}catch(H){G(`episode_bridge threw: ${H.message}`)}}return K}function b3($){process.stderr.write(`[learnings_writer] ${$}
|
|
402
|
+
`)}async function y3($,Z,Q,X={}){let z=`[${Q.severity}] ${Q.description}`;return z1($,{iteration:Z,trigger:"gate_failure",rootCause:z,fix:"pending: dev agent must address in next iteration or supply counter-evidence",preventInFuture:"if this finding recurs, lower its severity threshold or add a regression test",evidence:{reviewId:Q.reviewId,file:Q.file??void 0,line:Q.line??void 0,severity:Q.severity,reviewer:Q.reviewer}},X)}function f1($){return c0(p0($))}var D3;var O1=E(()=>{y1();m0();D3=u0("state","relevant-learnings.json")});var o0={};b(o0,{runOverrideCouncil:()=>u3,recordOverrideOutcome:()=>p3,loadCounterEvidence:()=>m3,canonicalFindingId:()=>g1,DEFAULT_OVERRIDE_JUDGES:()=>d0});import{existsSync as v3,readFileSync as h3}from"fs";import{join as f3}from"path";function m3($,Z){let Q=f3($,"state",`counter-evidence-${Z}.json`);if(!v3(Q))return null;try{let X=h3(Q,"utf-8"),z=JSON.parse(X);if(typeof z.iteration!=="number")return null;let K=Array.isArray(z.evidence)?z.evidence:[],V=[];for(let q of K){if(typeof q!=="object"||q===null)continue;let W=q;if(typeof W.findingId!=="string")continue;if(typeof W.claim!=="string")continue;if(typeof W.proofType!=="string")continue;let G=W.proofType;if(!g3.has(G))continue;let H=Array.isArray(W.artifacts)?W.artifacts:[];V.push({findingId:W.findingId,claim:W.claim,proofType:G,artifacts:H.filter((B)=>typeof B==="string")})}return{iteration:z.iteration,evidence:V}}catch{return null}}async function u3($,Z,Q,X={}){let z=X.judges??d0,K=new Set,V=new Set,q={},W=new Map;for(let G of Z.evidence)W.set(G.findingId,G);for(let G of $){let H=g1(G),B=W.get(H);if(!B){V.add(H);continue}let J=await Promise.all(z.map((j)=>Q({finding:G,evidence:B,judge:j})));if(q[H]=J,J.filter((j)=>j.verdict==="APPROVE_OVERRIDE").length>=2)K.add(H);else V.add(H)}return{approvedFindingIds:K,rejectedFindingIds:V,votes:q}}function g1($){let Z=$.raw.slice(0,80).replace(/\s+/g," ").trim();return`${$.reviewer}::${Z}`}async function p3($,Z,Q,X,z={}){let K={episodeBridge:z.episodeBridge===void 0?null:z.episodeBridge};for(let V of X){let q=g1(V);if(Q.approvedFindingIds.has(q))await z1($,{iteration:Z,trigger:"override_approved",rootCause:`[${V.severity}] ${V.description}`,fix:"override council approved counter-evidence; finding lifted",preventInFuture:"if this reviewer/file pair recurs, narrow the reviewer's selector OR add a baseline doc",evidence:{findingId:q,reviewId:V.reviewId,file:V.file??void 0,line:V.line??void 0,severity:V.severity,reviewer:V.reviewer}},K);else if(Q.rejectedFindingIds.has(q))await z1($,{iteration:Z,trigger:"override_rejected",rootCause:`[${V.severity}] ${V.description}`,fix:"override council rejected -- dev agent must fix the finding",preventInFuture:"address this finding in the next iteration",evidence:{findingId:q,reviewId:V.reviewId,file:V.file??void 0,line:V.line??void 0,severity:V.severity,reviewer:V.reviewer}},K)}}var g3,d0;var n0=E(()=>{O1();g3=new Set(["file-exists","test-passes","grep-miss","reviewer-misread","duplicate-code-path","out-of-scope"]);d0=["judge-primary","judge-secondary","judge-tertiary"]});var r0={};b(r0,{writeEscalationHandoff:()=>$5,renderHandoff:()=>a0,readLatestHandoff:()=>Q5});import{existsSync as c3,mkdirSync as l3,readdirSync as d3,readFileSync as o3,renameSync as n3,writeFileSync as a3}from"fs";import{dirname as s3,join as _1}from"path";function r3(){return new Date().toISOString()}function t3($){let Z=$.file?` (${$.file}${$.line!==null?":"+$.line:""})`:"";return` - [${$.severity}] ${$.description}${Z} -- ${$.reviewer}`}function i3($){let Z=$.evidence,Q=Z.file?` ${Z.file}${Z.line!==void 0?":"+Z.line:""}`:"";return` - **${$.trigger}** (iter ${$.iteration})${Q}: ${$.rootCause}`}function a0($,Z,Q){let X=[];if(X.push(`# Loki escalation handoff -- ${r3()}`),X.push(""),X.push(`Gate **${$.gateName}** has failed ${$.consecutiveFailures} consecutive times at iteration ${$.iteration}.`),X.push(""),X.push(`Reason: ${$.detail}`),X.push(""),Z.length>0){X.push(`## Outstanding findings (${Z.length})`),X.push("");for(let z of Z)X.push(t3(z));X.push("")}else X.push("## Outstanding findings"),X.push(""),X.push("(no per-finding records captured -- gate failed without populating reviewer outputs)"),X.push("");if(Q.length>0){X.push(`## Recent learnings (${Math.min(Q.length,10)})`),X.push("");for(let z of Q.slice(-10))X.push(i3(z));X.push("")}return X.push("## What the human must decide"),X.push(""),X.push("- Approve override? Write `.loki/state/counter-evidence-<iter>.json` with one entry per finding to dispute, then `rm .loki/PAUSE` to resume."),X.push("- Disable a gate? Set `LOKI_GATE_<NAME>=false` in env (see skills/quality-gates.md)."),X.push("- Tweak escalation? Set `LOKI_GATE_PAUSE_LIMIT` or `LOKI_GATE_ESCALATE_LIMIT`."),X.push("- Roll back? Switch to `LOKI_LEGACY_BASH=1` and re-run; the bash route does not consult this handoff doc."),X.push(""),X.push("To resume: address the findings (or supply counter-evidence) and `rm .loki/PAUSE`."),X.join(`
|
|
403
|
+
`)}function e3($,Z){l3(s3($),{recursive:!0});let Q=`${$}.tmp.${process.pid}.${++s0}`;a3(Q,Z),n3(Q,$)}function $5($,Z,Q={}){let X=Q.findings??v1($,Z.iteration).findings,z=Q.learnings??f1($).learnings,K=a0(Z,X,z),V=(Q.now?.()??new Date).toISOString().replace(/[-:.]/g,""),q=_1($,"escalations"),W=++s0,G=_1(q,`handoff-${V}-${process.pid}-${W}-${Z.gateName}.md`);return e3(G,K),{path:G,bytes:K.length}}function Q5($){let Z=_1($,"escalations");if(!c3(Z))return null;let Q;try{Q=d3(Z).filter((K)=>K.endsWith(".md"))}catch{return null}if(Q.length===0)return null;Q.sort();let X=Q[Q.length-1];if(!X)return null;let z=_1(Z,X);try{return{path:z,body:o3(z,"utf-8")}}catch{return null}}var s0=0;var t0=E(()=>{T1();O1()});var i0={};b(i0,{runInternalPhase1Hooks:()=>U5,_resolveForTests:()=>V5,_internalPhase1HooksHelp:()=>H5});import{existsSync as Z5,mkdirSync as X5,readdirSync as z5,statSync as K5}from"fs";import{join as K1,resolve as V5}from"path";async function U5($){let[Z,...Q]=$;switch(Z){case void 0:case"help":case"--help":case"-h":return process.stdout.write(m1),Z===void 0?1:0;case"reflect":return q5(Q);case"override":return W5(Q);case"handoff":return G5(Q);default:return process.stderr.write(`Unknown subcommand: ${Z}
|
|
404
|
+
`),process.stderr.write(m1),2}}async function q5($){let Z=u1($[0]);if(Z===null)return process.stderr.write(`reflect: missing or invalid <iter>
|
|
405
|
+
`),2;let Q=x();try{let z=(await Promise.resolve().then(() => (T1(),h1))).loadPreviousFindings(Q,Z);if(z.findings.length===0)return process.stdout.write(`reflect: no findings for iter ${Z} (nothing to do)
|
|
406
|
+
`),0;let K=K1(Q,"state");X5(K,{recursive:!0}),X1(K1(K,`findings-${Z}.json`),{review_id:z.reviewId,iteration:Z,findings:z.findings});let V=await Promise.resolve().then(() => (O1(),l0)),q=0;if(process.env.LOKI_AUTO_LEARNINGS!=="0"){for(let W of z.findings)if(W.severity==="Critical"||W.severity==="High")await V.appendFromGateFailure(Q,Z,W,{episodeBridge:null}),q+=1}return process.stdout.write(`reflect: persisted ${z.findings.length} findings + ${q} learnings (iter ${Z})
|
|
346
407
|
`),0}catch(X){return process.stderr.write(`reflect: ${X.message}
|
|
347
408
|
`),1}}async function W5($){let Z=u1($[0]);if(Z===null)return process.stderr.write(`override: missing or invalid <iter>
|
|
348
|
-
`),2;let Q=
|
|
349
|
-
`),0;let
|
|
350
|
-
`),0;let
|
|
351
|
-
`);else process.stdout.write(`override: BLOCKED -- ${J} approved, ${
|
|
409
|
+
`),2;let Q=x();try{let X=await Promise.resolve().then(() => (n0(),o0)),z=X.loadCounterEvidence(Q,Z);if(z===null||z.evidence.length===0)return process.stdout.write(`override: no counter-evidence for iter ${Z} (skip)
|
|
410
|
+
`),0;let V=(await Promise.resolve().then(() => (T1(),h1))).loadPreviousFindings(Q,Z),q=V.findings.filter((I)=>I.severity==="Critical"||I.severity==="High");if(q.length===0)return process.stdout.write(`override: no blocking findings for iter ${Z} (skip)
|
|
411
|
+
`),0;let W=new Set(["duplicate-code-path","file-exists","test-passes","grep-miss","out-of-scope"]),G=async(I)=>{let F=W.has(I.evidence.proofType);return{judge:I.judge,verdict:F?"APPROVE_OVERRIDE":"REJECT_OVERRIDE",reasoning:F?`[stub] proofType=${I.evidence.proofType} trusted`:`[stub] proofType=${I.evidence.proofType} requires manual review`}},H=await X.runOverrideCouncil(q,z,G);await X.recordOverrideOutcome(Q,Z,H,q);let B=K1(Q,"quality","reviews");if(Z5(B))try{let I=z5(B).filter((O)=>O.startsWith("review-")).sort(),F=I[I.length-1];if(F&&K5(K1(B,F)).isDirectory())X1(K1(B,F,`override-${Z}.json`),{review_id:V.reviewId,iteration:Z,approved_finding_ids:Array.from(H.approvedFindingIds),rejected_finding_ids:Array.from(H.rejectedFindingIds),votes:H.votes})}catch{}let J=H.approvedFindingIds.size,M=H.rejectedFindingIds.size;if(M===0&&J>0)process.stdout.write(`override: LIFTED -- ${J} approved, ${M} rejected
|
|
412
|
+
`);else process.stdout.write(`override: BLOCKED -- ${J} approved, ${M} rejected
|
|
352
413
|
`);return 0}catch(X){return process.stderr.write(`override: ${X.message}
|
|
353
414
|
`),1}}async function G5($){let Z=$[0],Q=Number.parseInt($[1]??"0",10),X=u1($[2]);if(!Z||!Number.isFinite(Q)||X===null)return process.stderr.write(`handoff: usage: handoff <gate> <consecutive-failures> <iter>
|
|
354
|
-
`),2;let z=
|
|
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)
|
|
355
416
|
`),0}catch(K){return process.stderr.write(`handoff: ${K.message}
|
|
356
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>
|
|
357
418
|
|
|
@@ -362,24 +423,24 @@ Subcommands:
|
|
|
362
423
|
|
|
363
424
|
This command is invoked by autonomy/run.sh between iterations. Users
|
|
364
425
|
should not run it directly -- run \`loki start\` instead.
|
|
365
|
-
`,
|
|
366
|
-
`),0}m();d();
|
|
426
|
+
`,H5;var e0=E(()=>{h();y1();H5=m1});h();import{readFileSync as U6}from"fs";import{resolve as q6,dirname as W6}from"path";import{fileURLToPath as G6}from"url";var l=null;function a1(){if(l!==null)return l;let $="7.5.5";if(typeof $==="string"&&$.length>0)return l=$,l;try{let Z=W6(G6(import.meta.url)),Q=P1(Z);l=U6(q6(Q,"VERSION"),"utf-8").trim()}catch{l="unknown"}return l}function s1(){return process.stdout.write(`Loki Mode v${a1()}
|
|
427
|
+
`),0}m();d();h();import{readFileSync as M6,existsSync as A6}from"fs";import{resolve as T6}from"path";var O6=["claude","codex","gemini","cline","aider"];function t1(){let $=T6(x(),"state","provider");if(!A6($))return"";try{return M6($,"utf-8").trim()}catch{return""}}function _6($,Z){return $||Z||process.env.LOKI_PROVIDER||"claude"}function j6($){let Z=t1(),Q=_6($,Z);switch(process.stdout.write(`${D}Current Provider${U}
|
|
367
428
|
`),process.stdout.write(`
|
|
368
429
|
`),process.stdout.write(`${A}Provider:${U} ${Q}
|
|
369
|
-
`),Q){case"claude":process.stdout.write(`${
|
|
370
|
-
`);break;case"cline":process.stdout.write(`${
|
|
430
|
+
`),Q){case"claude":process.stdout.write(`${k}Status:${U} Full features (subagents, parallel, MCP)
|
|
431
|
+
`);break;case"cline":process.stdout.write(`${k}Status:${U} Near-full mode (subagents, MCP, 12+ providers)
|
|
371
432
|
`);break;case"codex":case"gemini":case"aider":process.stdout.write(`${w}Status:${U} Degraded mode (sequential only)
|
|
372
433
|
`);break;default:break}if(Z)process.stdout.write(`${L}(saved in .loki/state/provider)${U}
|
|
373
434
|
`);else process.stdout.write(`${L}(default - not explicitly set)${U}
|
|
374
435
|
`);return process.stdout.write(`
|
|
375
436
|
`),process.stdout.write(`Switch provider: ${A}loki provider set <name>${U}
|
|
376
437
|
`),process.stdout.write(`Available: ${A}loki provider list${U}
|
|
377
|
-
`),0}async function
|
|
438
|
+
`),0}async function I6(){let Z=t1()||process.env.LOKI_PROVIDER||"claude";process.stdout.write(`${D}Available Providers${U}
|
|
378
439
|
`),process.stdout.write(`
|
|
379
|
-
`);let Q=await Promise.all(
|
|
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${U}`:`${P}not installed${U}`);let z=[["claude","claude - Claude Code (Anthropic) "],["codex","codex - Codex CLI (OpenAI) "],["gemini","gemini - Gemini CLI (Google) "],["cline","cline - Cline (multi-provider) "],["aider","aider - Aider (terminal pair prog) "]];for(let[K,V]of z){let q=Z===K?` ${A}(current)${U}`:"";process.stdout.write(` ${V} ${X.get(K)}${q}
|
|
380
441
|
`)}return process.stdout.write(`
|
|
381
442
|
`),process.stdout.write(`Set provider: ${A}loki provider set <name>${U}
|
|
382
|
-
`),0}function
|
|
443
|
+
`),0}function P6(){return process.stdout.write(`${D}Loki Mode Provider Management${U}
|
|
383
444
|
`),process.stdout.write(`
|
|
384
445
|
`),process.stdout.write(`Usage: loki provider <command>
|
|
385
446
|
`),process.stdout.write(`
|
|
@@ -397,17 +458,17 @@ should not run it directly -- run \`loki start\` instead.
|
|
|
397
458
|
`),process.stdout.write(` loki provider list
|
|
398
459
|
`),process.stdout.write(` loki provider info gemini
|
|
399
460
|
`),process.stdout.write(` loki provider models
|
|
400
|
-
`),0}async function
|
|
401
|
-
`))if(X.includes('"description"'))Q++;return Q}catch{return 0}}async function
|
|
461
|
+
`),0}async function i1($){let Z=$[0]??"show",Q=$.slice(1);switch(Z){case"show":case"current":return j6(Q[0]);case"list":return I6();case"set":case"info":case"models":return L6(["provider",Z,...Q]);default:return P6()}}async function L6($){let{run:Z}=await Promise.resolve().then(() => (m(),r1)),{resolve:Q}=await import("path"),{REPO_ROOT:X}=await Promise.resolve().then(() => (h(),n1)),z=Q(X,"autonomy","loki"),K=await Z([z,...$],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(K.stdout),process.stderr.write(K.stderr),K.exitCode}d();h();$1();m();import{existsSync as e1,readFileSync as w6}from"fs";import{resolve as s}from"path";import{mkdir as R6}from"fs/promises";var Q1=s(L1(),"learnings");function w1($){if(!e1($))return 0;try{let Z=w6($,"utf-8"),Q=0;for(let X of Z.split(`
|
|
462
|
+
`))if(X.includes('"description"'))Q++;return Q}catch{return 0}}async function F6(){await R6(Q1,{recursive:!0});let $=w1(s(Q1,"patterns.jsonl")),Z=w1(s(Q1,"mistakes.jsonl")),Q=w1(s(Q1,"successes.jsonl"));return process.stdout.write(`${D}Cross-Project Learnings${U}
|
|
402
463
|
`),process.stdout.write(`
|
|
403
|
-
`),process.stdout.write(` Patterns: ${
|
|
464
|
+
`),process.stdout.write(` Patterns: ${k}${$}${U}
|
|
404
465
|
`),process.stdout.write(` Mistakes: ${w}${Z}${U}
|
|
405
466
|
`),process.stdout.write(` Successes: ${A}${Q}${U}
|
|
406
467
|
`),process.stdout.write(`
|
|
407
|
-
`),process.stdout.write(`Location: ${
|
|
468
|
+
`),process.stdout.write(`Location: ${Q1}
|
|
408
469
|
`),process.stdout.write(`
|
|
409
470
|
`),process.stdout.write(`Use 'loki memory show <type>' to view entries
|
|
410
|
-
`),0}async function
|
|
471
|
+
`),0}async function E6($){if($){let X=`
|
|
411
472
|
try:
|
|
412
473
|
from memory.layers import IndexLayer
|
|
413
474
|
layer = IndexLayer('.loki/memory')
|
|
@@ -417,9 +478,9 @@ except ImportError:
|
|
|
417
478
|
print('Error: memory.layers module not found')
|
|
418
479
|
except Exception as e:
|
|
419
480
|
print(f'Error: {e}')
|
|
420
|
-
`.trim(),z=await o(X,{cwd:g});return process.stdout.write(z.stdout),0}let Z=
|
|
481
|
+
`.trim(),z=await o(X,{cwd:g});return process.stdout.write(z.stdout),0}let Z=s(x(),"memory","index.json");if(!e1(Z))return process.stdout.write(`No index found
|
|
421
482
|
`),0;let Q=await o(`import json, sys; sys.stdout.write(json.dumps(json.load(open(${JSON.stringify(Z)})), indent=4) + "\\n")`);if(Q.exitCode!==0)return process.stdout.write(`No index found
|
|
422
|
-
`),0;return process.stdout.write(Q.stdout),0}async function
|
|
483
|
+
`),0;return process.stdout.write(Q.stdout),0}async function $0($){switch($[0]??"list"){case"list":case"ls":return F6();case"index":return E6($[1]==="rebuild");default:{let Q=s(g,"autonomy","loki"),X=await N([Q,"memory",...$],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(X.stdout),process.stderr.write(X.stderr),X.exitCode}}}var $6=`Loki Mode (TypeScript port, Phase 2 of bash->Bun migration)
|
|
423
484
|
|
|
424
485
|
Usage: loki <command> [args...]
|
|
425
486
|
|
|
@@ -437,8 +498,11 @@ Phase 2 ported (Bun-native, fast):
|
|
|
437
498
|
|
|
438
499
|
All other commands fall through to the bash CLI (autonomy/loki).
|
|
439
500
|
Set LOKI_LEGACY_BASH=1 to force the bash CLI for every command.
|
|
440
|
-
`;async function
|
|
501
|
+
`;async function J5($){let Z=$[0],Q=$.slice(1);switch(Z){case void 0:case"help":case"--help":case"-h":return process.stdout.write($6),0;case"version":case"--version":case"-v":return s1();case"provider":return i1(Q);case"memory":return $0(Q);case"status":{let{runStatus:X}=await Promise.resolve().then(() => (q0(),U0));return X(Q)}case"stats":{let{runStats:X}=await Promise.resolve().then(() => (B0(),J0));return X(Q)}case"doctor":{let{runDoctor:X}=await Promise.resolve().then(() => (P0(),I0));return X(Q)}case"rollback":{let{runRollback:X}=await Promise.resolve().then(() => (C0(),D0));return X(Q)}case"internal":{let X=Q[0];if(!X||X==="--help"||X==="-h"||X==="help"){let K=["loki internal -- runtime hooks driven by autonomy/run.sh","","Subcommands:"," phase1-hooks Persist structured findings, run override council,"," append learnings, and write the escalation handoff"," doc once per iteration. Driven by run.sh; not"," intended for direct invocation.","","These commands are wired into the autonomous loop and may change","without notice. Do not script against them.",""].join(`
|
|
502
|
+
`);return process.stdout.write(`${K}
|
|
503
|
+
`),0}if(X==="phase1-hooks"){let{runInternalPhase1Hooks:K}=await Promise.resolve().then(() => (e0(),i0));return K(Q.slice(1))}return process.stderr.write(`Unknown internal subcommand: ${X}
|
|
504
|
+
`),process.stderr.write(`Run 'loki internal --help' for the supported list.
|
|
441
505
|
`),2}default:return process.stderr.write(`Unknown command: ${Z}
|
|
442
|
-
`),process.stderr.write(
|
|
506
|
+
`),process.stderr.write($6),2}}process.on("SIGINT",()=>process.exit(130));process.on("SIGTERM",()=>process.exit(143));var B5=await J5(Bun.argv.slice(2));process.exit(B5);
|
|
443
507
|
|
|
444
|
-
//# debugId=
|
|
508
|
+
//# debugId=F84BEA15062CBB6B64756E2164756E21
|
package/mcp/__init__.py
CHANGED