Some checks are pending
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / preflight (push) Waiting to run
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / vps-smoke (push) Waiting to run
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / rust-primitives (map[crates:frustration-matrix,kei-frustration-loop,kei-skill-importer,kei-projects-index,kei-projects-watcher,kei-gdrive-import,kei-leak-matrix,kei-skills,kei-gateway,kei-cron-scheduler,kei-export-trajectories,kei-backend-daytona,kei-d… (push) Blocked by required conditions
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / rust-primitives (map[crates:kei-compute-baremetal,kei-compute-vultr,kei-compute-linode,kei-compute-digitalocean,kei-svc-systemd,kei-llm-bridge-mlx name:hosted-sleep-compute]) (push) Blocked by required conditions
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / rust-primitives (map[crates:kei-diff,kei-scheduler,kei-watch,kei-prune,kei-discover,kei-brain-view,kei-hibernate,kei-ledger-sign,kei-fork name:wave13-15]) (push) Blocked by required conditions
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / rust-primitives (map[crates:kei-git-gitea,kei-git-forgejo,kei-git-gitlab,kei-git-bitbucket,kei-memory-sled,kei-memory-redis,kei-memory-postgres,kei-memory-sqlite,kei-auth-google,kei-auth-apple,kei-auth-magiclink,kei-auth-webauthn,kei-notify-slack,kei-n… (push) Blocked by required conditions
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / rust-primitives (map[crates:kei-ledger,kei-migrate,kei-changelog,kei-memory,kei-store,kei-conflict-scan,kei-refactor-engine,kei-graph-check,kei-shared,kei-dna-index,kei-pet name:core]) (push) Blocked by required conditions
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / rust-primitives (map[crates:kei-machine-probe,kei-llm-ollama,kei-llm-llamacpp,kei-llm-mlx,kei-llm-router,kei-model name:llm-stack]) (push) Blocked by required conditions
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / rust-primitives (map[crates:kei-router,kei-sage,kei-task,kei-chat-store,kei-crossdomain,kei-search-core,kei-content-store,kei-social-store,kei-curator,kei-auth,kei-artifact name:mcp-lbm]) (push) Blocked by required conditions
CI (Forgejo Actions — self-hosted runner on Mac, host mode) / rust-primitives (map[crates:keisei,kei-forge,kei-runtime,kei-runtime-core,kei-atom-discovery,kei-agent-runtime,kei-capability,kei-provision,kei-entity-store,kei-pipe,kei-cache,kei-spawn,kei-replay name:atom-substrate]) (push) Blocked by required conditions
Closes the "Claude Code as single primary" gap. Now `kei` (no args) execs
whichever CLI is configured as primary, and ANY MCP-capable orchestrator
can spawn KeiSeiKit agents on any backend via the built-in spawn_agent tool.
## A — orchestrator picker
bin/kei now reads ~/.claude/config/primary.toml and execs that CLI instead
of hardcoding claude. New arms:
kei pick interactive menu → set primary → launch it
kei --on=<backend> one-shot launch of <backend> (no primary write)
kei primary [<b>] get/set primary
Splash shows `primary CLI: <backend>` so the orchestrator is visible.
Failure mode: if primary's CLI isn't on PATH, prints install hint + offers
`kei pick` recovery.
scripts/kei-pick.sh — Constructor Pattern picker (<140 LOC). Lists all 6
backends with install status (✓/✗), highlights current primary, writes
choice to primary.toml, execs the picked CLI. Honors stdin TTY gate
(RULE TTY-INTERACTIVITY-GATE — -t 0, not -t 1) for non-interactive safety.
## B — spawn_agent MCP tool
_primitives/_rust/kei-mcp/src/handlers/tools.rs gains a built-in
`spawn_agent` tool, exposed alongside discovered atoms:
- inputSchema: { name: str, task: str, on?: backend-enum }
- Calls kei-agent-cli.sh internally with same DNA resolution
- 60s timeout, kill-on-drop
- Honors KEI_AGENT_CLI env for testing
Smoke 2026-05-26 (MCP stdio JSON-RPC round-trip):
spawn_agent(name=smoke-test, on=claude) → "SMOKE-OK" ✅
spawn_agent(name=smoke-test, on=grok) → "SMOKE-OK" ✅
Why it matters: Claude Code has a native Agent tool. Grok / Agy / Copilot /
Kimi don't have an equivalent native sub-agent surface — but they all speak
MCP. spawn_agent gives them KeiSeiKit's sub-agent capability when they're
the orchestrator. The chosen orchestrator no longer caps the sub-agent fleet.
## Other
_primitives/_rust/kei-mcp/Cargo.toml: tokio gains "io-std" feature (was
missing — main.rs uses tokio::io::stdin/stdout). This fixes a latent build
error unrelated to this PR (kei-mcp wasn't building cleanly before).
Tests: tools_list assertions updated for the +1 built-in tool (3 total
instead of 2 with atoms; 1 instead of 0 on empty root). All MCP tests pass.
Assembler 3/3 golden tests still pass (provider field is optional).
153 lines
4.5 KiB
Bash
Executable file
153 lines
4.5 KiB
Bash
Executable file
#!/usr/bin/env bash
|
|
# kei-pick — interactive orchestrator picker.
|
|
#
|
|
# Shows installed LLM CLIs, lets the user choose one, writes it to
|
|
# ~/.claude/config/primary.toml, then exec's it (so the shell becomes
|
|
# the picked orchestrator). Designed for `kei pick`.
|
|
#
|
|
# Non-interactive (no TTY): just shows status and exits 0.
|
|
|
|
set -eu
|
|
|
|
KEI_PRIMARY_CFG="${KEI_PRIMARY_CFG:-$HOME/.claude/config/primary.toml}"
|
|
|
|
# Mirrors scripts/kei-agent-cli.sh::backend_bin and bin/kei::backend_bin_for.
|
|
backend_bin() {
|
|
case "$1" in
|
|
claude) echo "claude" ;;
|
|
grok) echo "grok" ;;
|
|
agy|antigravity) echo "agy" ;;
|
|
copilot) echo "copilot" ;;
|
|
kimi) echo "kimi" ;;
|
|
codex) echo "codex" ;;
|
|
*) return 1 ;;
|
|
esac
|
|
}
|
|
|
|
backend_label() {
|
|
case "$1" in
|
|
claude) echo "Claude Code (Anthropic)" ;;
|
|
grok) echo "Grok Build TUI (xAI)" ;;
|
|
agy) echo "Antigravity / Gemini (Google)" ;;
|
|
copilot) echo "GitHub Copilot CLI (Microsoft/GitHub)" ;;
|
|
kimi) echo "Kimi Code CLI (Moonshot) — TUI-only for agents" ;;
|
|
codex) echo "Codex CLI (OpenAI)" ;;
|
|
esac
|
|
}
|
|
|
|
current_primary() {
|
|
[ -f "$KEI_PRIMARY_CFG" ] || { echo "claude"; return; }
|
|
awk -F'=' '/^provider[[:space:]]*=/ {
|
|
gsub(/^[[:space:]]+|[[:space:]]+$/, "", $2)
|
|
gsub(/^"|"$/, "", $2)
|
|
print $2; exit
|
|
}' "$KEI_PRIMARY_CFG"
|
|
}
|
|
|
|
# --- list installed backends ------------------------------------------
|
|
BACKENDS=(claude grok agy copilot kimi codex)
|
|
INSTALLED=()
|
|
for b in "${BACKENDS[@]}"; do
|
|
bin=$(backend_bin "$b")
|
|
if command -v "$bin" >/dev/null 2>&1; then
|
|
INSTALLED+=("$b")
|
|
fi
|
|
done
|
|
|
|
cur=$(current_primary)
|
|
|
|
# --- non-interactive: just show status --------------------------------
|
|
# Gate on stdin (RULE TTY-INTERACTIVITY-GATE): -t 0, not -t 1.
|
|
# curl|bash tees stdout, so -t 1 false ≠ non-interactive.
|
|
if [ ! -t 0 ]; then
|
|
echo "kei pick — non-interactive mode"
|
|
echo "current primary: $cur"
|
|
echo "installed backends: ${INSTALLED[*]:-none}"
|
|
echo "(run \`kei pick\` from a real terminal for the picker)"
|
|
exit 0
|
|
fi
|
|
|
|
# --- interactive picker -----------------------------------------------
|
|
C0="" CB="" CC="" CD=""
|
|
if [ -t 1 ]; then
|
|
C0=$'\033[0m'
|
|
CB=$'\033[1;38;5;39m' # blue
|
|
CC=$'\033[1;38;5;220m' # gold
|
|
CD=$'\033[2m' # dim
|
|
fi
|
|
|
|
cat <<EOF
|
|
|
|
${CB}╔════════════════════════════════════════════╗
|
|
║ KeiSeiKit · orchestrator picker ║
|
|
╚════════════════════════════════════════════╝${C0}
|
|
|
|
Pick the LLM CLI that becomes your primary shell.
|
|
Any agent invocation (\`kei agent <name>\`) routes here unless DNA overrides.
|
|
|
|
EOF
|
|
|
|
i=1
|
|
for b in "${BACKENDS[@]}"; do
|
|
bin=$(backend_bin "$b")
|
|
label=$(backend_label "$b")
|
|
if command -v "$bin" >/dev/null 2>&1; then
|
|
mark="${CC}✓${C0}"
|
|
else
|
|
mark="${CD}✗${C0}"
|
|
label="$label ${CD}(not installed)${C0}"
|
|
fi
|
|
cur_mark=""
|
|
[ "$b" = "$cur" ] && cur_mark="${CC} ← current${C0}"
|
|
printf " ${CB}%d${C0}) %s %-10s %s%s\n" "$i" "$mark" "$b" "$label" "$cur_mark"
|
|
i=$((i+1))
|
|
done
|
|
|
|
echo
|
|
printf " ${CB}q${C0}) cancel (keep current: ${CC}%s${C0})\n\n" "$cur"
|
|
printf "Pick [1-${#BACKENDS[@]}/q]: "
|
|
read -r choice
|
|
choice="${choice:-q}"
|
|
|
|
case "$choice" in
|
|
q|Q|"") echo "cancelled."; exit 0 ;;
|
|
[1-9])
|
|
idx=$((choice-1))
|
|
if [ $idx -ge ${#BACKENDS[@]} ] || [ $idx -lt 0 ]; then
|
|
echo "invalid choice: $choice" >&2; exit 2
|
|
fi
|
|
new="${BACKENDS[$idx]}"
|
|
;;
|
|
*) echo "invalid choice: $choice" >&2; exit 2 ;;
|
|
esac
|
|
|
|
bin=$(backend_bin "$new")
|
|
if ! command -v "$bin" >/dev/null 2>&1; then
|
|
echo
|
|
echo "${CC}'$new' is not installed.${C0}"
|
|
echo "Set as primary anyway (you'll need to install it before \`kei\` will work)? [y/N]: "
|
|
read -r confirm
|
|
case "$confirm" in y|Y|yes) ;; *) echo "cancelled."; exit 0 ;; esac
|
|
fi
|
|
|
|
mkdir -p "$(dirname "$KEI_PRIMARY_CFG")"
|
|
printf '# kei primary — written %s\nprovider = "%s"\n' \
|
|
"$(date -u +%Y-%m-%dT%H:%M:%SZ)" "$new" > "$KEI_PRIMARY_CFG"
|
|
|
|
echo
|
|
echo "${CC}✓${C0} primary set: $cur → ${CC}$new${C0}"
|
|
echo " config: $KEI_PRIMARY_CFG"
|
|
echo
|
|
|
|
if [ -n "${KEI_NO_LAUNCH:-}" ]; then
|
|
echo "(skipping launch — KEI_NO_LAUNCH set)"
|
|
exit 0
|
|
fi
|
|
|
|
if ! command -v "$bin" >/dev/null 2>&1; then
|
|
echo "${CD}skipping launch — $bin not on PATH; install it then run \`kei\`.${C0}"
|
|
exit 0
|
|
fi
|
|
|
|
echo "launching $new..."
|
|
exec "$bin"
|