trantor 0.17.46 → 0.17.47

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.
@@ -6,14 +6,14 @@
6
6
  },
7
7
  "metadata": {
8
8
  "description": "Trantor — the hub-world for AI agent crews: live message bus, presence, project Kanban/flow board + context-handoff for independent AI coding agents (Claude, Codex, Gemini, …)",
9
- "version": "0.17.46"
9
+ "version": "0.17.47"
10
10
  },
11
11
  "plugins": [
12
12
  {
13
13
  "name": "trantor",
14
14
  "source": "./",
15
15
  "description": "The hub-world for AI agent crews. Say \"fire up the crew\" and Claude becomes the architect: a plan-aware Advisor routes the work (solo / cheap inline calls / live crew of Codex, GLM, Kimi & DeepSeek in their own terminal windows), a Kanban/flow command center with a testing gate tracks it, and an economics brain (Scrooge) keeps the receipts. Includes the relay MCP, a SessionStart auto-discovery hook, and a PreCompact context-handoff so a fresh session can take over a full window instead of compacting.",
16
- "version": "0.17.46",
16
+ "version": "0.17.47",
17
17
  "author": {
18
18
  "name": "Sasha Bogojevic"
19
19
  },
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "trantor",
3
- "version": "0.17.46",
3
+ "version": "0.17.47",
4
4
  "description": "Trantor — the hub-world for AI agent crews: live message bus, presence, project Kanban/flow board + crew orchestration for independent AI coding agents (Claude, Codex, Gemini, Kimi, DeepSeek)",
5
5
  "mcpServers": {
6
6
  "relay": {
package/bin/advise.mjs CHANGED
@@ -19,52 +19,73 @@ import { pathToFileURL } from "node:url";
19
19
  const H = homedir();
20
20
  const read = (p, fb) => { try { return JSON.parse(readFileSync(p, "utf8")); } catch { return fb; } };
21
21
 
22
- // ---- canonical crew roster (single source of truth, mirrors bin/doctor.mjs) ----
23
- // Each token maps to: the CLI binary that must exist · the `trantor up` LAUNCH spec the
24
- // orchestrator runs · the bus SESSION name (= spec's agent part) · the profile PROVIDER
25
- // key that gates the seat and supplies the tier.
22
+ // ---- crew roster: BUILT-IN seats + ANY opencode provider the user has brought (BYOM) ----
23
+ // Each seat: the CLI binary that must exist (`cli`) · the `trantor up` LAUNCH spec · the bus
24
+ // SESSION label (its identity on the board) · the profile PROVIDER key (tier/cost) · for
25
+ // opencode-driven seats, the opencode PROVIDER id (`providerOc`) used to enumerate models + auth.
26
26
  //
27
- // GEMINI is deliberately ABSENT: Google retired the free Gemini CLI seat (2026-06-18), so
28
- // even though the `gemini` binary is usually still on PATH, `gemini --yolo` crashes (exit 1)
29
- // for everyone without a paid enterprise key. Its replacement seat is GLM via opencode.
30
- // (Gemini still works as a Scrooge cheap-model via GEMINI_API_KEY a separate path.) A
31
- // paid-key holder can still force it with an explicit `trantor up gemini`; we just never
32
- // auto-recommend a dead seat.
33
- export const ROSTER = {
34
- codex: { cli: "codex", launch: "codex", session: "codex", provider: "codex" },
35
- kimi: { cli: "kimi", launch: "kimi", session: "kimi", provider: "kimi" },
36
- deepseek: { cli: "opencode", launch: "deepseek:deepseek", session: "deepseek", provider: "deepseek" },
37
- glm: { cli: "opencode", launch: "opencode:zai-coding-plan", session: "opencode", provider: "zai" },
38
- // OpenRouter = the BYOM on-ramp: one key fronts hundreds of models (incl. vendors with no
39
- // CLI of their own). Rides the opencode runner like glm/deepseek, but under its OWN bus label
40
- // `openrouter` (distinct session, never collides with the glm `opencode` seat). The launcher
41
- // live-selects the best OpenRouter model for the work's difficulty (or pin one:
42
- // `openrouter:openrouter/<vendor>/<model>`). The model picked is what determines its strength,
43
- // so it sits LAST in every CREW_PREF tier — a wildcard that fills once the proven native seats
44
- // are taken, and the ONLY seat for a user who brought nothing but an OpenRouter key.
45
- openrouter: { cli: "opencode", launch: "openrouter:openrouter", session: "openrouter", provider: "openrouter" },
27
+ // GEMINI is deliberately absent: Google retired the free CLI seat (2026-06-18) → `gemini --yolo`
28
+ // crashes exit 1. Its replacement is GLM via opencode.
29
+ //
30
+ // The opencode-driven seats are the BYOM substrate: opencode is a UNIVERSAL adapter, so any
31
+ // provider the user configures in opencode (or declares in their profile) becomes a crew seat
32
+ // with ZERO code change here — `buildRoster()` discovers them at runtime. The built-ins below are
33
+ // just the curated defaults + the two opencode seats with non-obvious mappings (glm: profile key
34
+ // `zai` opencode provider `zai-coding-plan`).
35
+ export const BUILTIN_ROSTER = {
36
+ codex: { cli: "codex", launch: "codex", session: "codex", provider: "codex" },
37
+ kimi: { cli: "kimi", launch: "kimi", session: "kimi", provider: "kimi" },
38
+ deepseek: { cli: "opencode", launch: "deepseek:deepseek", session: "deepseek", provider: "deepseek", providerOc: "deepseek" },
39
+ glm: { cli: "opencode", launch: "opencode:zai-coding-plan", session: "opencode", provider: "zai", providerOc: "zai-coding-plan" },
40
+ openrouter: { cli: "opencode", launch: "openrouter:openrouter", session: "openrouter", provider: "openrouter", providerOc: "openrouter" },
46
41
  };
42
+ // opencode provider ids already claimed by a built-in (so discovery never duplicates them) + the
43
+ // names that are native CLIs / built-in profile aliases (never opencode-driven seats).
44
+ const BUILTIN_OC = new Set(Object.values(BUILTIN_ROSTER).filter(s => s.providerOc).map(s => s.providerOc));
45
+ const NEVER_DISCOVER = new Set(["claude", "codex", "kimi", "gemini", "zai", "opencode"]);
46
+
47
+ // Discover opencode providers the user has configured — from opencode.json `provider` keys AND
48
+ // from profile providers declared via `trantor provider add` — that aren't already built-in. Each
49
+ // becomes an opencode-driven seat under its OWN bus label (distinct session, no collisions). THIS
50
+ // is what lets a brought provider (Inception, a Japanese model, any opencode vendor) light up a
51
+ // seat with no code edit. T2's capability ingestion then makes it route well by difficulty.
52
+ export function discoverSeats(profile, ocConfig) {
53
+ const out = {};
54
+ const provKeys = new Set([...Object.keys(ocConfig?.provider || {}), ...Object.keys(profile?.providers || {})]);
55
+ for (const p of provKeys) {
56
+ if (BUILTIN_OC.has(p) || NEVER_DISCOVER.has(p)) continue;
57
+ const label = String(p).toLowerCase().replace(/[^a-z0-9-]/g, "-");
58
+ out[label] = { cli: "opencode", launch: `${label}:${p}`, session: label, provider: p, providerOc: p, discovered: true };
59
+ }
60
+ return out;
61
+ }
62
+
63
+ export function buildRoster(profile, ocConfig) {
64
+ return { ...BUILTIN_ROSTER, ...discoverSeats(profile, ocConfig) };
65
+ }
47
66
 
48
67
  export function loadWorld() {
49
68
  const profile = read(join(H, ".agent-bus", "profile.json"), { providers: {} });
50
69
  const registry = read(join(H, ".token-scrooge", "registry.json"), { models: {}, tasks: {} });
51
70
  const caps = read(join(H, ".token-scrooge", "capabilities.json"), {});
71
+ const ocConfig = read(join(H, ".config", "opencode", "opencode.json"), {});
72
+ const roster = buildRoster(profile, ocConfig);
52
73
  const has = (c) => { try { execSync(`command -v ${c}`, { stdio: "ignore", shell: "/bin/sh" }); return true; } catch { return false; } };
53
- const opencodeKey = (prov) => !!read(join(H, ".config", "opencode", "opencode.json"), {})?.provider?.[prov]?.options?.apiKey;
74
+ const opencodeKey = (prov) => !!ocConfig?.provider?.[prov]?.options?.apiKey;
54
75
  // a key the user already has for Scrooge counts too — the opencode runner sources these .env
55
- // files, so OPENROUTER_API_KEY in ~/.token-scrooge/.env lights up the crew seat with no extra setup.
76
+ // files, so e.g. OPENROUTER_API_KEY in ~/.token-scrooge/.env lights up the seat with no extra setup.
56
77
  const envHasKey = (k) => !!process.env[k] || [join(H, ".token-scrooge", ".env"), join(H, ".agent-bus", ".env")]
57
78
  .some(f => { try { return readFileSync(f, "utf8").includes(k); } catch { return false; } });
58
- // a seat is available only if its CLI exists AND its provider is actually set up — a present
59
- // binary with a dead/missing seat (gemini, or opencode with no provider key) must NOT be recommended.
79
+ // a seat is available only if its CLI exists AND (for opencode-driven seats) the provider is
80
+ // actually set up — a present binary with a dead/missing seat must NOT be recommended.
60
81
  const hasSeat = (tok) => {
61
- const r = ROSTER[tok]; if (!r || !has(r.cli)) return false;
62
- if (tok === "glm") return !!profile?.providers?.zai || opencodeKey("zai-coding-plan");
63
- if (tok === "openrouter") return !!profile?.providers?.openrouter || opencodeKey("openrouter") || envHasKey("OPENROUTER_API_KEY");
64
- return true;
82
+ const s = roster[tok]; if (!s || !has(s.cli)) return false;
83
+ if (s.cli !== "opencode") return true; // native CLI present = ready
84
+ const envKey = `${String(s.providerOc).toUpperCase().replace(/[^A-Z0-9]/g, "_")}_API_KEY`;
85
+ return opencodeKey(s.providerOc) || envHasKey(envKey) || !!profile?.providers?.[s.provider];
65
86
  };
66
- const agents = Object.keys(ROSTER).filter(hasSeat);
67
- return { profile, registry, caps, agents, scrooge: has("scrooge") };
87
+ const agents = Object.keys(roster).filter(hasSeat);
88
+ return { profile, registry, caps, roster, agents, scrooge: has("scrooge") };
68
89
  }
69
90
 
70
91
  const tierOf = (profile, prov) => profile?.providers?.[prov]?.tier || "api";
@@ -86,7 +107,11 @@ const FORECAST = { easy: 0.3e6, medium: 1.5e6, hard: 6e6 }; // tokens
86
107
  const CREW_PREF = { hard: ["codex", "glm", "kimi", "deepseek", "openrouter"], medium: ["kimi", "glm", "codex", "deepseek", "openrouter"], easy: ["deepseek", "kimi", "glm", "codex", "openrouter"] };
87
108
 
88
109
  export function advise(input, world = loadWorld()) {
89
- const { profile, registry, caps, agents, scrooge } = world;
110
+ const { profile, registry, caps, agents, scrooge, roster = BUILTIN_ROSTER } = world;
111
+ // brought (discovered) opencode providers extend the preference list — appended LAST in every
112
+ // tier (unknown strength a priori, like openrouter), so they fill once the curated seats are
113
+ // taken, and are the only option for a user who brought nothing but a custom provider.
114
+ const broughtPref = Object.keys(roster).filter(t => roster[t].discovered && !CREW_PREF.hard.includes(t));
90
115
  const pkgs = (input.packages || []).map(p => ({ title: p.title || "work", difficulty: ["easy", "medium", "hard"].includes(p.difficulty) ? p.difficulty : "medium", kind: p.kind || "code", owner: p.owner === "self" ? "self" : (/(foundation|integration|scaffold)/i.test(p.title) ? "self" : "") }));
91
116
  const horizon = input.horizon || (pkgs.length >= 4 ? "long" : pkgs.length >= 2 ? "medium" : "short");
92
117
  const orchTier = tierOf(profile, "claude");
@@ -121,10 +146,10 @@ export function advise(input, world = loadWorld()) {
121
146
  }
122
147
  if (p.owner === "self") return { ...p, executor: "orchestrator", pool: tierOf(profile, "claude"), reason: "architect-owned (foundation/integration doctrine) — the orchestrator keeps the shared contract in its own hands" };
123
148
  if (mode === "solo") return { ...p, executor: "orchestrator", pool: tierOf(profile, "claude"), reason: "small enough to do inline" };
124
- const pref = CREW_PREF[p.difficulty].filter(a => agents.includes(a));
149
+ const pref = [...CREW_PREF[p.difficulty], ...broughtPref].filter(a => agents.includes(a));
125
150
  const agent = pref.sort((a, b) => (used[a] || 0) - (used[b] || 0))[0] || agents[0] || "deepseek";
126
151
  used[agent] = (used[agent] || 0) + 1;
127
- const pool = tierOf(profile, ROSTER[agent]?.provider || agent);
152
+ const pool = tierOf(profile, roster[agent]?.provider || agent);
128
153
  let est = null;
129
154
  if (pool === "api") { // deepseek API etc — estimate real $ via registry
130
155
  const m = registry.models?.["deepseek-v4-flash"] || { cost_in: 0.14, cost_out: 0.28 };
@@ -165,7 +190,7 @@ export function advise(input, world = loadWorld()) {
165
190
  const selfPkgs = routing.filter(r => r.executor === "orchestrator");
166
191
  const foundationIdx = selfPkgs.length ? [1] : [];
167
192
  const cards = routing.map((r, i) => {
168
- const seat = ROSTER[r.executor];
193
+ const seat = roster[r.executor];
169
194
  return {
170
195
  order: i + 1, title: r.title, difficulty: r.difficulty,
171
196
  // bus identity = the runner's session name (spec's agent part) — glm rides the `opencode`
package/bin/cli.mjs CHANGED
@@ -18,6 +18,8 @@ switch (cmd) {
18
18
  case "doctor": run("bin/doctor.mjs"); break;
19
19
  case "connect": run("bin/connect.mjs"); break;
20
20
  case "profile": run("bin/profile.mjs"); break;
21
+ case "provider": case "providers": run("bin/provider.mjs"); break;
22
+ case "models": run("bin/models.mjs"); break;
21
23
  case "advise": run("bin/advise.mjs"); break;
22
24
  case "verify": run("bin/crew-verify.mjs"); break;
23
25
  case "up": process.argv.splice(2, 1); spawn("/bin/bash", [join(ROOT, "bin/crew.sh"), "up", ...args], { stdio: "inherit", cwd: process.cwd() }).on("exit", c => process.exit(c ?? 0)); break;
@@ -49,6 +51,8 @@ switch (cmd) {
49
51
  trantor doctor where do I stand? hub/plugin/CLIs/auth/keys/profile, with copy-paste fixes
50
52
  trantor connect (re)wire every installed AI CLI to the bus
51
53
  trantor profile declare your plans: trantor profile set claude=max codex=plus deepseek=api
54
+ trantor provider bring ANY model (BYOM): list seats · add <name> --key … · remove <name>
55
+ trantor models browse live models behind each seat + the router's pick per difficulty
52
56
  trantor up … spawn a crew here: trantor up codex kimi deepseek:deepseek opencode:zai-coding-plan
53
57
  trantor down tear the crew down (kills processes, closes windows, no dialogs)
54
58
  trantor ui open the live dashboard (board + flow views)
@@ -78,8 +78,14 @@ const CLI = {
78
78
  claude: { first: `claude{M} -p "$(cat {P})" --dangerously-skip-permissions`,
79
79
  next: `claude -c{M} -p "$(cat {P})" --dangerously-skip-permissions`, mflag: " --model " },
80
80
  };
81
- const cli = CLI[AGENT];
82
- if (!cli) { console.error(`unknown agent '${AGENT}' (known: ${Object.keys(CLI).join(", ")})`); process.exit(1); }
81
+ // BYOM: any agent label that isn't a known native CLI is treated as an opencode-driven provider
82
+ // seat (opencode is the universal adapter). This is what lets a BROUGHT provider — `trantor up
83
+ // <label>:<provider>` for any opencode vendor the user configured — run with no per-provider code
84
+ // here; its model id arrives pre-qualified (`<provider>/<model>`) as CREW_MODEL.
85
+ const NATIVE = new Set(["codex", "gemini", "kimi", "claude"]);
86
+ const cli = CLI[AGENT] || (NATIVE.has(AGENT) ? null : CLI.opencode);
87
+ if (!cli) { console.error(`unknown agent '${AGENT}' (native: ${[...NATIVE].join(", ")}; any other name = an opencode provider seat)`); process.exit(1); }
88
+ if (!CLI[AGENT]) log(`'${AGENT}' is not a built-in seat — running it as an opencode provider (BYOM)`);
83
89
 
84
90
  const RULES = `Rules: you are ${SESSION} on the trantor crew. Work your assigned file(s), report on the bus (relay_send, <280 chars), move your Kanban card as you go (doing -> testing -> done; run the tests in 'testing', use 'failed' + a report if they break). When your work for THIS message is finished, END YOUR TURN — do NOT park, do NOT loop relay_wait; the runner waits for you and will wake you with the next message.`;
85
91
 
package/bin/crew.sh CHANGED
@@ -70,18 +70,18 @@ SCROOGE="$BUS_DIR/engine/bin/scrooge"
70
70
  [ -f "$SCROOGE" ] || SCROOGE="$(command -v scrooge 2>/dev/null || echo scrooge)"
71
71
 
72
72
  # resolve_model <agent> <provider> <task> <diff> -> echoes a runner-ready model id, or empty
73
- # (→ CLI default). Enumeration is CLI-aware and never guesses an endpoint: opencode-managed
74
- # agents list via `opencode models <provider>`; others self-enumerate via the provider's /models.
73
+ # (→ CLI default). PROVIDER-AGNOSTIC: if opencode knows the provider (ANY of its vendors — built-in
74
+ # or BROUGHT), enumerate via `opencode models <provider>`; else self-enumerate via the provider's
75
+ # OpenAI-compatible /models. No hardcoded provider list — a newly-brought opencode provider routes
76
+ # with zero change here. Never guesses an endpoint.
75
77
  resolve_model() {
76
78
  local agent="$1" provider="$2" task="$3" diff="$4" cands="" out=""
77
- case "$agent" in
78
- opencode|deepseek|openrouter)
79
- cands="$(opencode models "$provider" 2>/dev/null | tr '\n' ' ')"
80
- [ -n "$cands" ] || { echo "[crew] no live models via 'opencode models $provider' — CLI default" >&2; return 0; }
81
- out="$(python3 "$SCROOGE" route --candidates "$cands" -t "$task" -d "$diff" --json 2>/dev/null)" ;;
82
- *)
83
- out="$(python3 "$SCROOGE" route --provider "$provider" -t "$task" -d "$diff" --json 2>/dev/null)" ;;
84
- esac
79
+ cands="$(opencode models "$provider" 2>/dev/null | tr '\n' ' ')"
80
+ if [ -n "$cands" ]; then
81
+ out="$(python3 "$SCROOGE" route --candidates "$cands" -t "$task" -d "$diff" --json 2>/dev/null)"
82
+ else
83
+ out="$(python3 "$SCROOGE" route --provider "$provider" -t "$task" -d "$diff" --json 2>/dev/null)"
84
+ fi
85
85
  [ -n "$out" ] || { echo "[crew] live model selection failed for $agent:$provider — CLI default" >&2; return 0; }
86
86
  printf '%s' "$out" | python3 -c 'import json,sys
87
87
  try: print(json.load(sys.stdin).get("qualified") or "")
package/bin/models.mjs ADDED
@@ -0,0 +1,67 @@
1
+ #!/usr/bin/env node
2
+ // trantor models — browse the live models behind your crew seats (opencode is the adapter).
3
+ //
4
+ // trantor models # every opencode-driven seat + how many live models it serves
5
+ // trantor models <provider> # list that provider's live models + what the router picks per
6
+ // # difficulty (so you can see hard→strong, easy→cheap at a glance)
7
+ import { execSync } from "node:child_process";
8
+ import { dirname, join } from "node:path";
9
+ import { homedir } from "node:os";
10
+ import { existsSync } from "node:fs";
11
+ import { pathToFileURL, fileURLToPath } from "node:url";
12
+ import { buildRoster, loadWorld } from "./advise.mjs";
13
+
14
+ const H = homedir();
15
+ const C = { dim: "\x1b[2m", grn: "\x1b[32m", gold: "\x1b[38;5;208m", off: "\x1b[0m" };
16
+ const has = (c) => { try { execSync(`command -v ${c}`, { stdio: "ignore", shell: "/bin/sh" }); return true; } catch { return false; } };
17
+ const SCROOGE = (() => {
18
+ const bundled = join(dirname(dirname(fileURLToPath(import.meta.url))), "engine", "bin", "scrooge");
19
+ if (existsSync(bundled)) return bundled;
20
+ try { return execSync("command -v scrooge", { encoding: "utf8" }).trim(); } catch { return "scrooge"; }
21
+ })();
22
+
23
+ const liveModels = (providerOc) => {
24
+ try { return execSync(`opencode models ${providerOc} 2>/dev/null`, { encoding: "utf8" }).split("\n").filter(Boolean); }
25
+ catch { return []; }
26
+ };
27
+
28
+ function routePick(candList, diff) {
29
+ try {
30
+ const out = execSync(`python3 ${SCROOGE} route --candidates ${JSON.stringify(candList.join(" "))} -t code -d ${diff} --json 2>/dev/null`, { encoding: "utf8" });
31
+ return JSON.parse(out).qualified || "?";
32
+ } catch { return "?"; }
33
+ }
34
+
35
+ function listAll() {
36
+ const { roster, agents } = loadWorld();
37
+ if (!has("opencode")) { console.log("opencode not on PATH — it's the adapter that serves crew models. Install it, then re-run."); return; }
38
+ console.log("CREW MODELS — opencode-driven seats (● available now)\n");
39
+ for (const [label, s] of Object.entries(roster)) {
40
+ if (s.cli !== "opencode") continue;
41
+ const n = liveModels(s.providerOc).length;
42
+ const dot = agents.includes(label) ? `${C.grn}●${C.off}` : `${C.dim}○${C.off}`;
43
+ console.log(` ${dot} ${label.padEnd(14)} ${C.dim}${s.providerOc}${C.off} ${n} live models ${C.dim}trantor models ${label}${C.off}`);
44
+ }
45
+ console.log(`\n${C.dim}detail + routing preview:${C.off} trantor models <provider>`);
46
+ }
47
+
48
+ function detail(name) {
49
+ const { roster } = loadWorld();
50
+ const seat = roster[name] || Object.values(roster).find(s => s.providerOc === name);
51
+ const providerOc = seat?.providerOc || name;
52
+ const models = liveModels(providerOc);
53
+ if (!models.length) { console.log(`No live models for '${providerOc}' (opencode offline / no key / unknown provider).`); return; }
54
+ console.log(`${C.gold}${providerOc}${C.off} — ${models.length} live models\n`);
55
+ for (const m of models.slice(0, 60)) console.log(` ${m}`);
56
+ if (models.length > 60) console.log(` ${C.dim}… +${models.length - 60} more${C.off}`);
57
+ if (existsSync(SCROOGE) || has("scrooge")) {
58
+ console.log(`\n${C.gold}router picks (code task):${C.off}`);
59
+ for (const d of ["easy", "medium", "hard"]) console.log(` ${d.padEnd(7)} → ${routePick(models, d)}`);
60
+ console.log(`${C.dim}(run scrooge-capabilities to (re)score the catalog for accurate difficulty routing)${C.off}`);
61
+ }
62
+ }
63
+
64
+ if (process.argv[1] && import.meta.url === pathToFileURL(process.argv[1]).href) {
65
+ const arg = process.argv[2];
66
+ if (!arg) listAll(); else detail(arg.toLowerCase());
67
+ }
@@ -0,0 +1,115 @@
1
+ #!/usr/bin/env node
2
+ // trantor provider — bring ANY model to the crew (BYOM). opencode is a universal adapter, so a
3
+ // provider you configure there (or declare here) becomes a crew seat with no code change.
4
+ //
5
+ // trantor provider # list seats: built-in + discovered, with availability + tier
6
+ // trantor provider add <name> [--key sk-…] [--plan api|coding-plan|max] [--label <bus-name>]
7
+ // trantor provider remove <name> # drop it from your profile (leaves the key in place)
8
+ //
9
+ // `add` writes <NAME>_API_KEY to ~/.agent-bus/.env (if --key given), declares the plan in your
10
+ // quota profile, verifies opencode can see the provider's models, and prints the seat spec. The
11
+ // Advisor then routes to it automatically; run `scrooge-capabilities` so it routes well by difficulty.
12
+ import { readFileSync, writeFileSync, existsSync, mkdirSync, chmodSync, appendFileSync } from "node:fs";
13
+ import { join, dirname } from "node:path";
14
+ import { homedir } from "node:os";
15
+ import { execSync } from "node:child_process";
16
+ import { pathToFileURL } from "node:url";
17
+ import { buildRoster, loadWorld } from "./advise.mjs";
18
+
19
+ const H = homedir();
20
+ const ENV = join(H, ".agent-bus", ".env");
21
+ const read = (p, fb) => { try { return JSON.parse(readFileSync(p, "utf8")); } catch { return fb; } };
22
+ const has = (c) => { try { execSync(`command -v ${c}`, { stdio: "ignore", shell: "/bin/sh" }); return true; } catch { return false; } };
23
+ const C = { dim: "\x1b[2m", grn: "\x1b[32m", red: "\x1b[31m", yel: "\x1b[33m", gold: "\x1b[38;5;208m", off: "\x1b[0m" };
24
+ const envKeyName = (p) => `${String(p).toUpperCase().replace(/[^A-Z0-9]/g, "_")}_API_KEY`;
25
+
26
+ function opencodeModelCount(providerOc) {
27
+ if (!has("opencode")) return null;
28
+ try { return execSync(`opencode models ${providerOc} 2>/dev/null`, { encoding: "utf8" }).split("\n").filter(Boolean).length; }
29
+ catch { return 0; }
30
+ }
31
+
32
+ function listSeats() {
33
+ const world = loadWorld();
34
+ const { roster, agents, profile } = world;
35
+ console.log("CREW SEATS — built-in + brought (BYOM). ● = available now\n");
36
+ for (const [label, s] of Object.entries(roster)) {
37
+ const live = agents.includes(label);
38
+ const tier = profile?.providers?.[s.provider]?.tier || (s.cli === "opencode" ? "—" : "");
39
+ const kind = s.discovered ? "brought " : "built-in";
40
+ const models = s.cli === "opencode" ? opencodeModelCount(s.providerOc) : null;
41
+ const mtxt = models == null ? "" : `${models} models`;
42
+ const dot = live ? `${C.grn}●${C.off}` : `${C.dim}○${C.off}`;
43
+ console.log(` ${dot} ${label.padEnd(14)} ${C.dim}${kind}${C.off} launch: ${s.launch.padEnd(26)} ${tier ? `tier=${tier}` : ""} ${C.dim}${mtxt}${C.off}`);
44
+ }
45
+ console.log(`\n${C.dim}add one:${C.off} trantor provider add <name> --key sk-… --plan api`);
46
+ console.log(`${C.dim}browse models:${C.off} trantor models [<provider>]`);
47
+ }
48
+
49
+ function addProvider(name, opts) {
50
+ if (!name) { console.error("usage: trantor provider add <name> [--key sk-…] [--plan api] [--label <bus-name>]"); process.exit(1); }
51
+ const provider = name.toLowerCase();
52
+ const label = (opts.label || provider).toLowerCase().replace(/[^a-z0-9-]/g, "-");
53
+ const plan = (opts.plan || "api").toLowerCase();
54
+
55
+ // 1) key → ~/.agent-bus/.env (the runner sources it; opencode reads <NAME>_API_KEY for known providers)
56
+ if (opts.key) {
57
+ mkdirSync(dirname(ENV), { recursive: true });
58
+ const k = envKeyName(provider);
59
+ let cur = existsSync(ENV) ? readFileSync(ENV, "utf8") : "";
60
+ if (new RegExp(`^${k}=`, "m").test(cur)) {
61
+ cur = cur.replace(new RegExp(`^${k}=.*$`, "m"), `${k}=${opts.key}`);
62
+ writeFileSync(ENV, cur);
63
+ } else {
64
+ appendFileSync(ENV, `${cur && !cur.endsWith("\n") ? "\n" : ""}# ${provider} — brought via 'trantor provider add'\n${k}=${opts.key}\n`);
65
+ }
66
+ try { chmodSync(ENV, 0o600); } catch {}
67
+ console.log(`${C.grn}✓${C.off} wrote ${envKeyName(provider)} → ~/.agent-bus/.env (chmod 600)`);
68
+ }
69
+
70
+ // 2) declare the plan in the quota profile (drives the Advisor's tier/cost reasoning)
71
+ try {
72
+ execSync(`node ${join(dirname(new URL(import.meta.url).pathname), "profile.mjs")} set ${provider}=${plan}`, { stdio: "ignore" });
73
+ console.log(`${C.grn}✓${C.off} profile: ${provider}=${plan}`);
74
+ } catch (e) { console.log(`${C.yel}⚠${C.off} could not set profile (run: trantor profile set ${provider}=${plan})`); }
75
+
76
+ // 3) verify opencode can see the provider's models
77
+ const n = opencodeModelCount(provider);
78
+ if (n == null) console.log(`${C.yel}⚠${C.off} opencode not on PATH — install it to run this seat (it's the universal adapter)`);
79
+ else if (n === 0) console.log(`${C.yel}⚠${C.off} opencode lists 0 models for '${provider}'. If it's a known provider, the key above is enough; for a custom endpoint, add it to ~/.config/opencode/opencode.json (provider config). Then re-check: trantor models ${provider}`);
80
+ else console.log(`${C.grn}✓${C.off} opencode sees ${n} models for '${provider}'`);
81
+
82
+ // 4) score it for difficulty-aware routing + show the seat
83
+ console.log(`\n${C.gold}Seat ready.${C.off} Launch it:`);
84
+ console.log(` trantor up ${label === provider ? label : `${label}:${provider}`} ${C.dim}# live-selects the best model for the work${C.off}`);
85
+ console.log(` trantor up ${label}:${provider}/<model> ${C.dim}# pin a specific model${C.off}`);
86
+ console.log(`${C.dim}For difficulty-aware routing across its catalog, score it once (weekly):${C.off} scrooge-capabilities`);
87
+ }
88
+
89
+ function removeProvider(name) {
90
+ if (!name) { console.error("usage: trantor provider remove <name>"); process.exit(1); }
91
+ const FILE = join(H, ".agent-bus", "profile.json");
92
+ const prof = read(FILE, { providers: {} });
93
+ if (prof.providers && prof.providers[name.toLowerCase()]) {
94
+ delete prof.providers[name.toLowerCase()];
95
+ writeFileSync(FILE, JSON.stringify(prof, null, 2) + "\n");
96
+ console.log(`${C.grn}✓${C.off} removed '${name}' from your profile (the ${envKeyName(name)} key is left in place; delete it from ~/.agent-bus/.env if you want it gone)`);
97
+ } else {
98
+ console.log(`'${name}' is not in your profile.`);
99
+ }
100
+ }
101
+
102
+ if (process.argv[1] && import.meta.url === pathToFileURL(process.argv[1]).href) {
103
+ const [, , sub, ...rest] = process.argv;
104
+ const opts = {}; const pos = [];
105
+ for (let i = 0; i < rest.length; i++) {
106
+ if (rest[i] === "--key") opts.key = rest[++i];
107
+ else if (rest[i] === "--plan") opts.plan = rest[++i];
108
+ else if (rest[i] === "--label") opts.label = rest[++i];
109
+ else pos.push(rest[i]);
110
+ }
111
+ if (!sub || sub === "list") listSeats();
112
+ else if (sub === "add") addProvider(pos[0], opts);
113
+ else if (sub === "remove" || sub === "rm") removeProvider(pos[0]);
114
+ else { console.error(`unknown subcommand '${sub}' — use: list | add | remove`); process.exit(1); }
115
+ }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "trantor",
3
- "version": "0.17.46",
3
+ "version": "0.17.47",
4
4
  "type": "module",
5
5
  "bin": {
6
6
  "trantor": "bin/cli.mjs"
@@ -54,11 +54,15 @@ EXACT launch spec per provider (do not improvise these):
54
54
 
55
55
  `agent:provider` live-selects the best model now; `agent:provider/model` pins one. Example:
56
56
  `trantor up codex kimi deepseek:deepseek opencode:zai-coding-plan --task code --difficulty hard`.
57
- **Whatever the advisor's `launch` field says, run that verbatim** — the roster above is the menu,
58
- but the advisor already picked the right seats and specs for THIS work (a user who's only brought
59
- an OpenRouter key, for instance, gets `openrouter` for everything; one who brought five providers
60
- gets a load-balanced spread). New providers a user brings (OpenRouter today; any opencode-supported
61
- vendor next) appear automatically once declared via `trantor profile set <name>=api` + a key.
57
+ **Whatever the advisor's `launch` field says, run that verbatim** — the roster above is just the
58
+ built-in menu, but the advisor picks the right seats/specs for THIS work (a user who's only brought
59
+ an OpenRouter key gets `openrouter` for everything; one who brought five providers gets a
60
+ load-balanced spread). **BYOM is fully general:** the roster is DERIVED, not hardcoded — ANY
61
+ opencode-supported provider the user configures becomes a seat with its own bus label, no code
62
+ change. A user adds one with `trantor provider add <name> --key … --plan api` (then
63
+ `scrooge-capabilities` to score it for difficulty routing); `trantor provider` lists all seats and
64
+ `trantor models [<provider>]` browses the live models + the router's pick per difficulty. If the
65
+ advisor routes to a brought provider you don't recognize, that's expected — run its `launch` spec.
62
66
 
63
67
  ⚠️ **Gemini CLI is RETIRED (Google killed the free seat 2026-06-18).** The advisor no longer
64
68
  offers it and you must NOT fire up `gemini` — `gemini --yolo` exits 1 and crash-loops on the