loki-mode 7.3.0 → 7.4.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -49,6 +49,45 @@ Or skip scaffolding and go straight to a quick task:
49
49
  loki quick "build a landing page with a signup form"
50
50
  ```
51
51
 
52
+ ---
53
+
54
+ ## Runtime Architecture
55
+
56
+ Loki Mode is in the middle of a phased migration from a Bash-based runtime to a TypeScript/Bun runtime. The work is happening on the `feat/bun-migration` branch and is being shipped incrementally.
57
+
58
+ **What ships today:**
59
+
60
+ - A small set of read-only commands is routed to the Bun runtime when `bun` is on `PATH`. The router lives in `bin/loki` and currently routes: `version`, `--version`, `-v`, `status`, `stats`, `doctor`, `provider` (covers `provider show` and `provider list`), `memory` (covers `memory list` and `memory index`).
61
+ - Every other command continues to execute on the existing Bash CLI (`autonomy/loki`).
62
+ - If `bun` is not on `PATH`, the shim falls through to Bash silently. Existing users without Bun installed see no behavior change.
63
+
64
+ **Rollback flag:**
65
+
66
+ Force every command to take the legacy Bash path:
67
+
68
+ ```bash
69
+ LOKI_LEGACY_BASH=1 loki <cmd>
70
+ ```
71
+
72
+ This is the documented escape hatch for any user who hits a regression on the Bun route. The Bash path remains the source of truth through Phase 5.
73
+
74
+ **Phase 6 (planned, calendar TBD):**
75
+
76
+ The next major release sunsets the Bash runtime entirely. There is no firm calendar date. Users who need to stay on the Bash route should pin the last v7.x release.
77
+
78
+ **Cost:**
79
+
80
+ - Adds a Bun runtime dependency (Bun 1.3.0 or newer recommended; the shim works as long as `bun` resolves).
81
+ - Adds a Bun toolchain to the system (Bun itself is roughly 50 MB installed via `brew install` or the official curl installer). The published `loki-ts/dist/loki.js` bundle inside the npm tarball is approximately 152 KB.
82
+ - Speedup on the ported commands is measured in `.loki/metrics/migration_bench_soak.jsonl` and analysed in [ADR-001](docs/architecture/ADR-001-runtime-migration.md). Recorded soak results show roughly 3x to 5x faster execution on the ported commands (per-command range 2.9x to 5.0x); treat as indicative, not contractual.
83
+
84
+ **More:**
85
+
86
+ - [UPGRADING.md](UPGRADING.md) -- per-version upgrade and rollback guidance.
87
+ - [ADR-001: Runtime Migration](docs/architecture/ADR-001-runtime-migration.md) -- design rationale and phase definitions.
88
+
89
+ ---
90
+
52
91
  <details>
53
92
  <summary><strong>Other install methods</strong></summary>
54
93
 
package/SKILL.md CHANGED
@@ -3,12 +3,14 @@ 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.3.0
6
+ # Loki Mode v7.4.8
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
  **New in v5.0.0:** Multi-provider support (Claude/Codex/Gemini/Cline/Aider), abstract model tiers, degraded mode for non-Claude providers. See `skills/providers.md`.
11
11
 
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
+
12
14
  ---
13
15
 
14
16
  ## PRIORITY 1: Load Context (Every Turn)
@@ -320,4 +322,4 @@ The following features are documented in skill modules but not yet fully automat
320
322
  | Quality gates 3-reviewer system | Implemented (v5.35.0) | 5 specialist reviewers in `skills/quality-gates.md`; execution in run.sh |
321
323
  | Benchmarks (HumanEval, SWE-bench) | Infrastructure only | Runner scripts and datasets exist in `benchmarks/`; no published results |
322
324
 
323
- **v7.3.0 | [Autonomi](https://www.autonomi.dev/) flagship product | ~260 lines core**
325
+ **v7.4.8 | [Autonomi](https://www.autonomi.dev/) flagship product | ~260 lines core**
package/VERSION CHANGED
@@ -1 +1 @@
1
- 7.3.0
1
+ 7.4.8
package/autonomy/loki CHANGED
@@ -31,6 +31,13 @@ BOLD='\033[1m'
31
31
  DIM='\033[2m'
32
32
  NC='\033[0m'
33
33
 
34
+ # v7.4.5 (BUG-15 bash route): honor NO_COLOR convention (https://no-color.org/)
35
+ # Mirrors the equivalent fix in loki-ts/src/util/colors.ts. Required for parity
36
+ # when the shim falls through to the bash route (e.g., when bun is missing).
37
+ if [ -n "${NO_COLOR:-}" ]; then
38
+ RED=''; GREEN=''; YELLOW=''; BLUE=''; CYAN=''; BOLD=''; DIM=''; NC=''
39
+ fi
40
+
34
41
  # Logging functions (portable across bash/zsh)
35
42
  log_info() { echo -e "${GREEN}[INFO]${NC} $*"; }
36
43
  log_error() { echo -e "${RED}[ERROR]${NC} $*"; }
@@ -2529,7 +2536,8 @@ if reviews_total > 0:
2529
2536
  if gate_failures:
2530
2537
  failure_parts = [f'{k} ({v})' for k, v in gate_failures.items() if v > 0]
2531
2538
  if failure_parts:
2532
- print(f' Gate failures: {', '.join(failure_parts)}')
2539
+ sep = ', '
2540
+ print(f' Gate failures: {sep.join(failure_parts)}')
2533
2541
  print()
2534
2542
 
2535
2543
  # Efficiency
package/bin/loki CHANGED
@@ -34,18 +34,33 @@ BASH_CLI="$REPO_ROOT/autonomy/loki"
34
34
 
35
35
  # Resolve which Bun entry to use:
36
36
  # 1. LOKI_TS_ENTRY=... -- explicit override (custom builds, tests)
37
- # 2. BUN_FROM_SOURCE=1 -- force src/cli.ts (used by bench --compare-dist
37
+ # 2. BUN_FROM_SOURCE=1 -- prefer src/cli.ts (used by bench --compare-dist
38
38
  # and during Phase 3 development before dist ships)
39
39
  # 3. dist/loki.js exists -- production path (npm/Docker/Homebrew artifact)
40
40
  # 4. fall back to src -- in-repo development before first build
41
+ #
42
+ # v7.4.2 fix (BUG-1): BUN_FROM_SOURCE=1 used to hard-fail on npm/Docker/brew
43
+ # installs because src/ is excluded by .npmignore. Now we warn once and fall
44
+ # back to dist if src/cli.ts is missing.
41
45
  if [ -n "${LOKI_TS_ENTRY:-}" ]; then
42
46
  BUN_CLI="$LOKI_TS_ENTRY"
43
47
  elif [ "${BUN_FROM_SOURCE:-0}" = "1" ] || [ "${BUN_FROM_SOURCE:-}" = "true" ]; then
44
- BUN_CLI="$REPO_ROOT/loki-ts/src/cli.ts"
48
+ if [ -f "$REPO_ROOT/loki-ts/src/cli.ts" ]; then
49
+ BUN_CLI="$REPO_ROOT/loki-ts/src/cli.ts"
50
+ elif [ -f "$REPO_ROOT/loki-ts/dist/loki.js" ]; then
51
+ echo "WARN: BUN_FROM_SOURCE set but loki-ts/src/cli.ts missing (typical for npm/Docker/brew installs); using dist/loki.js" >&2
52
+ BUN_CLI="$REPO_ROOT/loki-ts/dist/loki.js"
53
+ else
54
+ echo "ERROR: BUN_FROM_SOURCE set but neither src/cli.ts nor dist/loki.js found; falling through to bash" >&2
55
+ exec "$BASH_CLI" "$@"
56
+ fi
45
57
  elif [ -f "$REPO_ROOT/loki-ts/dist/loki.js" ]; then
46
58
  BUN_CLI="$REPO_ROOT/loki-ts/dist/loki.js"
47
- else
59
+ elif [ -f "$REPO_ROOT/loki-ts/src/cli.ts" ]; then
48
60
  BUN_CLI="$REPO_ROOT/loki-ts/src/cli.ts"
61
+ else
62
+ # Neither dist nor src available; fall through to bash silently.
63
+ exec "$BASH_CLI" "$@"
49
64
  fi
50
65
 
51
66
  # Force-fall-through to bash when the rollback flag is set.
package/bin/loki-mode.js CHANGED
@@ -1,17 +1,22 @@
1
1
  #!/usr/bin/env node
2
2
  /**
3
- * Loki Mode CLI wrapper for npm distribution
4
- * Delegates to the bash CLI
3
+ * Loki Mode npm wrapper.
4
+ *
5
+ * Delegates to bin/loki (the runtime-aware shim) so that ported commands
6
+ * route through the Bun runtime when bun is on PATH and unported commands
7
+ * fall through to autonomy/loki (bash). Pre-v7.4.7 this script bypassed
8
+ * the shim and went straight to bash, which silently disabled the Bun
9
+ * route for users invoking the `loki-mode` binary instead of `loki`.
5
10
  */
6
11
 
7
12
  const { spawn } = require('child_process');
8
13
  const path = require('path');
9
14
 
10
- const lokiScript = path.join(__dirname, '..', 'autonomy', 'loki');
15
+ const shim = path.join(__dirname, 'loki');
11
16
  const args = process.argv.slice(2);
12
17
 
13
- const child = spawn(lokiScript, args, {
14
- stdio: 'inherit'
18
+ const child = spawn(shim, args, {
19
+ stdio: 'inherit',
15
20
  });
16
21
 
17
22
  child.on('close', (code) => {
@@ -20,6 +25,6 @@ child.on('close', (code) => {
20
25
 
21
26
  child.on('error', (err) => {
22
27
  console.error('Error running loki:', err.message);
23
- console.error('Make sure bash is available on your system');
28
+ console.error('Make sure bash is available on your system (bin/loki shim requires /bin/bash).');
24
29
  process.exit(1);
25
30
  });
@@ -7,7 +7,7 @@ Modules:
7
7
  control: Session control API (start/stop/pause/resume)
8
8
  """
9
9
 
10
- __version__ = "7.3.0"
10
+ __version__ = "7.4.8"
11
11
 
12
12
  # Expose the control app for easy import
13
13
  try:
@@ -0,0 +1,76 @@
1
+ # ARM64 Image Verification
2
+
3
+ This document describes the manual procedure to verify that a published `asklokesh/loki-mode` Docker image actually runs on a real ARM64 host. CI does not exercise the runtime on ARM64; see `docs/UNREACHABLE-TESTS.md` for the rationale.
4
+
5
+ This procedure should be run by a maintainer (or any contributor with an ARM64 host) after each release that bumps the published image tag.
6
+
7
+ ---
8
+
9
+ ## Prerequisites
10
+
11
+ - An ARM64 host. Examples:
12
+ - Apple Silicon Mac (M1, M2, M3, or M4).
13
+ - AWS Graviton instance.
14
+ - Raspberry Pi 4 or 5 with a 64-bit OS.
15
+ - Docker Engine or Docker Desktop installed and running.
16
+ - Network access to Docker Hub.
17
+
18
+ Confirm the host is genuinely ARM64:
19
+
20
+ ```bash
21
+ uname -m
22
+ # expected output: arm64 (macOS) or aarch64 (Linux)
23
+ ```
24
+
25
+ If the output is anything else (`x86_64`, `amd64`), this procedure does not apply -- you need an ARM64 host.
26
+
27
+ ---
28
+
29
+ ## Procedure
30
+
31
+ Replace `7.X.Y` with the tag you intend to verify.
32
+
33
+ ```bash
34
+ docker run --rm --platform linux/arm64 asklokesh/loki-mode:7.X.Y loki version
35
+ ```
36
+
37
+ The `--platform linux/arm64` flag forces Docker to pull the ARM64 variant of the manifest list even if a multi-arch image is being served. This catches the case where the image was published as x86_64-only by accident.
38
+
39
+ ### Expected output
40
+
41
+ The command prints the version string of the bundled CLI and exits 0. The exact format follows the bash CLI's `version` command. A non-zero exit, a Python or Bun runtime error, or a missing-binary error all indicate a broken ARM64 build.
42
+
43
+ ### Additional smoke checks
44
+
45
+ Run a few more commands to surface ARM64-specific runtime issues:
46
+
47
+ ```bash
48
+ docker run --rm --platform linux/arm64 asklokesh/loki-mode:7.X.Y loki doctor
49
+ docker run --rm --platform linux/arm64 asklokesh/loki-mode:7.X.Y loki status
50
+ ```
51
+
52
+ `loki doctor` prints the runtime environment summary (Bun version, Python version, provider CLIs detected). On the published image, the absence of any provider CLI is expected; the doctor command itself should still complete without an internal error.
53
+
54
+ `loki status` reads `.loki/` state. With no mounted state directory, it should report `stopped` or an equivalent empty state -- not crash.
55
+
56
+ ---
57
+
58
+ ## Recording the result
59
+
60
+ When verification passes, note it in the release commit body or in the GitHub Release notes under a "Verified channels" section. Include:
61
+
62
+ - Host architecture (`uname -m` output)
63
+ - Host OS (`uname -s` plus distribution if Linux)
64
+ - Docker version (`docker --version`)
65
+ - Image tag verified
66
+ - Date of verification
67
+
68
+ When verification fails, open an issue with the same details plus the full output of the failing command. Do not promote the release to "stable" until the failure is fixed.
69
+
70
+ ---
71
+
72
+ ## Why this is manual
73
+
74
+ The `parity-drift.yml` and image-publish workflows use `buildx` with QEMU emulation, which produces a multi-arch manifest but does not exercise the runtime on real ARM64 silicon. QEMU translation can mask runtime issues that only surface on real hardware (for example, native binary dependencies, mmap layout differences, or timing-sensitive code).
75
+
76
+ Until the project provisions an ARM64 self-hosted runner, this manual procedure is the only honest verification.
@@ -2,7 +2,7 @@
2
2
 
3
3
  The flagship product of [Autonomi](https://www.autonomi.dev/). Complete installation instructions for all platforms and use cases.
4
4
 
5
- **Version:** v7.3.0
5
+ **Version:** v7.4.8
6
6
 
7
7
  ---
8
8
 
@@ -91,6 +91,26 @@ loki setup-skill
91
91
 
92
92
  ---
93
93
 
94
+ ## PyPI / Python SDK
95
+
96
+ **The `loki` CLI is NOT available via `pip install loki-mode`.** PyPI hosts only the
97
+ Python REST client SDK at `loki-mode-sdk` (v7.4.8+). The dashboard, MCP server,
98
+ and orchestrator components ship via npm, Docker, and Homebrew only.
99
+
100
+ ```bash
101
+ # Install the Python REST client only
102
+ pip install loki-mode-sdk
103
+
104
+ # Install the full CLI (recommended)
105
+ npm install -g loki-mode # or: brew tap asklokesh/tap && brew install loki-mode
106
+ ```
107
+
108
+ The naming asymmetry (`loki-mode` on npm vs `loki-mode-sdk` on PyPI) is
109
+ intentional: PyPI's `loki-mode` namespace is reserved for a future server
110
+ package, while `loki-mode-sdk` is the thin client.
111
+
112
+ ---
113
+
94
114
  ## Quick Start
95
115
 
96
116
  ```bash
package/docs/SLO.md ADDED
@@ -0,0 +1,52 @@
1
+ # Service Level Objectives
2
+
3
+ This document records baseline targets for the user-visible behavior of the CLI and the runner. These are honest first-pass targets. **No SLI/SLO infrastructure is shipped yet.** Tracking is manual, via the parity-drift workflow and soak-window observation.
4
+
5
+ The targets below should be treated as goals, not contracts. They will be revised as measurement infrastructure is built out.
6
+
7
+ ---
8
+
9
+ ## Latency
10
+
11
+ | Command | Route | p99 target | Source of measurement |
12
+ |---------|-------|------------|----------------------|
13
+ | `loki version` | Bun | < 100 ms | `.loki/metrics/migration_bench_soak.jsonl` (recorded p95 ~30 ms; p99 not recorded but headroom is large) |
14
+ | `loki version` | Bash | < 200 ms | `.loki/metrics/migration_bench_soak.jsonl` (recorded p95 ~141 ms) |
15
+ | `loki status` | either route | < 500 ms | not currently measured in CI; manual observation only |
16
+
17
+ **Caveats:**
18
+
19
+ - The soak file records p50 and p95, not p99. The p99 targets above assume modest tail behavior on top of the recorded p95. They should be tightened or loosened once a proper p99 measurement is in place.
20
+ - All numbers are wall-clock cold starts measured via `hyperfine` on developer hardware. Latency on slower machines (low-end CI runners, ARM64 emulated, containers under heavy load) will be worse.
21
+
22
+ ---
23
+
24
+ ## Parity
25
+
26
+ | Property | Target | Measurement |
27
+ |----------|--------|-------------|
28
+ | Byte-divergence between Bash and Bun routes for the ported commands | 0 divergences | `parity-drift.yml` workflow runs the Bun and Bash routes side by side; any non-empty diff fails the job |
29
+
30
+ The eight commands currently in scope: `version`, `status`, `stats`, `doctor`, `provider show`, `provider list`, `memory list`, `memory index`.
31
+
32
+ ---
33
+
34
+ ## Reliability
35
+
36
+ | Metric | Target | Status |
37
+ |--------|--------|--------|
38
+ | `loki start` completion rate on standard PRDs | 99% | aspirational; no automated measurement yet |
39
+ | Dashboard availability when invoked from `loki start` | 99.9% per session | aspirational; no automated measurement yet |
40
+ | Pre-publish tarball validation pass rate before any release | 100% | enforced manually per `CLAUDE.md` "Pre-Publish Validation"; failure blocks release |
41
+
42
+ The reliability numbers above are explicitly aspirational. The system does not currently emit a "completion / non-completion" signal that could be aggregated into a reliability number across users. The 99% target represents what the maintainer wants to be true, not what is currently measured.
43
+
44
+ ---
45
+
46
+ ## Notes on measurement
47
+
48
+ - The only continuous SLI shipping today is the parity-drift workflow.
49
+ - Latency measurements are recorded in `.loki/metrics/migration_bench_soak.jsonl` when a benchmark run is performed; they are not collected continuously.
50
+ - Reliability is tracked by manual review during the v7.3.0 soak window.
51
+
52
+ These are baseline targets. They will be revised when SLI/SLO infrastructure ships.
@@ -0,0 +1,123 @@
1
+ # Unreachable Tests
2
+
3
+ This document is an honest log of test scenarios that the project cannot exercise in CI today, why each one is blocked, and what (if anything) substitutes for the missing coverage.
4
+
5
+ The intent is to keep this list short and embarrassing so the gaps stay visible. If a row here can be closed, close it.
6
+
7
+ ---
8
+
9
+ ## Real Claude / Codex / Gemini API calls
10
+
11
+ **What we cannot test in CI:** end-to-end runs that actually invoke a hosted Claude, OpenAI Codex, or Google Gemini agent loop with real credentials.
12
+
13
+ **Blocker:**
14
+
15
+ - Per-call cost. A full PRD run can rack up double-digit dollars per provider; running on every PR is not affordable.
16
+ - Auth. Storing real provider tokens in CI gives any contributor on a fork-PR an attack surface to exfiltrate them. We do not pass secrets to fork PRs by policy.
17
+ - Agent-loop safety. A real provider call has full tool-use authority and can take destructive actions (file writes, shell commands) that are unsafe to run unattended in CI.
18
+
19
+ **Manual procedure that closes the gap:**
20
+
21
+ Run the relevant `tests/integration/` script locally with real credentials in a sandbox container or VM.
22
+
23
+ **Partial substitute we ship:**
24
+
25
+ - Provider loader unit tests under `tests/test-provider-loader.sh` cover flag selection, model name mapping, and fallback wiring without invoking the provider.
26
+ - The Bun test suite uses recorded fixtures for `build_prompt` and the runner state machine.
27
+
28
+ ---
29
+
30
+ ## Windows runtime
31
+
32
+ **What we cannot test in CI:** Loki Mode running natively on Windows (PowerShell or cmd, or Windows-native bash).
33
+
34
+ **Blocker:** No Windows host in the CI pool. The runtime relies heavily on POSIX bash semantics; Windows bash via WSL is the only realistic path and we have not provisioned a runner for it.
35
+
36
+ **Manual procedure that closes the gap:**
37
+
38
+ Install on a Windows host with WSL2 + Ubuntu, run `loki doctor` and the smoke commands, file an issue with the output. Document any divergence in `docs/PLATFORM-SUPPORT.md` (file does not exist yet -- create it on first finding).
39
+
40
+ **Partial substitute we ship:**
41
+
42
+ None. Windows is not a supported platform today.
43
+
44
+ ---
45
+
46
+ ## Real ARM64 runtime
47
+
48
+ **What we cannot test in CI:** the published Docker image actually starting up and answering `loki version` on a real ARM64 host.
49
+
50
+ **Blocker:** GitHub Actions provides ARM64 emulation via QEMU under buildx, but we do not have an ARM64 runner in the pool. Buildx can produce the image; it cannot meaningfully exercise the runtime.
51
+
52
+ **Manual procedure that closes the gap:**
53
+
54
+ See `docs/ARM64-VERIFICATION.md` for the manual verification procedure on an Apple Silicon Mac or another ARM64 host.
55
+
56
+ **Partial substitute we ship:**
57
+
58
+ - `buildx` produces the multi-arch image and verifies the build does not fail.
59
+ - The same TypeScript/Bun runtime ships unchanged across architectures, so x86_64 unit tests cover most of the runtime logic.
60
+
61
+ ---
62
+
63
+ ## Real PRD end-to-end
64
+
65
+ **What we cannot test in CI:** a real PRD running through the full RARV loop to completion, with provider calls, code generation, tests, and deploy artifacts.
66
+
67
+ **Blocker:**
68
+
69
+ - Cost (see "Real Claude / Codex / Gemini API calls" above).
70
+ - Nondeterminism. The provider's output varies, so an end-to-end pass/fail signal is noisy.
71
+ - Wall-clock duration. A standard PRD takes 30-90 minutes; CI runners time out long before.
72
+
73
+ **Manual procedure that closes the gap:**
74
+
75
+ Maintainer runs the templated PRDs under `templates/` against the current build before a release and inspects the output. This is captured in the release checklist informally.
76
+
77
+ **Partial substitute we ship:**
78
+
79
+ - Unit tests against recorded provider outputs for the prompt builder and council voter.
80
+ - Smoke tests of the runner state machine that exercise every phase transition without invoking a provider.
81
+
82
+ ---
83
+
84
+ ## 1-hour-plus stress runs
85
+
86
+ **What we cannot test in CI:** sessions that run for an hour or more to surface memory leaks, file-descriptor leaks, or queue churn under sustained load.
87
+
88
+ **Blocker:** GitHub-hosted runner minutes. A one-hour test on every PR would consume the project's monthly minute allocation in a few days.
89
+
90
+ **Manual procedure that closes the gap:**
91
+
92
+ Run `./benchmarks/run-benchmarks.sh` on a self-hosted machine for the full duration and capture process metrics with `ps`/`top` / a simple sampling script. Record findings under `.loki/metrics/`.
93
+
94
+ **Partial substitute we ship:**
95
+
96
+ - The completion council circuit breaker and budget breaker are unit-tested for trigger logic.
97
+ - The dashboard tracks per-iteration cost and context usage; large regressions show up in `.loki/metrics/efficiency/`.
98
+
99
+ ---
100
+
101
+ ## Brew install via the real tap
102
+
103
+ **What we cannot test in CI:** `brew tap asklokesh/tap && brew install loki-mode` against the real tap and a real macOS host.
104
+
105
+ **Blocker:**
106
+
107
+ - Running `brew install` on a CI runner mutates the runner's Homebrew prefix. This interferes with subsequent jobs on the same runner (GitHub-hosted runners are ephemeral, but the side effects still apply for the remainder of the job).
108
+ - Cross-tap mutation in CI risks accidentally publishing a broken formula if the install path is reused for verification.
109
+
110
+ **Manual procedure that closes the gap:**
111
+
112
+ Maintainer runs the brew install on a clean macOS host (or a fresh VM) after every release and checks `loki version` plus a smoke command. Verification is captured in the release checklist.
113
+
114
+ **Partial substitute we ship:**
115
+
116
+ - The `homebrew-tap` repository's CI lints the formula on every update.
117
+ - Tarball validation in `CLAUDE.md` "Pre-Publish Validation" exercises the npm path, which shares most of the same artifacts.
118
+
119
+ ---
120
+
121
+ ## How to use this document
122
+
123
+ When you discover a new untestable scenario, add it here. When you close a gap, remove the row. Keep the document short.
@@ -1,55 +1,55 @@
1
1
  // @bun
2
- var $0=Object.defineProperty;var Q0=(z)=>z;function X0(z,Q){this[z]=Q0.bind(null,Q)}var l=(z,Q)=>{for(var $ in Q)$0(z,$,{get:Q[$],enumerable:!0,configurable:!0,set:X0.bind(Q,$)})};var f=(z,Q)=>()=>(z&&(Q=z(z=0)),Q);var Z0=import.meta.require;var L1={};l(L1,{runOrThrow:()=>B0,run:()=>O,commandVersion:()=>J0,commandExists:()=>R,ShellError:()=>K1});async function O(z,Q={}){let $=Bun.spawn({cmd:[...z],stdout:"pipe",stderr:"pipe",env:Q.env?{...process.env,...Q.env}:process.env,cwd:Q.cwd}),X;if(Q.timeoutMs&&Q.timeoutMs>0)X=setTimeout(()=>$.kill(),Q.timeoutMs);let[Z,H,V]=await Promise.all([new Response($.stdout).text(),new Response($.stderr).text(),$.exited]);if(X)clearTimeout(X);return{stdout:Z,stderr:H,exitCode:V}}async function B0(z,Q={}){let $=await O(z,Q);if($.exitCode!==0)throw new K1(`command failed (${$.exitCode}): ${z.join(" ")}`,$.exitCode,$.stdout,$.stderr);return $}async function R(z){let Q=q0(z),$=await O(["sh","-c",`command -v ${Q}`]);if($.exitCode===0)return $.stdout.trim()||null;return null}function q0(z){if(!/^[A-Za-z0-9._/-]+$/.test(z))throw Error(`refused to shell-escape suspect token: ${z}`);return z}async function J0(z,Q="--version"){if(!await R(z))return null;let X=await O([z,Q],{timeoutMs:5000});if(X.exitCode!==0)return null;return((X.stdout||X.stderr).split(/\r?\n/)[0]?.trim()??"")||null}var K1;var g=f(()=>{K1=class K1 extends Error{message;exitCode;stdout;stderr;constructor(z,Q,$,X){super(z);this.message=z;this.exitCode=Q;this.stdout=$;this.stderr=X;this.name="ShellError"}}});var k="\x1B[0;31m",x="\x1B[0;32m",T="\x1B[1;33m",J="\x1B[0;36m",F="\x1B[1m",Y="\x1B[2m",K="\x1B[0m";var h=()=>{};var k1={};l(k1,{lokiDir:()=>E,homeLokiDir:()=>H1,findSkillDir:()=>j0,REPO_ROOT:()=>y});import{resolve as b,dirname as P1}from"path";import{fileURLToPath as G0}from"url";import{existsSync as z1}from"fs";import{homedir as R1}from"os";function M0(){let z=x1;for(let Q=0;Q<6;Q++){if(z1(b(z,"VERSION"))&&z1(b(z,"autonomy/run.sh")))return z;let $=P1(z);if($===z)break;z=$}return b(x1,"..","..","..")}function E(){return process.env.LOKI_DIR??b(process.cwd(),".loki")}function H1(){return b(R1(),".loki")}function j0(){let z=[y,b(R1(),".claude/skills/loki-mode"),process.cwd()];for(let Q of z)if(z1(b(Q,"SKILL.md"))&&z1(b(Q,"autonomy/run.sh")))return Q;return null}var x1,y;var m=f(()=>{x1=P1(G0(import.meta.url));y=M0()});import{existsSync as x0}from"fs";async function n(){if(o!==void 0)return o;let z="/opt/homebrew/bin/python3.12";if(x0(z))return o=z,z;let Q=await R("python3.12");if(Q)return o=Q,Q;let $=await R("python3");return o=$,$}async function r(z,Q={}){let $=await n();if(!$)return{stdout:"",stderr:"python3 not found",exitCode:127};return O([$,"-c",z],Q)}var o;var $1=f(()=>{g()});var h1={};l(h1,{runStatus:()=>g0});import{existsSync as L,readFileSync as i,readdirSync as N1,statSync as C1}from"fs";import{resolve as w,basename as E0}from"path";async function b0(){if(await R("jq"))return!0;return process.stdout.write(`${k}Error: jq is required but not installed.${K}
2
+ var Q0=Object.defineProperty;var X0=(z)=>z;function Z0(z,Q){this[z]=X0.bind(null,Q)}var r=(z,Q)=>{for(var $ in Q)Q0(z,$,{get:Q[$],enumerable:!0,configurable:!0,set:Z0.bind(Q,$)})};var g=(z,Q)=>()=>(z&&(Q=z(z=0)),Q);var K0=import.meta.require;var P1={};r(P1,{lokiDir:()=>E,homeLokiDir:()=>B1,findSkillDir:()=>B0,findRepoRootForVersion:()=>W1,REPO_ROOT:()=>y});import{resolve as k,dirname as H1}from"path";import{fileURLToPath as H0}from"url";import{existsSync as u}from"fs";import{homedir as x1}from"os";function W0(){let z=L1;for(let Q=0;Q<6;Q++){if(u(k(z,"VERSION"))&&u(k(z,"autonomy/run.sh")))return z;let $=H1(z);if($===z)break;z=$}return k(L1,"..","..","..")}function W1(z){let Q=z;for(let $=0;$<6;$++){if(u(k(Q,"VERSION"))&&u(k(Q,"autonomy/run.sh")))return Q;let X=H1(Q);if(X===Q)break;Q=X}return k(z,"..","..","..")}function E(){return process.env.LOKI_DIR??k(process.cwd(),".loki")}function B1(){return k(x1(),".loki")}function B0(){let z=[y,k(x1(),".claude/skills/loki-mode"),process.cwd()];for(let Q of z)if(u(k(Q,"SKILL.md"))&&u(k(Q,"autonomy/run.sh")))return Q;return null}var L1,y;var h=g(()=>{L1=H1(H0(import.meta.url));y=W0()});var S1={};r(S1,{runOrThrow:()=>J0,run:()=>I,commandVersion:()=>M0,commandExists:()=>R,ShellError:()=>U1});async function I(z,Q={}){let $=Bun.spawn({cmd:[...z],stdout:"pipe",stderr:"pipe",env:Q.env?{...process.env,...Q.env}:process.env,cwd:Q.cwd}),X;if(Q.timeoutMs&&Q.timeoutMs>0)X=setTimeout(()=>$.kill(),Q.timeoutMs);let[Z,H,W]=await Promise.all([new Response($.stdout).text(),new Response($.stderr).text(),$.exited]);if(X)clearTimeout(X);return{stdout:Z,stderr:H,exitCode:W}}async function J0(z,Q={}){let $=await I(z,Q);if($.exitCode!==0)throw new U1(`command failed (${$.exitCode}): ${z.join(" ")}`,$.exitCode,$.stdout,$.stderr);return $}async function R(z){let Q=j0(z),$=await I(["sh","-c",`command -v ${Q}`],{timeoutMs:5000});if($.exitCode===0)return $.stdout.trim()||null;return null}function j0(z){if(!/^[A-Za-z0-9._/-]+$/.test(z))throw Error(`refused to shell-escape suspect token: ${z}`);return z}async function M0(z,Q="--version"){if(!await R(z))return null;let X=await I([z,Q],{timeoutMs:5000});if(X.exitCode!==0)return null;return((X.stdout||X.stderr).split(/\r?\n/)[0]?.trim()??"")||null}var U1;var m=g(()=>{U1=class U1 extends Error{message;exitCode;stdout;stderr;constructor(z,Q,$,X){super(z);this.message=z;this.exitCode=Q;this.stdout=$;this.stderr=X;this.name="ShellError"}}});function N(z){return Y0?"":z}var Y0,S,x,_,A3,G,F,A,K;var p=g(()=>{Y0=(process.env.NO_COLOR??"").length>0;S=N("\x1B[0;31m"),x=N("\x1B[0;32m"),_=N("\x1B[1;33m"),A3=N("\x1B[0;34m"),G=N("\x1B[0;36m"),F=N("\x1B[1m"),A=N("\x1B[2m"),K=N("\x1B[0m")});import{existsSync as P0}from"fs";async function a(){if(t!==void 0)return t;let z="/opt/homebrew/bin/python3.12";if(P0(z))return t=z,z;let Q=await R("python3.12");if(Q)return t=Q,Q;let $=await R("python3");return t=$,$}async function s(z,Q={}){let $=await a();if(!$)return{stdout:"",stderr:"python3 not found",exitCode:127};return I([$,"-c",z],Q)}var t;var Q1=g(()=>{m()});var v1={};r(v1,{runStatus:()=>y0});import{existsSync as L,readFileSync as e,readdirSync as C1,statSync as f1}from"fs";import{resolve as w,basename as E0}from"path";async function N0(){if(await R("jq"))return!0;return process.stdout.write(`${S}Error: jq is required but not installed.${K}
3
3
  `),process.stdout.write(`Install with:
4
4
  `),process.stdout.write(` brew install jq (macOS)
5
5
  `),process.stdout.write(` apt install jq (Debian/Ubuntu)
6
6
  `),process.stdout.write(` yum install jq (RHEL/CentOS)
7
- `),!1}function Q1(z){if(!Number.isFinite(z)||z<=0)return!1;try{return process.kill(z,0),!0}catch{return!1}}function X1(z){if(!L(z))return null;try{let Q=i(z,"utf-8").trim();if(!Q)return null;let $=Number.parseInt(Q,10);return Number.isFinite($)?$:null}catch{return null}}function N0(z){let Q=[],$=X1(w(z,"loki.pid"));if($!==null&&Q1($))Q.push(`global:${$}`);let X=w(z,"sessions");if(L(X)){let Z=[];try{Z=N1(X)}catch{Z=[]}for(let H of Z){let V=w(X,H);try{if(!C1(V).isDirectory())continue}catch{continue}let W=w(V,"loki.pid"),M=X1(W);if(M!==null&&Q1(M))Q.push(`${H}:${M}`)}}if(L(z)){let Z=[];try{Z=N1(z)}catch{Z=[]}for(let H of Z){if(!H.startsWith("run-")||!H.endsWith(".pid"))continue;let V=w(z,H);try{if(!C1(V).isFile())continue}catch{continue}let W=E0(H,".pid").slice(4),M=X1(V);if(M!==null&&Q1(M)){if(!Q.some((I)=>I.startsWith(`${W}:`)))Q.push(`${W}:${M}`)}}}return Q}async function f1(z,Q){let $=await O(["jq","-r",z,Q]);if($.exitCode!==0)return null;return $.stdout.trim()}function g1(z,Q){try{let $=i(z,"utf-8"),Z=JSON.parse($)[Q];if(typeof Z==="number"){if(Q==="budget_used"){let H=Math.round(Z*100)/100;if(Number.isInteger(H))return String(H);return String(H)}return String(Z)}if(Z===void 0||Z===null)return"0";return String(Z)}catch{return"0"}}function y1(z,Q,$){try{let X=i(z,"utf-8"),H=JSON.parse(X)[Q];if(typeof H==="number"&&Number.isFinite(H))return H;return $}catch{return $}}async function C0(){let z=E();if(!await b0())return 1;if(!L(z))return process.stdout.write(`${F}Loki Mode Status${K}
7
+ `),!1}function X1(z){if(!Number.isFinite(z)||z<=0)return!1;try{return process.kill(z,0),!0}catch{return!1}}function Z1(z){if(!L(z))return null;try{let Q=e(z,"utf-8").trim();if(!Q)return null;let $=Number.parseInt(Q,10);return Number.isFinite($)?$:null}catch{return null}}function C0(z){let Q=[],$=Z1(w(z,"loki.pid"));if($!==null&&X1($))Q.push(`global:${$}`);let X=w(z,"sessions");if(L(X)){let Z=[];try{Z=C1(X)}catch{Z=[]}for(let H of Z){let W=w(X,H);try{if(!f1(W).isDirectory())continue}catch{continue}let B=w(W,"loki.pid"),j=Z1(B);if(j!==null&&X1(j))Q.push(`${H}:${j}`)}}if(L(z)){let Z=[];try{Z=C1(z)}catch{Z=[]}for(let H of Z){if(!H.startsWith("run-")||!H.endsWith(".pid"))continue;let W=w(z,H);try{if(!f1(W).isFile())continue}catch{continue}let B=E0(H,".pid").slice(4),j=Z1(W);if(j!==null&&X1(j)){if(!Q.some((T)=>T.startsWith(`${B}:`)))Q.push(`${B}:${j}`)}}}return Q}async function g1(z,Q){let $=await I(["jq","-r",z,Q]);if($.exitCode!==0)return null;return $.stdout.trim()}function y1(z,Q){try{let $=e(z,"utf-8"),Z=JSON.parse($)[Q];if(typeof Z==="number"){if(Q==="budget_used"){let H=Math.round(Z*100)/100;if(Number.isInteger(H))return String(H);return String(H)}return String(Z)}if(Z===void 0||Z===null)return"0";return String(Z)}catch{return"0"}}function h1(z,Q,$){try{let X=e(z,"utf-8"),H=JSON.parse(X)[Q];if(typeof H==="number"&&Number.isFinite(H))return H;return $}catch{return $}}async function f0(){let z=E();if(!await N0())return 1;if(!L(z))return process.stdout.write(`${F}Loki Mode Status${K}
8
8
  `),process.stdout.write(`
9
- `),process.stdout.write(`${T}No active session found.${K}
9
+ `),process.stdout.write(`${_}No active session found.${K}
10
10
  `),process.stdout.write(`Loki Mode has not been initialized in this directory.
11
11
  `),process.stdout.write(`
12
12
  `),process.stdout.write(`To start a session:
13
13
  `),process.stdout.write(` loki start <prd> - Start with a PRD file
14
14
  `),process.stdout.write(` loki start - Start without a PRD
15
15
  `),process.stdout.write(`
16
- `),process.stdout.write(`${Y}Current directory: ${process.cwd()}${K}
16
+ `),process.stdout.write(`${A}Current directory: ${process.cwd()}${K}
17
17
  `),0;process.stdout.write(`${F}Loki Mode Status${K}
18
18
  `),process.stdout.write(`
19
- `);let Q="",$=w(z,"state","provider");if(L($))try{Q=i($,"utf-8").trim()}catch{Q=""}let X=Q||process.env.LOKI_PROVIDER||"claude",Z="full features";switch(X){case"codex":case"gemini":case"aider":Z="degraded mode";break;case"cline":Z="near-full mode";break;default:Z="full features";break}process.stdout.write(`${J}Provider:${K} ${X} (${Z})
20
- `),process.stdout.write(`${Y} Switch with: loki provider set <claude|codex|gemini|cline|aider>${K}
19
+ `);let Q="",$=w(z,"state","provider");if(L($))try{Q=e($,"utf-8").trim()}catch{Q=""}let X=Q||process.env.LOKI_PROVIDER||"claude",Z="full features";switch(X){case"codex":case"gemini":case"aider":Z="degraded mode";break;case"cline":Z="near-full mode";break;default:Z="full features";break}process.stdout.write(`${G}Provider:${K} ${X} (${Z})
20
+ `),process.stdout.write(`${A} Switch with: loki provider set <claude|codex|gemini|cline|aider>${K}
21
21
  `),process.stdout.write(`
22
- `);let H=N0(z);if(H.length>0){process.stdout.write(`${x}Active Sessions: ${H.length}${K}
23
- `);for(let B of H){let U=B.indexOf(":"),_=U>=0?B.slice(0,U):B,D=U>=0?B.slice(U+1):"";if(_==="global")process.stdout.write(` ${J}[global]${K} PID ${D}
24
- `);else process.stdout.write(` ${J}[#${_}]${K} PID ${D}
22
+ `);let H=C0(z);if(H.length>0){process.stdout.write(`${x}Active Sessions: ${H.length}${K}
23
+ `);for(let V of H){let U=V.indexOf(":"),Y=U>=0?V.slice(0,U):V,b=U>=0?V.slice(U+1):"";if(Y==="global")process.stdout.write(` ${G}[global]${K} PID ${b}
24
+ `);else process.stdout.write(` ${G}[#${Y}]${K} PID ${b}
25
25
  `)}process.stdout.write(`
26
- `),process.stdout.write(`${Y} Stop specific: loki stop <session-id>${K}
27
- `),process.stdout.write(`${Y} Stop all: loki stop${K}
26
+ `),process.stdout.write(`${A} Stop specific: loki stop <session-id>${K}
27
+ `),process.stdout.write(`${A} Stop all: loki stop${K}
28
28
  `),process.stdout.write(`
29
- `)}if(L(w(z,"PAUSE")))process.stdout.write(`${T}Status: PAUSED${K}
30
- `),process.stdout.write(`${Y} Resume with: loki resume${K}
29
+ `)}if(L(w(z,"PAUSE")))process.stdout.write(`${_}Status: PAUSED${K}
30
+ `),process.stdout.write(`${A} Resume with: loki resume${K}
31
31
  `),process.stdout.write(`
32
- `);else if(L(w(z,"STOP")))process.stdout.write(`${k}Status: STOPPED${K}
33
- `),process.stdout.write(`${Y} Clear with: loki resume${K}
32
+ `);else if(L(w(z,"STOP")))process.stdout.write(`${S}Status: STOPPED${K}
33
+ `),process.stdout.write(`${A} Clear with: loki resume${K}
34
34
  `),process.stdout.write(`
35
- `);let V=w(z,"STATUS.txt");if(L(V)){process.stdout.write(`${J}Session Info:${K}
36
- `);try{process.stdout.write(i(V,"utf-8"))}catch{}process.stdout.write(`
37
- `)}let W=w(z,"state","orchestrator.json");if(L(W)){process.stdout.write(`${J}Orchestrator State:${K}
38
- `);let B=await f1('.currentPhase // "unknown"',W);process.stdout.write(`${B??"unknown"}
39
- `)}let M=w(z,"queue","pending.json");if(L(M)){let B=await f1('if type == "array" then length elif .tasks then .tasks | length else 0 end',M);process.stdout.write(`${J}Pending Tasks:${K} ${B??"0"}
40
- `)}let A=w(z,"metrics","budget.json");if(L(A)){let B=g1(A,"budget_limit"),U=g1(A,"budget_used");if(B!=="0")process.stdout.write(`${J}Budget:${K} $${U} / $${B}
41
- `);else process.stdout.write(`${J}Cost:${K} $${U} (no limit)
42
- `)}let I=w(z,"state","context-usage.json");if(L(I)){let B=y1(I,"window_size",200000),U=y1(I,"used_tokens",0),_=0;if(B>0)_=Math.floor(U*100/B);process.stdout.write(`${J}Context:${K} ${_}% (${U} / ${B} tokens)
43
- `)}let P=w(z,"dashboard","dashboard.pid");if(L(P)){let B=X1(P);if(B!==null&&Q1(B)){let U=process.env.LOKI_DASHBOARD_PORT||"57374";process.stdout.write(`${J}Dashboard:${K} http://127.0.0.1:${U}/
35
+ `);let W=w(z,"STATUS.txt");if(L(W)){process.stdout.write(`${G}Session Info:${K}
36
+ `);try{process.stdout.write(e(W,"utf-8"))}catch{}process.stdout.write(`
37
+ `)}let B=w(z,"state","orchestrator.json");if(L(B)){process.stdout.write(`${G}Orchestrator State:${K}
38
+ `);let V=await g1('.currentPhase // "unknown"',B);process.stdout.write(`${V??"unknown"}
39
+ `)}let j=w(z,"queue","pending.json");if(L(j)){let V=await g1('if type == "array" then length elif .tasks then .tasks | length else 0 end',j);process.stdout.write(`${G}Pending Tasks:${K} ${V??"0"}
40
+ `)}let O=w(z,"metrics","budget.json");if(L(O)){let V=y1(O,"budget_limit"),U=y1(O,"budget_used");if(V!=="0")process.stdout.write(`${G}Budget:${K} $${U} / $${V}
41
+ `);else process.stdout.write(`${G}Cost:${K} $${U} (no limit)
42
+ `)}let T=w(z,"state","context-usage.json");if(L(T)){let V=h1(T,"window_size",200000),U=h1(T,"used_tokens",0),Y=0;if(V>0)Y=Math.floor(U*100/V);process.stdout.write(`${G}Context:${K} ${Y}% (${U} / ${V} tokens)
43
+ `)}let P=w(z,"dashboard","dashboard.pid");if(L(P)){let V=Z1(P);if(V!==null&&X1(V)){let U=process.env.LOKI_DASHBOARD_PORT||"57374";process.stdout.write(`${G}Dashboard:${K} http://127.0.0.1:${U}/
44
44
  `)}}return process.stdout.write(`
45
- `),process.stdout.write(`${Y} Tip: loki context show - detailed token breakdown${K}
46
- `),process.stdout.write(`${Y} Tip: loki code overview - codebase intelligence${K}
47
- `),0}async function f0(){let z=await n();if(!z)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
48
- `),1;let Q=y,$=E(),X=process.env.LOKI_DASHBOARD_PORT||"57374",Z=process.env.LOKI_PROVIDER||"claude",H=await O([z,"-c",D0,Q,$,X,Z]);if(H.exitCode!==0)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
49
- `),1;return process.stdout.write(H.stdout),0}async function g0(z){let Q=[...z];while(Q.length>0){let $=Q[0];if($==="--json")return f0();if($==="--help"||$==="-h")return process.stdout.write(`Usage: loki status [--json]
50
- `),0;return process.stdout.write(`${k}Unknown flag: ${$}${K}
45
+ `),process.stdout.write(`${A} Tip: loki context show - detailed token breakdown${K}
46
+ `),process.stdout.write(`${A} Tip: loki code overview - codebase intelligence${K}
47
+ `),0}async function g0(){let z=await a();if(!z)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
48
+ `),1;let Q=y,$=E(),X=process.env.LOKI_DASHBOARD_PORT||"57374",Z=process.env.LOKI_PROVIDER||"claude",H=await I([z,"-c",b0,Q,$,X,Z],{timeoutMs:30000});if(H.exitCode!==0)return process.stderr.write(`{"error": "Failed to generate JSON status. Ensure python3 is available."}
49
+ `),1;return process.stdout.write(H.stdout),0}async function y0(z){let Q=[...z];while(Q.length>0){let $=Q[0];if($==="--json")return g0();if($==="--help"||$==="-h")return process.stdout.write(`Usage: loki status [--json]
50
+ `),0;return process.stdout.write(`${S}Unknown flag: ${$}${K}
51
51
  `),process.stdout.write(`Usage: loki status [--json]
52
- `),1}return C0()}var D0=`
52
+ `),1}return f0()}var b0=`
53
53
  import json, os, sys, time
54
54
 
55
55
  skill_dir = sys.argv[1]
@@ -195,9 +195,9 @@ if os.path.isdir(queue_dir):
195
195
  result['task_counts'] = task_counts
196
196
 
197
197
  print(json.dumps(result, indent=2))
198
- `;var m1=f(()=>{g();$1();h();m()});var c1={};l(c1,{runStats:()=>p0,computeStats:()=>p1});import{readdirSync as v1,readFileSync as y0,statSync as u1}from"fs";import{join as N}from"path";function u(z){try{if(!u1(z).isFile())return null;return JSON.parse(y0(z,"utf-8"))}catch{return null}}function B1(z){try{return u1(z).isDirectory()}catch{return!1}}function h0(z){if(!B1(z))return[];try{let Q=v1(z).filter(($)=>$.startsWith("iteration-")&&$.endsWith(".json"));return Q.sort(),Q.map(($)=>N(z,$))}catch{return[]}}function p(z){return Math.trunc(z).toLocaleString("en-US")}function W1(z){let Q=Math.trunc(z);if(Q<60)return`${Q}s`;let $=Math.trunc(Q/3600),X=Math.trunc(Q%3600/60),Z=Q%60;if($>0)return`${$}h ${String(X).padStart(2,"0")}m`;return`${X}m ${String(Z).padStart(2,"0")}s`}function C(z,Q=0){let $=Math.pow(10,Q);return Math.round(z*$)/$}function c(z,Q){return z.toFixed(Q)}function U1(z,Q){return z.length>=Q?z:z+" ".repeat(Q-z.length)}function m0(z){let Q="N/A",$=0,X=u(N(z,"state","orchestrator.json"));if(X&&typeof X==="object"){if(typeof X.currentPhase==="string")Q=X.currentPhase;if(typeof X.currentIteration==="number")$=X.currentIteration}let Z=N(z,"metrics","efficiency"),H=h0(Z),V=[];for(let q of H){let j=u(q);if(j&&typeof j==="object")V.push(j)}if(V.length>0)$=Math.max($,V.length);let W=V.reduce((q,j)=>q+(j.input_tokens??0),0),M=V.reduce((q,j)=>q+(j.output_tokens??0),0),A=W+M,I=V.reduce((q,j)=>q+(j.cost_usd??0),0),P=V.reduce((q,j)=>q+(j.duration_seconds??0),0),B=0,U=0,_=u(N(z,"metrics","budget.json"));if(_&&typeof _==="object"){if(typeof _.budget_limit==="number")B=_.budget_limit;if(typeof _.budget_used==="number")U=_.budget_used}let D=0,a=0,d=u(N(z,"state","quality-gates.json"));if(d&&typeof d==="object"){if(Array.isArray(d)){for(let q of d)if(a+=1,q===!0)D+=1;else if(q&&typeof q==="object"){let j=q;if(j.passed===!0||j.status==="passed")D+=1}}else for(let q of Object.values(d))if(typeof q==="boolean"){if(a+=1,q)D+=1}else if(q&&typeof q==="object"){a+=1;let j=q;if(j.passed===!0||j.status==="passed")D+=1}}let _1={},s=u(N(z,"quality","gate-failure-count.json"));if(s&&typeof s==="object"&&!Array.isArray(s)){let q={};for(let[j,S]of Object.entries(s))if(typeof S==="number")q[j]=S;_1=q}let Y1=0,A1=0,T1=0,Z1=N(z,"quality");if(B1(Z1)){let q=[];try{q=v1(Z1)}catch{q=[]}for(let j of q){if(!j.endsWith(".json")||j==="gate-failure-count.json")continue;let S=u(N(Z1,j));if(!S||typeof S!=="object")continue;if(!(("verdict"in S)||("approved"in S)||("reviewers"in S)))continue;Y1+=1;let I1=(S.verdict??"").toString().toLowerCase();if(S.approved===!0||["approved","approve","pass"].includes(I1))A1+=1;else if(["revision","revise","changes_requested","reject"].includes(I1))T1+=1}}return{phase:Q,iterationCount:$,iterations:V,totalInput:W,totalOutput:M,totalTokens:A,totalCost:I,totalDuration:P,budgetLimit:B,budgetUsed:U,gatesPassed:D,gatesTotal:a,gateFailures:_1,reviewsTotal:Y1,reviewsApproved:A1,reviewsRevision:T1}}function v0(z,Q){let $=z.iterationCount,X={session:{iterations:$,duration_seconds:z.totalDuration,phase:z.phase},tokens:{input:z.totalInput,output:z.totalOutput,total:z.totalTokens,cost_usd:C(z.totalCost,2)},quality:{gates_passed:z.gatesPassed,gates_total:z.gatesTotal,reviews_total:z.reviewsTotal,reviews_approved:z.reviewsApproved,reviews_revision:z.reviewsRevision,gate_failures:z.gateFailures},efficiency:{avg_tokens_per_iteration:$>0?C(z.totalTokens/$,0):0,avg_cost_per_iteration:$>0?C(z.totalCost/$,2):0,avg_duration_per_iteration:$>0?C(z.totalDuration/$,1):0},budget:{used:C(z.budgetUsed,2),limit:z.budgetLimit,percent:z.budgetLimit>0?C(z.budgetUsed/z.budgetLimit*100,1):0}};if(Q)X.iterations=z.iterations.map((V,W)=>({number:W+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 H(V,W){if(!W)return;let M=new RegExp(`("${V}": )(-?\\d+)(,?)$`,"m");Z=Z.replace(M,(A,I,P,B)=>`${I}${P}.0${B}`)}if(H("avg_duration_per_iteration",$>0&&Number.isInteger(X.efficiency.avg_duration_per_iteration)),H("percent",z.budgetLimit>0&&Number.isInteger(X.budget.percent)),H("cost_usd",$>0&&Number.isInteger(X.tokens.cost_usd)),Q)Z=Z.replace(/("cost_usd": )(-?\d+)(,?)$/gm,(V,W,M,A)=>`${W}${M}.0${A}`);return Z}function u0(z,Q){let $=[];if($.push("Loki Mode Session Statistics"),$.push("============================"),$.push(""),$.push("Session"),$.push(` Iterations completed: ${z.iterationCount}`),$.push(` Duration: ${W1(z.totalDuration)}`),$.push(` Current phase: ${z.phase}`),$.push(""),$.push("Token Usage"),z.iterations.length>0)$.push(` Input tokens: ${p(z.totalInput)}`),$.push(` Output tokens: ${p(z.totalOutput)}`),$.push(` Total tokens: ${p(z.totalTokens)}`),$.push(` Estimated cost: $${c(z.totalCost,2)}`);else $.push(" N/A (no iteration metrics found)");if($.push(""),$.push("Quality Gates"),z.gatesTotal>0){let X=Math.round(z.gatesPassed/z.gatesTotal*100);$.push(` Gates passed: ${z.gatesPassed}/${z.gatesTotal} (${X}%)`)}else $.push(" Gates passed: N/A");if(z.reviewsTotal>0){let X=[];if(z.reviewsApproved>0)X.push(`${z.reviewsApproved} approved`);if(z.reviewsRevision>0)X.push(`${z.reviewsRevision} revision requested`);let Z=X.length>0?X.join(", "):"N/A";$.push(` Code reviews: ${z.reviewsTotal} (${Z})`)}if(Object.keys(z.gateFailures).length>0){let X=Object.entries(z.gateFailures).filter(([,Z])=>Z>0).map(([Z,H])=>`${Z} (${H})`);if(X.length>0)$.push(` Gate failures: ${X.join(", ")}`)}if($.push(""),$.push("Efficiency"),z.iterationCount>0&&z.iterations.length>0){let X=Math.round(z.totalTokens/z.iterationCount),Z=z.totalCost/z.iterationCount,H=z.totalDuration/z.iterationCount;$.push(` Avg tokens/iteration: ${p(X)}`),$.push(` Avg cost/iteration: $${c(Z,2)}`),$.push(` Avg duration/iteration: ${W1(H)}`)}else $.push(" N/A (no iteration metrics found)");if($.push(""),$.push("Budget"),z.budgetLimit>0){let X=C(z.budgetUsed/z.budgetLimit*100,1),Z=Number.isInteger(X)?`${X}.0`:`${X}`;$.push(` Used: $${c(z.budgetUsed,2)} / $${c(z.budgetLimit,2)} (${Z}%)`)}else if(z.budgetUsed>0)$.push(` Used: $${c(z.budgetUsed,2)} (no limit set)`);else $.push(" N/A");if(Q&&z.iterations.length>0)$.push(""),$.push("Per-Iteration Breakdown"),z.iterations.forEach((X,Z)=>{let H=Z+1,V=U1(p(X.input_tokens??0),10),W=U1(p(X.output_tokens??0),10),M=X.cost_usd??0,A=W1(X.duration_seconds??0),I=U1(`${H}`,3);$.push(` #${I} input: ${V} output: ${W} cost: $${c(M,2)} time: ${A}`)});return $.join(`
199
- `)}function p1(z){let Q=!1,$=!1;for(let V of z)if(V==="--json")Q=!0;else if(V==="--efficiency")$=!0;let X=E();if(!B1(X)){if(Q)return{exitCode:0,stdout:'{"error": "No active session"}'};return{exitCode:0,stdout:`${T}No active session found.${K}
200
- Start a session with: loki start <prd>`}}let Z=m0(X);return{exitCode:0,stdout:Q?v0(Z,$):u0(Z,$)}}async function p0(z){let Q=p1(z);return console.log(Q.stdout),Q.exitCode}var d1=f(()=>{m();h()});var s1={};l(s1,{runDoctor:()=>z6,httpReachable:()=>M1,checkTool:()=>r1,checkSkills:()=>t1,checkDisk:()=>j1,buildDoctorJson:()=>a1});import{existsSync as c0,lstatSync as d0,readlinkSync as l0,statfsSync as o0}from"fs";import{homedir as o1}from"os";import{resolve as l1}from"path";function r0(z){let Q=z.match(n0);return Q?Q[1]:null}async function t0(z){try{let Q=await O([z,"--version"],{timeoutMs:5000}),$=(Q.stdout||Q.stderr||"").trim();return r0($)}catch{return null}}function n1(z,Q){let $=z.split(".").map((Z)=>parseInt(Z,10)),X=Q.split(".").map((Z)=>parseInt(Z,10));while($.length<2)$.push(0);while(X.length<2)X.push(0);for(let Z=0;Z<2;Z++){let H=$[Z]??0,V=X[Z]??0;if(Number.isNaN(H)||Number.isNaN(V))return 0;if(H!==V)return H-V}return 0}async function r1(z,Q,$,X=null){let Z=await R(Q),H=Z!==null,V=H?await t0(Q):null,W="pass";if(!H)W=$==="required"?"fail":"warn";else if(X&&V){if(n1(V,X)<0)W=$==="required"?"fail":"warn"}return{name:z,command:Q,found:H,version:V,required:$,min_version:X,status:W,path:Z}}function j1(){let z=null;try{let $=o0(o1()),X=Number($.bavail)*Number($.bsize);z=Math.round(X/1073741824*10)/10}catch{z=null}let Q="pass";if(z!==null){if(z<1)Q="fail";else if(z<5)Q="warn"}return{available_gb:z,status:Q}}async function M1(z,Q=2000){try{return(await fetch(z,{signal:AbortSignal.timeout(Q)})).ok}catch{return!1}}async function q1(z,Q=!1){let $=`import ${z}`;if(!Q)return(await r($,{timeoutMs:5000})).exitCode===0;let X=await n();if(!X)return!1;return(await O([X,"-c",$],{timeoutMs:5000})).exitCode===0}function t1(){let z=o1();return i0.map(({name:Q,dir:$})=>{let X=l1(z,$),Z=X,H=l1(X,"SKILL.md");if(c0(H))return{name:Q,path:Z,status:"pass",detail:""};try{if(d0(X).isSymbolicLink()){let W="unknown";try{W=l0(X)}catch{}return{name:Q,path:Z,status:"fail",detail:`(broken symlink -> ${W})`}}}catch{}return{name:Q,path:Z,status:"warn",detail:"(not found - run 'loki setup-skill')"}})}async function i1(){return Promise.all(a0.map(async(z)=>{return{...await r1(z.jsonName,z.cmd,z.required,z.min??null),displayName:z.displayName}}))}async function a1(){let Q=(await i1()).map(({displayName:V,...W})=>W),$=j1(),X=0,Z=0,H=0;for(let V of Q)if(V.status==="pass")X++;else if(V.status==="fail")Z++;else H++;if($.status==="pass")X++;else if($.status==="fail")Z++;else H++;return{checks:Q,disk:$,summary:{passed:X,failed:Z,warnings:H,ok:Z===0}}}function G(z){switch(z){case"pass":return`${x}PASS${K}`;case"fail":return`${k}FAIL${K}`;case"warn":return`${T}WARN${K}`}}function J1(z){let Q=z.version?` (v${z.version})`:"",$=z.displayName;if(!z.found){let X=z.required==="required"?"not found":z.required==="recommended"?"not found (recommended)":"not found (optional)";return` ${G(z.status)} ${$} - ${X}`}if(z.min_version&&z.version&&n1(z.version,z.min_version)<0){let X=z.required==="required"?"requires":"recommended";return` ${G(z.status)} ${$}${Q} - ${X} >= ${z.min_version}`}return` ${G(z.status)} ${$}${Q}`}function G1(z,Q){if(Q==="pass")z.pass++;else if(Q==="fail")z.fail++;else z.warn++}function s0(){process.stdout.write(`${F}loki doctor${K} - Check system prerequisites
198
+ `;var m1=g(()=>{m();Q1();p();h()});var l1={};r(l1,{runStats:()=>c0,computeStats:()=>c1});import{readdirSync as u1,readFileSync as h0,statSync as p1}from"fs";import{join as C}from"path";function l(z){try{if(!p1(z).isFile())return null;return JSON.parse(h0(z,"utf-8"))}catch{return null}}function J1(z){try{return p1(z).isDirectory()}catch{return!1}}function v0(z){if(!J1(z))return[];try{let Q=u1(z).filter(($)=>$.startsWith("iteration-")&&$.endsWith(".json"));return Q.sort(),Q.map(($)=>C(z,$))}catch{return[]}}function d(z){return Math.trunc(z).toLocaleString("en-US")}function q1(z){let Q=Math.trunc(z);if(Q<60)return`${Q}s`;let $=Math.trunc(Q/3600),X=Math.trunc(Q%3600/60),Z=Q%60;if($>0)return`${$}h ${String(X).padStart(2,"0")}m`;return`${X}m ${String(Z).padStart(2,"0")}s`}function f(z,Q=0){let $=Math.pow(10,Q);return Math.round(z*$)/$}function o(z,Q){return z.toFixed(Q)}function G1(z,Q){return z.length>=Q?z:z+" ".repeat(Q-z.length)}function m0(z){let Q="N/A",$=0,X=l(C(z,"state","orchestrator.json"));if(X&&typeof X==="object"){if(typeof X.currentPhase==="string")Q=X.currentPhase;if(typeof X.currentIteration==="number")$=X.currentIteration}let Z=C(z,"metrics","efficiency"),H=v0(Z),W=[];for(let q of H){let M=l(q);if(M&&typeof M==="object")W.push(M)}if(W.length>0)$=Math.max($,W.length);let B=W.reduce((q,M)=>q+(M.input_tokens??0),0),j=W.reduce((q,M)=>q+(M.output_tokens??0),0),O=B+j,T=W.reduce((q,M)=>q+(M.cost_usd??0),0),P=W.reduce((q,M)=>q+(M.duration_seconds??0),0),V=0,U=0,Y=l(C(z,"metrics","budget.json"));if(Y&&typeof Y==="object"){if(typeof Y.budget_limit==="number")V=Y.budget_limit;if(typeof Y.budget_used==="number")U=Y.budget_used}let b=0,z1=0,n=l(C(z,"state","quality-gates.json"));if(n&&typeof n==="object"){if(Array.isArray(n)){for(let q of n)if(z1+=1,q===!0)b+=1;else if(q&&typeof q==="object"){let M=q;if(M.passed===!0||M.status==="passed")b+=1}}else for(let q of Object.values(n))if(typeof q==="boolean"){if(z1+=1,q)b+=1}else if(q&&typeof q==="object"){z1+=1;let M=q;if(M.passed===!0||M.status==="passed")b+=1}}let _1={},$1=l(C(z,"quality","gate-failure-count.json"));if($1&&typeof $1==="object"&&!Array.isArray($1)){let q={};for(let[M,D]of Object.entries($1))if(typeof D==="number")q[M]=D;_1=q}let T1=0,I1=0,w1=0,K1=C(z,"quality");if(J1(K1)){let q=[];try{q=u1(K1)}catch{q=[]}for(let M of q){if(!M.endsWith(".json")||M==="gate-failure-count.json")continue;let D=l(C(K1,M));if(!D||typeof D!=="object")continue;if(!(("verdict"in D)||("approved"in D)||("reviewers"in D)))continue;T1+=1;let F1=(D.verdict??"").toString().toLowerCase();if(D.approved===!0||["approved","approve","pass"].includes(F1))I1+=1;else if(["revision","revise","changes_requested","reject"].includes(F1))w1+=1}}return{phase:Q,iterationCount:$,iterations:W,totalInput:B,totalOutput:j,totalTokens:O,totalCost:T,totalDuration:P,budgetLimit:V,budgetUsed:U,gatesPassed:b,gatesTotal:z1,gateFailures:_1,reviewsTotal:T1,reviewsApproved:I1,reviewsRevision:w1}}function u0(z,Q){let $=z.iterationCount,X={session:{iterations:$,duration_seconds:z.totalDuration,phase:z.phase},tokens:{input:z.totalInput,output:z.totalOutput,total:z.totalTokens,cost_usd:f(z.totalCost,2)},quality:{gates_passed:z.gatesPassed,gates_total:z.gatesTotal,reviews_total:z.reviewsTotal,reviews_approved:z.reviewsApproved,reviews_revision:z.reviewsRevision,gate_failures:z.gateFailures},efficiency:{avg_tokens_per_iteration:$>0?f(z.totalTokens/$,0):0,avg_cost_per_iteration:$>0?f(z.totalCost/$,2):0,avg_duration_per_iteration:$>0?f(z.totalDuration/$,1):0},budget:{used:f(z.budgetUsed,2),limit:z.budgetLimit,percent:z.budgetLimit>0?f(z.budgetUsed/z.budgetLimit*100,1):0}};if(Q)X.iterations=z.iterations.map((W,B)=>({number:B+1,input_tokens:W.input_tokens??0,output_tokens:W.output_tokens??0,cost_usd:f(W.cost_usd??0,2),duration_seconds:W.duration_seconds??0}));let Z=JSON.stringify(X,null,2);function H(W,B){if(!B)return;let j=new RegExp(`("${W}": )(-?\\d+)(,?)$`,"m");Z=Z.replace(j,(O,T,P,V)=>`${T}${P}.0${V}`)}if(H("avg_duration_per_iteration",$>0&&Number.isInteger(X.efficiency.avg_duration_per_iteration)),H("percent",z.budgetLimit>0&&Number.isInteger(X.budget.percent)),H("cost_usd",$>0&&Number.isInteger(X.tokens.cost_usd)),Q)Z=Z.replace(/("cost_usd": )(-?\d+)(,?)$/gm,(W,B,j,O)=>`${B}${j}.0${O}`);return Z}function p0(z,Q){let $=[];if($.push("Loki Mode Session Statistics"),$.push("============================"),$.push(""),$.push("Session"),$.push(` Iterations completed: ${z.iterationCount}`),$.push(` Duration: ${q1(z.totalDuration)}`),$.push(` Current phase: ${z.phase}`),$.push(""),$.push("Token Usage"),z.iterations.length>0)$.push(` Input tokens: ${d(z.totalInput)}`),$.push(` Output tokens: ${d(z.totalOutput)}`),$.push(` Total tokens: ${d(z.totalTokens)}`),$.push(` Estimated cost: $${o(z.totalCost,2)}`);else $.push(" N/A (no iteration metrics found)");if($.push(""),$.push("Quality Gates"),z.gatesTotal>0){let X=Math.round(z.gatesPassed/z.gatesTotal*100);$.push(` Gates passed: ${z.gatesPassed}/${z.gatesTotal} (${X}%)`)}else $.push(" Gates passed: N/A");if(z.reviewsTotal>0){let X=[];if(z.reviewsApproved>0)X.push(`${z.reviewsApproved} approved`);if(z.reviewsRevision>0)X.push(`${z.reviewsRevision} revision requested`);let Z=X.length>0?X.join(", "):"N/A";$.push(` Code reviews: ${z.reviewsTotal} (${Z})`)}if(Object.keys(z.gateFailures).length>0){let X=Object.entries(z.gateFailures).filter(([,Z])=>Z>0).map(([Z,H])=>`${Z} (${H})`);if(X.length>0)$.push(` Gate failures: ${X.join(", ")}`)}if($.push(""),$.push("Efficiency"),z.iterationCount>0&&z.iterations.length>0){let X=Math.round(z.totalTokens/z.iterationCount),Z=z.totalCost/z.iterationCount,H=z.totalDuration/z.iterationCount;$.push(` Avg tokens/iteration: ${d(X)}`),$.push(` Avg cost/iteration: $${o(Z,2)}`),$.push(` Avg duration/iteration: ${q1(H)}`)}else $.push(" N/A (no iteration metrics found)");if($.push(""),$.push("Budget"),z.budgetLimit>0){let X=f(z.budgetUsed/z.budgetLimit*100,1),Z=Number.isInteger(X)?`${X}.0`:`${X}`;$.push(` Used: $${o(z.budgetUsed,2)} / $${o(z.budgetLimit,2)} (${Z}%)`)}else if(z.budgetUsed>0)$.push(` Used: $${o(z.budgetUsed,2)} (no limit set)`);else $.push(" N/A");if(Q&&z.iterations.length>0)$.push(""),$.push("Per-Iteration Breakdown"),z.iterations.forEach((X,Z)=>{let H=Z+1,W=G1(d(X.input_tokens??0),10),B=G1(d(X.output_tokens??0),10),j=X.cost_usd??0,O=q1(X.duration_seconds??0),T=G1(`${H}`,3);$.push(` #${T} input: ${W} output: ${B} cost: $${o(j,2)} time: ${O}`)});return $.join(`
199
+ `)}function c1(z){let Q=!1,$=!1;for(let W of z)if(W==="--json")Q=!0;else if(W==="--efficiency")$=!0;let X=E();if(!J1(X)){if(Q)return{exitCode:0,stdout:'{"error": "No active session"}'};return{exitCode:0,stdout:`${_}No active session found.${K}
200
+ Start a session with: loki start <prd>`}}let Z=m0(X);return{exitCode:0,stdout:Q?u0(Z,$):p0(Z,$)}}async function c0(z){let Q=c1(z);return console.log(Q.stdout),Q.exitCode}var d1=g(()=>{h();p()});var e1={};r(e1,{runDoctor:()=>$3,httpReachable:()=>A1,checkTool:()=>t1,checkSkills:()=>a1,checkDisk:()=>O1,buildDoctorJson:()=>i1});import{existsSync as l0,lstatSync as d0,readlinkSync as o0,statfsSync as n0}from"fs";import{homedir as n1}from"os";import{resolve as o1}from"path";function t0(z){let Q=z.match(r0);return Q?Q[1]:null}async function a0(z){try{let Q=await I([z,"--version"],{timeoutMs:5000}),$=(Q.stdout||Q.stderr||"").trim();return t0($)}catch{return null}}function r1(z,Q){let $=z.split(".").map((Z)=>parseInt(Z,10)),X=Q.split(".").map((Z)=>parseInt(Z,10));while($.length<2)$.push(0);while(X.length<2)X.push(0);for(let Z=0;Z<2;Z++){let H=$[Z]??0,W=X[Z]??0;if(Number.isNaN(H)||Number.isNaN(W))return 0;if(H!==W)return H-W}return 0}async function t1(z,Q,$,X=null){let Z=await R(Q),H=Z!==null,W=H?await a0(Q):null,B="pass";if(!H)B=$==="required"?"fail":"warn";else if(X&&W){if(r1(W,X)<0)B=$==="required"?"fail":"warn"}return{name:z,command:Q,found:H,version:W,required:$,min_version:X,status:B,path:Z}}function O1(){let z=null;try{let $=n0(n1()),X=Number($.bavail)*Number($.bsize);z=Math.round(X/1073741824*10)/10}catch{z=null}let Q="pass";if(z!==null){if(z<1)Q="fail";else if(z<5)Q="warn"}return{available_gb:z,status:Q}}async function A1(z,Q=2000){try{return(await fetch(z,{signal:AbortSignal.timeout(Q)})).ok}catch{return!1}}async function j1(z,Q=!1){let $=`import ${z}`,X=Q?30000:5000;if(!Q)return(await s($,{timeoutMs:X})).exitCode===0;let Z=await a();if(!Z)return!1;return(await I([Z,"-c",$],{timeoutMs:X})).exitCode===0}function a1(){let z=n1();return s0.map(({name:Q,dir:$})=>{let X=o1(z,$),Z=X,H=o1(X,"SKILL.md");if(l0(H))return{name:Q,path:Z,status:"pass",detail:""};try{if(d0(X).isSymbolicLink()){let B="unknown";try{B=o0(X)}catch{}return{name:Q,path:Z,status:"fail",detail:`(broken symlink -> ${B})`}}}catch{}return{name:Q,path:Z,status:"warn",detail:"(not found - run 'loki setup-skill')"}})}async function s1(){return Promise.all(i0.map(async(z)=>{return{...await t1(z.jsonName,z.cmd,z.required,z.min??null),displayName:z.displayName}}))}async function i1(){let Q=(await s1()).map(({displayName:W,...B})=>B),$=O1(),X=0,Z=0,H=0;for(let W of Q)if(W.status==="pass")X++;else if(W.status==="fail")Z++;else H++;if($.status==="pass")X++;else if($.status==="fail")Z++;else H++;return{checks:Q,disk:$,summary:{passed:X,failed:Z,warnings:H,ok:Z===0}}}function J(z){switch(z){case"pass":return`${x}PASS${K}`;case"fail":return`${S}FAIL${K}`;case"warn":return`${_}WARN${K}`}}function M1(z){let Q=z.version?` (v${z.version})`:"",$=z.displayName;if(!z.found){let X=z.required==="required"?"not found":z.required==="recommended"?"not found (recommended)":"not found (optional)";return` ${J(z.status)} ${$} - ${X}`}if(z.min_version&&z.version&&r1(z.version,z.min_version)<0){let X=z.required==="required"?"requires":"recommended";return` ${J(z.status)} ${$}${Q} - ${X} >= ${z.min_version}`}return` ${J(z.status)} ${$}${Q}`}function Y1(z,Q){if(Q==="pass")z.pass++;else if(Q==="fail")z.fail++;else z.warn++}function e0(){process.stdout.write(`${F}loki doctor${K} - Check system prerequisites
201
201
 
202
202
  `),process.stdout.write(`Usage: loki doctor [--json]
203
203
 
@@ -206,80 +206,80 @@ Start a session with: loki start <prd>`}}let Z=m0(X);return{exitCode:0,stdout:Q?
206
206
 
207
207
  `),process.stdout.write(`Checks: node, python3, jq, git, curl, bash version,
208
208
  `),process.stdout.write(` claude/codex/gemini CLIs, and disk space.
209
- `)}async function e0(){process.stdout.write(`${F}Loki Mode Doctor${K}
209
+ `)}async function z3(){process.stdout.write(`${F}Loki Mode Doctor${K}
210
210
 
211
211
  `),process.stdout.write(`Checking system prerequisites...
212
212
 
213
- `);let z={pass:0,fail:0,warn:0},Q=await i1(),$=new Map(Q.map((U)=>[U.command,U]));process.stdout.write(`${J}Required:${K}
214
- `);for(let U of["node","python3","jq","git","curl"]){let _=$.get(U);process.stdout.write(J1(_)+`
215
- `),G1(z,_.status)}process.stdout.write(`
216
- `),process.stdout.write(`${J}AI Providers:${K}
217
- `);let X=["claude","codex","gemini","cline","aider"],Z=!1;for(let U of X){let _=$.get(U);if(process.stdout.write(J1(_)+`
218
- `),G1(z,_.status),_.found)Z=!0}if(!Z)process.stdout.write(` ${G("fail")} No AI provider CLI installed -- at least one is required
219
- `),process.stdout.write(` ${T}Install: npm install -g @anthropic-ai/claude-code${K}
213
+ `);let z={pass:0,fail:0,warn:0},Q=await s1(),$=new Map(Q.map((U)=>[U.command,U]));process.stdout.write(`${G}Required:${K}
214
+ `);for(let U of["node","python3","jq","git","curl"]){let Y=$.get(U);process.stdout.write(M1(Y)+`
215
+ `),Y1(z,Y.status)}process.stdout.write(`
216
+ `),process.stdout.write(`${G}AI Providers:${K}
217
+ `);let X=["claude","codex","gemini","cline","aider"],Z=!1;for(let U of X){let Y=$.get(U);if(process.stdout.write(M1(Y)+`
218
+ `),Y1(z,Y.status),Y.found)Z=!0}if(!Z)process.stdout.write(` ${J("fail")} No AI provider CLI installed -- at least one is required
219
+ `),process.stdout.write(` ${_}Install: npm install -g @anthropic-ai/claude-code${K}
220
220
  `),z.fail++;process.stdout.write(`
221
- `),process.stdout.write(`${J}API Keys:${K}
222
- `);let H=$.get("claude").found,V=$.get("codex").found,W=$.get("gemini").found,M=process.env;if(M.ANTHROPIC_API_KEY)process.stdout.write(` ${G("pass")} ANTHROPIC_API_KEY is set
223
- `),z.pass++;else if(H)process.stdout.write(` ${Y} -- ${K} ANTHROPIC_API_KEY not set (Claude CLI uses its own login)
224
- `);if(M.OPENAI_API_KEY)process.stdout.write(` ${G("pass")} OPENAI_API_KEY is set
225
- `),z.pass++;else if(V)process.stdout.write(` ${Y} -- ${K} OPENAI_API_KEY not set (Codex CLI uses its own login)
226
- `);if(M.GOOGLE_API_KEY||M.GEMINI_API_KEY)process.stdout.write(` ${G("pass")} GOOGLE_API_KEY is set
227
- `),z.pass++;else if(W)process.stdout.write(` ${Y} -- ${K} GOOGLE_API_KEY not set (Gemini CLI uses its own login)
221
+ `),process.stdout.write(`${G}API Keys:${K}
222
+ `);let H=$.get("claude").found,W=$.get("codex").found,B=$.get("gemini").found,j=process.env;if(j.ANTHROPIC_API_KEY)process.stdout.write(` ${J("pass")} ANTHROPIC_API_KEY is set
223
+ `),z.pass++;else if(H)process.stdout.write(` ${A} -- ${K} ANTHROPIC_API_KEY not set (Claude CLI uses its own login)
224
+ `);if(j.OPENAI_API_KEY)process.stdout.write(` ${J("pass")} OPENAI_API_KEY is set
225
+ `),z.pass++;else if(W)process.stdout.write(` ${A} -- ${K} OPENAI_API_KEY not set (Codex CLI uses its own login)
226
+ `);if(j.GOOGLE_API_KEY||j.GEMINI_API_KEY)process.stdout.write(` ${J("pass")} GOOGLE_API_KEY is set
227
+ `),z.pass++;else if(B)process.stdout.write(` ${A} -- ${K} GOOGLE_API_KEY not set (Gemini CLI uses its own login)
228
228
  `);process.stdout.write(`
229
- `),process.stdout.write(`${J}Skills:${K}
230
- `);for(let U of t1())if(U.status==="pass")process.stdout.write(` ${G("pass")} ${U.name} ${Y}${U.path}${K}
231
- `),z.pass++;else if(U.status==="fail")process.stdout.write(` ${G("fail")} ${U.name} ${Y}${U.detail}${K}
232
- `),process.stdout.write(` ${T}Fix: loki setup-skill${K}
233
- `),z.fail++;else process.stdout.write(` ${G("warn")} ${U.name} ${Y}${U.detail}${K}
229
+ `),process.stdout.write(`${G}Skills:${K}
230
+ `);for(let U of a1())if(U.status==="pass")process.stdout.write(` ${J("pass")} ${U.name} ${A}${U.path}${K}
231
+ `),z.pass++;else if(U.status==="fail")process.stdout.write(` ${J("fail")} ${U.name} ${A}${U.detail}${K}
232
+ `),process.stdout.write(` ${_}Fix: loki setup-skill${K}
233
+ `),z.fail++;else process.stdout.write(` ${J("warn")} ${U.name} ${A}${U.detail}${K}
234
234
  `),z.warn++;if(process.stdout.write(`
235
- `),process.stdout.write(`${J}Integrations:${K}
236
- `),await q1("mcp"))process.stdout.write(` ${G("pass")} MCP SDK (Python)
237
- `),z.pass++;else process.stdout.write(` ${G("warn")} MCP SDK - not installed (pip3 install mcp)
238
- `),z.warn++;if(await q1("numpy",!0))process.stdout.write(` ${G("pass")} numpy (vector search)
239
- `),z.pass++;else process.stdout.write(` ${G("warn")} numpy - not installed (pip3 install numpy)
240
- `),z.warn++;if(await q1("sentence_transformers",!0))process.stdout.write(` ${G("pass")} sentence-transformers (embeddings)
241
- `),z.pass++;else process.stdout.write(` ${G("warn")} sentence-transformers - not installed (loki memory vectors setup)
242
- `),z.warn++;if(await M1("http://localhost:8100/api/v2/heartbeat"))process.stdout.write(` ${G("pass")} ChromaDB server (port 8100)
243
- `),z.pass++;else process.stdout.write(` ${G("warn")} ChromaDB - not running (docker start loki-chroma)
244
- `),z.warn++;let A=process.env.LOKI_MIROFISH_URL;if(A)if(await M1(`${A}/health`))process.stdout.write(` ${G("pass")} MiroFish server (${A})
245
- `),z.pass++;else process.stdout.write(` ${G("warn")} MiroFish - not running (loki start --mirofish-docker <image>)
246
- `),z.warn++;if(process.env.LOKI_OTEL_ENDPOINT)process.stdout.write(` ${G("pass")} OTEL endpoint: ${process.env.LOKI_OTEL_ENDPOINT}
247
- `),z.pass++;else process.stdout.write(` ${G("warn")} OTEL - not configured (set LOKI_OTEL_ENDPOINT)
235
+ `),process.stdout.write(`${G}Integrations:${K}
236
+ `),await j1("mcp"))process.stdout.write(` ${J("pass")} MCP SDK (Python)
237
+ `),z.pass++;else process.stdout.write(` ${J("warn")} MCP SDK - not installed (pip3 install mcp)
238
+ `),z.warn++;if(await j1("numpy",!0))process.stdout.write(` ${J("pass")} numpy (vector search)
239
+ `),z.pass++;else process.stdout.write(` ${J("warn")} numpy - not installed (pip3 install numpy)
240
+ `),z.warn++;if(await j1("sentence_transformers",!0))process.stdout.write(` ${J("pass")} sentence-transformers (embeddings)
241
+ `),z.pass++;else process.stdout.write(` ${J("warn")} sentence-transformers - not installed (loki memory vectors setup)
242
+ `),z.warn++;if(await A1("http://localhost:8100/api/v2/heartbeat"))process.stdout.write(` ${J("pass")} ChromaDB server (port 8100)
243
+ `),z.pass++;else process.stdout.write(` ${J("warn")} ChromaDB - not running (docker start loki-chroma)
244
+ `),z.warn++;let O=process.env.LOKI_MIROFISH_URL;if(O)if(await A1(`${O}/health`))process.stdout.write(` ${J("pass")} MiroFish server (${O})
245
+ `),z.pass++;else process.stdout.write(` ${J("warn")} MiroFish - not running (loki start --mirofish-docker <image>)
246
+ `),z.warn++;if(process.env.LOKI_OTEL_ENDPOINT)process.stdout.write(` ${J("pass")} OTEL endpoint: ${process.env.LOKI_OTEL_ENDPOINT}
247
+ `),z.pass++;else process.stdout.write(` ${J("warn")} OTEL - not configured (set LOKI_OTEL_ENDPOINT)
248
248
  `),z.warn++;process.stdout.write(`
249
- `),process.stdout.write(`${J}System:${K}
250
- `);let I=$.get("bash");process.stdout.write(J1(I)+`
251
- `),G1(z,I.status);let P=j1(),B=P.available_gb===null?null:Math.floor(P.available_gb);if(B===null)process.stdout.write(` ${G("warn")} Disk space: unable to determine
252
- `),z.warn++;else if(P.status==="fail")process.stdout.write(` ${G("fail")} Disk space: ${B}GB available (need >= 1GB)
253
- `),z.fail++;else if(P.status==="warn")process.stdout.write(` ${G("warn")} Disk space: ${B}GB available (low)
254
- `),z.warn++;else process.stdout.write(` ${G("pass")} Disk space: ${B}GB available
249
+ `),process.stdout.write(`${G}System:${K}
250
+ `);let T=$.get("bash");process.stdout.write(M1(T)+`
251
+ `),Y1(z,T.status);let P=O1(),V=P.available_gb===null?null:Math.floor(P.available_gb);if(V===null)process.stdout.write(` ${J("warn")} Disk space: unable to determine
252
+ `),z.warn++;else if(P.status==="fail")process.stdout.write(` ${J("fail")} Disk space: ${V}GB available (need >= 1GB)
253
+ `),z.fail++;else if(P.status==="warn")process.stdout.write(` ${J("warn")} Disk space: ${V}GB available (low)
254
+ `),z.warn++;else process.stdout.write(` ${J("pass")} Disk space: ${V}GB available
255
255
  `),z.pass++;if(process.stdout.write(`
256
- `),process.stdout.write(`${F}Summary:${K} ${x}${z.pass} passed${K}, ${k}${z.fail} failed${K}, ${T}${z.warn} warnings${K}
256
+ `),process.stdout.write(`${F}Summary:${K} ${x}${z.pass} passed${K}, ${S}${z.fail} failed${K}, ${_}${z.warn} warnings${K}
257
257
 
258
- `),z.fail>0)return process.stdout.write(`${k}Some required prerequisites are missing.${K}
258
+ `),z.fail>0)return process.stdout.write(`${S}Some required prerequisites are missing.${K}
259
259
  `),process.stdout.write(`Install missing dependencies and run 'loki doctor' again.
260
- `),1;if(z.warn>0)return process.stdout.write(`${T}All required checks passed with some warnings.${K}
260
+ `),1;if(z.warn>0)return process.stdout.write(`${_}All required checks passed with some warnings.${K}
261
261
  `),0;return process.stdout.write(`${x}All checks passed. System is ready for Loki Mode.${K}
262
- `),0}async function z6(z){let Q=!1;for(let $ of z)if($==="--json")Q=!0;else if($==="--help"||$==="-h")return s0(),0;else return process.stderr.write(`${k}Unknown option: ${$}${K}
262
+ `),0}async function $3(z){let Q=!1;for(let $ of z)if($==="--json")Q=!0;else if($==="--help"||$==="-h")return e0(),0;else return process.stderr.write(`${S}Unknown option: ${$}${K}
263
263
  `),process.stderr.write(`Usage: loki doctor [--json]
264
- `),1;if(Q){let $=await a1();return process.stdout.write(JSON.stringify($,null,2)+`
265
- `),0}return e0()}var n0,i0,a0;var e1=f(()=>{g();$1();h();n0=/(\d+\.\d+(?:\.\d+)*)/;i0=[{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"}];a0=[{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:"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{readFileSync as K0}from"fs";import{resolve as O1,dirname as H0}from"path";import{fileURLToPath as V0}from"url";var W0=H0(V0(import.meta.url)),U0=O1(W0,"..",".."),e=null;function w1(){if(e===null)try{e=K0(O1(U0,"VERSION"),"utf-8").trim()}catch{e="unknown"}return e}function F1(){return process.stdout.write(`Loki Mode v${w1()}
266
- `),0}g();h();m();import{readFileSync as _0,existsSync as Y0}from"fs";import{resolve as A0}from"path";var T0=["claude","codex","gemini","cline","aider"];function S1(){let z=A0(E(),"state","provider");if(!Y0(z))return"";try{return _0(z,"utf-8").trim()}catch{return""}}function I0(z,Q){return z||Q||process.env.LOKI_PROVIDER||"claude"}function O0(z){let Q=S1(),$=I0(z,Q);switch(process.stdout.write(`${F}Current Provider${K}
264
+ `),1;if(Q){let $=await i1();return process.stdout.write(JSON.stringify($,null,2)+`
265
+ `),0}return z3()}var r0,s0,i0;var z0=g(()=>{m();Q1();p();r0=/(\d+\.\d+(?:\.\d+)*)/;s0=[{name:"Claude Code",dir:".claude/skills/loki-mode"},{name:"Codex CLI",dir:".codex/skills/loki-mode"},{name:"Gemini CLI",dir:".gemini/skills/loki-mode"},{name:"Cline CLI",dir:".cline/skills/loki-mode"},{name:"Aider CLI",dir:".aider/skills/loki-mode"}];i0=[{displayName:"Node.js (>= 18)",jsonName:"Node.js",cmd:"node",required:"required",min:"18.0"},{displayName:"Python 3 (>= 3.8)",jsonName:"Python 3",cmd:"python3",required:"required",min:"3.8"},{displayName:"jq",jsonName:"jq",cmd:"jq",required:"required"},{displayName:"git",jsonName:"git",cmd:"git",required:"required"},{displayName:"curl",jsonName:"curl",cmd:"curl",required:"required"},{displayName:"bash (>= 4.0)",jsonName:"bash",cmd:"bash",required:"recommended",min:"4.0"},{displayName:"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"}]});h();import{readFileSync as U0}from"fs";import{resolve as V0,dirname as q0}from"path";import{fileURLToPath as G0}from"url";var v=null;function k1(){if(v!==null)return v;let z="7.4.8";if(typeof z==="string"&&z.length>0)return v=z,v;try{let Q=q0(G0(import.meta.url)),$=W1(Q);v=U0(V0($,"VERSION"),"utf-8").trim()}catch{v="unknown"}return v}function R1(){return process.stdout.write(`Loki Mode v${k1()}
266
+ `),0}m();p();h();import{readFileSync as A0,existsSync as O0}from"fs";import{resolve as _0}from"path";var T0=["claude","codex","gemini","cline","aider"];function D1(){let z=_0(E(),"state","provider");if(!O0(z))return"";try{return A0(z,"utf-8").trim()}catch{return""}}function I0(z,Q){return z||Q||process.env.LOKI_PROVIDER||"claude"}function w0(z){let Q=D1(),$=I0(z,Q);switch(process.stdout.write(`${F}Current Provider${K}
267
267
  `),process.stdout.write(`
268
- `),process.stdout.write(`${J}Provider:${K} ${$}
268
+ `),process.stdout.write(`${G}Provider:${K} ${$}
269
269
  `),$){case"claude":process.stdout.write(`${x}Status:${K} Full features (subagents, parallel, MCP)
270
270
  `);break;case"cline":process.stdout.write(`${x}Status:${K} Near-full mode (subagents, MCP, 12+ providers)
271
- `);break;case"codex":case"gemini":case"aider":process.stdout.write(`${T}Status:${K} Degraded mode (sequential only)
272
- `);break;default:break}if(Q)process.stdout.write(`${Y}(saved in .loki/state/provider)${K}
273
- `);else process.stdout.write(`${Y}(default - not explicitly set)${K}
271
+ `);break;case"codex":case"gemini":case"aider":process.stdout.write(`${_}Status:${K} Degraded mode (sequential only)
272
+ `);break;default:break}if(Q)process.stdout.write(`${A}(saved in .loki/state/provider)${K}
273
+ `);else process.stdout.write(`${A}(default - not explicitly set)${K}
274
274
  `);return process.stdout.write(`
275
- `),process.stdout.write(`Switch provider: ${J}loki provider set <name>${K}
276
- `),process.stdout.write(`Available: ${J}loki provider list${K}
277
- `),0}async function w0(){let Q=S1()||process.env.LOKI_PROVIDER||"claude";process.stdout.write(`${F}Available Providers${K}
275
+ `),process.stdout.write(`Switch provider: ${G}loki provider set <name>${K}
276
+ `),process.stdout.write(`Available: ${G}loki provider list${K}
277
+ `),0}async function F0(){let Q=D1()||process.env.LOKI_PROVIDER||"claude";process.stdout.write(`${F}Available Providers${K}
278
278
  `),process.stdout.write(`
279
- `);let $=await Promise.all(T0.map(async(H)=>[H,await R(H)!==null])),X=new Map;for(let[H,V]of $)X.set(H,V?`${x}installed${K}`:`${k}not installed${K}`);let Z=[["claude","claude - Claude Code (Anthropic) "],["codex","codex - Codex CLI (OpenAI) "],["gemini","gemini - Gemini CLI (Google) "],["cline","cline - Cline (multi-provider) "],["aider","aider - Aider (terminal pair prog) "]];for(let[H,V]of Z){let W=Q===H?` ${J}(current)${K}`:"";process.stdout.write(` ${V} ${X.get(H)}${W}
279
+ `);let $=await Promise.all(T0.map(async(H)=>[H,await R(H)!==null])),X=new Map;for(let[H,W]of $)X.set(H,W?`${x}installed${K}`:`${S}not installed${K}`);let Z=[["claude","claude - Claude Code (Anthropic) "],["codex","codex - Codex CLI (OpenAI) "],["gemini","gemini - Gemini CLI (Google) "],["cline","cline - Cline (multi-provider) "],["aider","aider - Aider (terminal pair prog) "]];for(let[H,W]of Z){let B=Q===H?` ${G}(current)${K}`:"";process.stdout.write(` ${W} ${X.get(H)}${B}
280
280
  `)}return process.stdout.write(`
281
- `),process.stdout.write(`Set provider: ${J}loki provider set <name>${K}
282
- `),0}function F0(){return process.stdout.write(`${F}Loki Mode Provider Management${K}
281
+ `),process.stdout.write(`Set provider: ${G}loki provider set <name>${K}
282
+ `),0}function L0(){return process.stdout.write(`${F}Loki Mode Provider Management${K}
283
283
  `),process.stdout.write(`
284
284
  `),process.stdout.write(`Usage: loki provider <command>
285
285
  `),process.stdout.write(`
@@ -297,17 +297,17 @@ Start a session with: loki start <prd>`}}let Z=m0(X);return{exitCode:0,stdout:Q?
297
297
  `),process.stdout.write(` loki provider list
298
298
  `),process.stdout.write(` loki provider info gemini
299
299
  `),process.stdout.write(` loki provider models
300
- `),0}async function E1(z){let Q=z[0]??"show",$=z.slice(1);switch(Q){case"show":case"current":return O0($[0]);case"list":return w0();case"set":case"info":case"models":return L0(["provider",Q,...$]);default:return F0()}}async function L0(z){let{run:Q}=await Promise.resolve().then(() => (g(),L1)),{resolve:$}=await import("path"),{REPO_ROOT:X}=await Promise.resolve().then(() => (m(),k1)),Z=$(X,"autonomy","loki"),H=await Q([Z,...z],{env:{LOKI_LEGACY_BASH:"1"}});return process.stdout.write(H.stdout),process.stderr.write(H.stderr),H.exitCode}h();m();$1();g();import{existsSync as D1,readFileSync as P0}from"fs";import{resolve as v}from"path";import{mkdir as R0}from"fs/promises";var t=v(H1(),"learnings");function V1(z){if(!D1(z))return 0;try{let Q=P0(z,"utf-8"),$=0;for(let X of Q.split(`
301
- `))if(X.includes('"description"'))$++;return $}catch{return 0}}async function k0(){await R0(t,{recursive:!0});let z=V1(v(t,"patterns.jsonl")),Q=V1(v(t,"mistakes.jsonl")),$=V1(v(t,"successes.jsonl"));return process.stdout.write(`${F}Cross-Project Learnings${K}
300
+ `),0}async function E1(z){let Q=z[0]??"show",$=z.slice(1);switch(Q){case"show":case"current":return w0($[0]);case"list":return F0();case"set":case"info":case"models":return x0(["provider",Q,...$]);default:return L0()}}async function x0(z){let{run:Q}=await Promise.resolve().then(() => (m(),S1)),{resolve:$}=await import("path"),{REPO_ROOT:X}=await Promise.resolve().then(() => (h(),P1)),Z=$(X,"autonomy","loki"),H=await Q([Z,...z],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(H.stdout),process.stderr.write(H.stderr),H.exitCode}p();h();Q1();m();import{existsSync as b1,readFileSync as k0}from"fs";import{resolve as c}from"path";import{mkdir as R0}from"fs/promises";var i=c(B1(),"learnings");function V1(z){if(!b1(z))return 0;try{let Q=k0(z,"utf-8"),$=0;for(let X of Q.split(`
301
+ `))if(X.includes('"description"'))$++;return $}catch{return 0}}async function S0(){await R0(i,{recursive:!0});let z=V1(c(i,"patterns.jsonl")),Q=V1(c(i,"mistakes.jsonl")),$=V1(c(i,"successes.jsonl"));return process.stdout.write(`${F}Cross-Project Learnings${K}
302
302
  `),process.stdout.write(`
303
303
  `),process.stdout.write(` Patterns: ${x}${z}${K}
304
- `),process.stdout.write(` Mistakes: ${T}${Q}${K}
305
- `),process.stdout.write(` Successes: ${J}${$}${K}
304
+ `),process.stdout.write(` Mistakes: ${_}${Q}${K}
305
+ `),process.stdout.write(` Successes: ${G}${$}${K}
306
306
  `),process.stdout.write(`
307
- `),process.stdout.write(`Location: ${t}
307
+ `),process.stdout.write(`Location: ${i}
308
308
  `),process.stdout.write(`
309
309
  `),process.stdout.write(`Use 'loki memory show <type>' to view entries
310
- `),0}async function S0(z){if(z){let X=`
310
+ `),0}async function D0(z){if(z){let X=`
311
311
  try:
312
312
  from memory.layers import IndexLayer
313
313
  layer = IndexLayer('.loki/memory')
@@ -317,9 +317,9 @@ except ImportError:
317
317
  print('Error: memory.layers module not found')
318
318
  except Exception as e:
319
319
  print(f'Error: {e}')
320
- `.trim(),Z=await r(X,{cwd:y});return process.stdout.write(Z.stdout),0}let Q=v(E(),"memory","index.json");if(!D1(Q))return process.stdout.write(`No index found
321
- `),0;let $=await r(`import json, sys; sys.stdout.write(json.dumps(json.load(open(${JSON.stringify(Q)})), indent=4) + "\\n")`);if($.exitCode!==0)return process.stdout.write(`No index found
322
- `),0;return process.stdout.write($.stdout),0}async function b1(z){switch(z[0]??"list"){case"list":case"ls":return k0();case"index":return S0(z[1]==="rebuild");default:{let $=v(y,"autonomy","loki"),X=await O([$,"memory",...z],{env:{LOKI_LEGACY_BASH:"1"}});return process.stdout.write(X.stdout),process.stderr.write(X.stderr),X.exitCode}}}var z0=`Loki Mode (TypeScript port, Phase 2 of bash->Bun migration)
320
+ `.trim(),Z=await s(X,{cwd:y});return process.stdout.write(Z.stdout),0}let Q=c(E(),"memory","index.json");if(!b1(Q))return process.stdout.write(`No index found
321
+ `),0;let $=await s(`import json, sys; sys.stdout.write(json.dumps(json.load(open(${JSON.stringify(Q)})), indent=4) + "\\n")`);if($.exitCode!==0)return process.stdout.write(`No index found
322
+ `),0;return process.stdout.write($.stdout),0}async function N1(z){switch(z[0]??"list"){case"list":case"ls":return S0();case"index":return D0(z[1]==="rebuild");default:{let $=c(y,"autonomy","loki"),X=await I([$,"memory",...z],{env:{LOKI_LEGACY_BASH:"1"},timeoutMs:3600000});return process.stdout.write(X.stdout),process.stderr.write(X.stderr),X.exitCode}}}var $0=`Loki Mode (TypeScript port, Phase 2 of bash->Bun migration)
323
323
 
324
324
  Usage: loki <command> [args...]
325
325
 
@@ -335,7 +335,7 @@ Phase 2 ported (Bun-native, fast):
335
335
 
336
336
  All other commands fall through to the bash CLI (autonomy/loki).
337
337
  Set LOKI_LEGACY_BASH=1 to force the bash CLI for every command.
338
- `;async function $6(z){let Q=z[0],$=z.slice(1);switch(Q){case void 0:case"help":case"--help":case"-h":return process.stdout.write(z0),0;case"version":case"--version":case"-v":return F1();case"provider":return E1($);case"memory":return b1($);case"status":{let{runStatus:X}=await Promise.resolve().then(() => (m1(),h1));return X($)}case"stats":{let{runStats:X}=await Promise.resolve().then(() => (d1(),c1));return X($)}case"doctor":{let{runDoctor:X}=await Promise.resolve().then(() => (e1(),s1));return X($)}default:return process.stderr.write(`Unknown command: ${Q}
339
- `),process.stderr.write(z0),2}}process.on("SIGINT",()=>process.exit(130));process.on("SIGTERM",()=>process.exit(143));var Q6=await $6(Bun.argv.slice(2));process.exit(Q6);
338
+ `;async function Q3(z){let Q=z[0],$=z.slice(1);switch(Q){case void 0:case"help":case"--help":case"-h":return process.stdout.write($0),0;case"version":case"--version":case"-v":return R1();case"provider":return E1($);case"memory":return N1($);case"status":{let{runStatus:X}=await Promise.resolve().then(() => (m1(),v1));return X($)}case"stats":{let{runStats:X}=await Promise.resolve().then(() => (d1(),l1));return X($)}case"doctor":{let{runDoctor:X}=await Promise.resolve().then(() => (z0(),e1));return X($)}default:return process.stderr.write(`Unknown command: ${Q}
339
+ `),process.stderr.write($0),2}}process.on("SIGINT",()=>process.exit(130));process.on("SIGTERM",()=>process.exit(143));var X3=await Q3(Bun.argv.slice(2));process.exit(X3);
340
340
 
341
- //# debugId=1CBFE5CD7E7E8D9B64756E2164756E21
341
+ //# debugId=7F3CF7B4F1CA56A864756E2164756E21
package/mcp/__init__.py CHANGED
@@ -57,4 +57,4 @@ try:
57
57
  except ImportError:
58
58
  __all__ = ['mcp']
59
59
 
60
- __version__ = '7.3.0'
60
+ __version__ = '7.4.8'
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "loki-mode",
3
- "version": "7.3.0",
3
+ "version": "7.4.8",
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",
@@ -125,6 +125,9 @@
125
125
  "@opentelemetry/sdk-trace-base": "^1.30.0",
126
126
  "@opentelemetry/exporter-trace-otlp-http": "^0.57.0"
127
127
  },
128
+ "overrides": {
129
+ "protobufjs": ">=7.5.5"
130
+ },
128
131
  "devDependencies": {
129
132
  "@types/node": "^25.2.0",
130
133
  "jest": "^29.7.0",