`homeboy runner`
Synopsis
homeboy runner <COMMAND>runner manages durable execution backends. SSH runners are a capability on a homeboy server record, so the common Lab flow uses one ID for the machine and its runner. Local runners remain standalone because they describe this machine rather than an SSH server.
Runner configuration separates printable environment from secrets:
envis for non-secret values that are useful in diagnostics, such asHOMEBOY_PUBLIC_ARTIFACT_BASE_URL.secret_envis for execution-time secret references like{ "env": "NAME" }or{ "file": "~/.config/homeboy/secrets/name" }.- Command output redacts sensitive names in
envand prints onlysecret_envreferences, never resolved secret values.
Subcommands
add
homeboy runner add <id> --workspace-root <path>
homeboy runner add <server-id> --server <server-id> --workspace-root <path>
homeboy runner add --json <spec>Options:
--kind local|ssh: explicit runner kind. Defaults tosshwhen--serveris set, otherwiselocal.--server <server-id>: existinghomeboy serverrecord for SSH runners. For SSH runners,<id>must match<server-id>.--workspace-root <path>: workspace root on the runner machine.--homeboy-path <path>: Homeboy binary path on the runner machine.--daemon: marks the runner as daemon-preferred for future commands.--concurrency-limit <n>: maximum concurrent workflows this runner should accept.--artifact-policy <label>: artifact policy label reserved for future execution commands.
enable
homeboy runner enable <server-id> --workspace-root <path>
homeboy runner enable <server-id> --workspace-root <path> --concurrency-limit 4 --artifact-policy copyEnables runner capability on an existing SSH server. This is the recommended onboarding path for any machine that should accept Homeboy runner work:
homeboy server create <runner-id> --host <host> --user <user> --port 22
homeboy runner enable <runner-id> --workspace-root <workspace-root> --concurrency-limit 4 --artifact-policy copy
homeboy runner connect <runner-id>After this, <runner-id> is both the server ID and the runner ID.
Commands that are both resource-policy hot and portable for Lab offload (audit, full lint, test, bench run, and trace) auto-select a default runner when --runner is omitted. Selection is conservative:
--runner <id>always wins.--force-hotonly suppresses the resource-policy warning. If a default Lab runner is available for a portable hot command, Homeboy refuses to use--force-hotas an implicit local bypass.--force-hot --allow-local-hotkeeps a portable hot command local even when a default Lab runner is available, unless a command-specific host policy denies local execution. For benchmarks,homeboy config set /bench/local_execution '"denied"'makes localhomeboy benchexecution fail closed until the global config is changed back.lab.preferred_runneris used when it names an SSH runner, even if that runner is not connected yet.- Without
lab.preferred_runner, Homeboy auto-selects only when exactly one SSH runner is configured or exactly one SSH runner is already connected. - With a preferred or uniquely configured Lab runner,
homeboy bench <component>routes to Lab directly;--runner <id>is only needed to override an ambiguous or non-default runner selection. - Local runners are never auto-selected.
- If the auto-selected runner is disconnected, Homeboy attempts a short bounded
runner connectbefore execution. Connection failure prints the reason and falls back to local execution. - Explicit
--runner <id>also attempts to connect a disconnected runner, but connection failure remains a command error instead of falling back silently.
Observation metadata records the routing decision under metadata.lab_offload when an observed run is created. The stable contract is schema: "homeboy/lab-offload/v1" and keeps the existing top-level compatibility fields: source is automatic or explicit; status is offloaded, skipped, or fallback; successful offloads include runner_id plus remote_workspace; local fallback records the runner and fallback_reason; skipped local execution records why no automatic offload was used, such as force_hot, force_hot_local_override, or no_default_runner. The same object also carries plan_id and plan-derived phase fields including sync_mode, capability_preflight, extension_parity, and patch_captured.
Lab offload support is intentionally command-specific:
| Command | Auto offload | Explicit --runner | Decision |
|---|---|---|---|
audit full workspace | Yes | Yes | Safe single-workspace replay after snapshot sync. |
audit --changed-since | No | No | Runs locally for now because changed-since audit depends on git base refs that Lab sync may not have fetched. The Lab plan records the skipped local-only decision. |
bench run / default bench run | Yes | Yes | Safe single-workspace replay; local baseline/ratchet writes are treated as mutation flags. |
lint full workspace | Yes | Yes | Safe single-workspace replay; --fix is treated as a mutation flag. |
lint --changed-since / lint --changed-only | No | No | Runs locally for now because changed-file scopes are not represented in the Lab portability contract yet. The Lab plan records the skipped local-only decision. |
test full workspace | Yes | Yes | Safe single-workspace replay with runner extension parity preflight. |
test --changed-since | No | No | Runs locally for now because changed-since test selection depends on git base refs that Lab sync may not have fetched. The Lab plan records the skipped local-only decision. |
trace | Yes | Yes | Safe single-workspace replay with Playwright/browser capability gate. |
rig up | No | No | Stays local because rig pipelines manage local services, leases, ports, and declared filesystem paths that the current single-workspace snapshot cannot safely mirror. |
fleet exec | No | No | Stays local because fleet execution depends on local fleet/project/server config before opening SSH sessions to each project; runner-side config parity is not guaranteed. |
Local-only resource-pressure commands still get resource-policy warnings, but those warnings explain why Lab offload is unavailable instead of suggesting --runner.
Configure a preferred runner with:
homeboy config set /lab/preferred_runner '"<runner-id>"'doctor
homeboy runner doctor local
homeboy runner doctor <runner-id>
homeboy runner doctor <runner-id> --path <component-path> --extension rust
homeboy runner doctor <runner-id> --require-tool zip --require-tool unzipDiagnoses a local or configured SSH runner without mutating it. Use local,
localhost, or self to inspect this machine without creating a runner record.
The JSON payload uses command: "runner.doctor" and includes runner_id,
status, capabilities, and warning/error details when a capability probe fails.
Use doctor before connect when you need to know whether Homeboy, Git, SSH,
and the configured workspace root are usable on the target machine.
Pass one or more --require-tool <command> values when a provider or job path
knows it needs additional runner-side commands before starting expensive work.
Doctor resolves each command on the runner PATH and reports missing requested
tools as tool.required.<command> errors with install/setup remediation. This is
generic: provider layers declare their own tools; Homeboy core only checks command
availability.
Programmatic runner execution can use the same generic boundary through
RunnerCapabilityPreflight.required_commands. Those commands are checked before
remote execution starts, alongside existing required tools, components, and
environment variables.
Pass one or more --extension <id> values to validate extension parity before
Lab offload. Doctor runs the same homeboy extension show <id> contract on the
target runner that test offload uses at execution time. --path sets the probe
working directory when the extension should resolve from a specific component
checkout. Missing extensions are reported as extension.parity errors with an
install command such as homeboy extension install <source> --id rust.
Lab offload for portable resource-pressure commands uses the same capability vocabulary before running on
an explicit --runner. Homeboy currently gates lint, test, audit,
bench, and trace against the source worktree’s lightweight tool signals:
package.jsonrequiresnodeandnpm.pnpm-lock.yamlrequiresnodeandpnpm.composer.jsonrequiresphpandcomposer.- Docker/Compose files require
docker. tracerequires Playwright plus browser binaries.
When an explicit runner is missing required tools, the command fails before
workspace sync with a runner_capabilities validation error and remediation.
The same central policy returns a local fallback reason for future automatic
Lab offload selection.
connect
homeboy runner connect <runner-id>
homeboy runner connect <controller-id> --reverse --reverse-runner <runner-id> --broker-url <url>Starts a loopback-only Homeboy daemon on the runner and opens an SSH tunnel to
it. This is the preferred Lab execution path because later runner exec calls
can use the daemon session instead of ad-hoc SSH command execution. The JSON
payload uses command: "runner.connect" and reports connection state such as
the runner ID, tunnel endpoint, daemon endpoint, and persisted session metadata.
Reverse runner connections record the runner-initiated session substrate and use
the controller daemon as the broker. A reverse runner can register itself with
POST /runner/sessions; the controller then reports that runner as connected
and routes runner exec through brokered jobs instead of a direct daemon URL.
The broker exposes POST /runner/jobs, POST /runner/jobs/claim,
POST /runner/jobs/<job-id>/events, and POST /runner/jobs/<job-id>/finish so
controllers can queue work and reverse runners can claim, stream progress, and
return results without inbound access to the lab machine.
Daemon and broker HTTP responses use one canonical envelope. The outer response
reports transport success and the endpoint payload always lives under
data.body; runner clients require that shape and do not parse legacy direct
data payloads.
{
"success": true,
"data": {
"status": 200,
"endpoint": "runner.jobs.submit",
"body": {
"job": {},
"poll": {}
}
}
}For the generic controller-to-runner operator path, see Controller to runner reverse-runner setup. That guide is machine-agnostic and intentionally explicit about what is available now and what remains gated by #2990, #2991, #2992, and #2947 before production broker exposure.
work
homeboy runner work <runner-id> --broker-url <url>
homeboy runner work <runner-id> --broker-url <url> --project <project-id> --lease-ms 30000
homeboy runner work <runner-id> --broker-url <url> --loopClaims one brokered reverse-runner job for the runner, executes it on the runner
machine under the runner’s local policy, streams a progress event, and finishes
the broker job with stdout, stderr, and exit code. This is the runner-side half
of reverse runner exec; it uses outbound HTTP from the lab to the controller
broker and does not require inbound SSH or a public listening port on the lab.
The command exits 0 when no job is available, with claimed: false in the JSON
payload. When a job is claimed, the process exit code matches the executed
command’s exit code.
Use --loop for a long-running reverse runner service. Loop mode emits one
structured JSON status line per lifecycle event to stderr so systemd/journald can
index startup, idle backoff, job completion, transient broker failures, and
shutdown without mixing those events into the final stdout JSON payload. Empty
queues use exponential backoff controlled by --idle-backoff-ms and
--max-idle-backoff-ms, so workers do not hot-spin when no work is available.
Transient broker failures sleep for --broker-failure-backoff-ms and exit
non-zero after --broker-retry-limit consecutive failures. SIGINT and
SIGTERM request graceful shutdown after the current claim attempt or job.
Minimal Homeboy Lab systemd unit:
[Unit]
Description=Homeboy reverse runner worker
After=network-online.target
Wants=network-online.target
[Service]
Type=simple
User=chubes
WorkingDirectory=/home/chubes
ExecStart=/usr/local/bin/homeboy runner work homeboy-lab --broker-url https://controller.example.com --loop --idle-backoff-ms 1000 --max-idle-backoff-ms 30000 --broker-retry-limit 12
Restart=on-failure
RestartSec=10
KillSignal=SIGTERM
[Install]
WantedBy=multi-user.targetThe unit intentionally leaves authentication out until the broker auth contract lands; configure the broker URL and any future auth material using the production mechanism for issue #2990 rather than embedding secrets in the unit file.
status
homeboy runner status <runner-id>Shows the persisted tunnel/session state for a runner. Use this to determine
whether runner exec will use a connected daemon or needs an explicit fallback.
The JSON payload uses command: "runner.status" and reports whether a saved
session exists, whether the tunnel still appears live, and the recorded endpoint
details.
disconnect
homeboy runner disconnect <runner-id>Closes a persisted runner tunnel session and removes its local session state.
The JSON payload uses command: "runner.disconnect" and reports which session
state was removed. This is safe to run when no live session exists; it is the
explicit cleanup counterpart to runner connect.
list
homeboy runner listshow
homeboy runner show <id>set
homeboy runner set <id> --json <JSON>
homeboy runner set <id> --base64 <BASE64_JSON>
homeboy runner set <id> --json '{"workspace_root":"/srv/homeboy","concurrency_limit":4}'Updates a runner by merging a JSON object into the runner config. SSH runner settings live under servers/<id>.json as the server’s runner capability; local runners live under runners/<id>.json.
Arbitrary runner updates must use --json or --base64; positional key=value and trailing arbitrary --key value updates are not accepted.
trust
homeboy runner trust <runner-id> --project <project-id> --command test --command bench --allow-raw-exec false
homeboy runner trust <runner-id> --workspace-root <runner-workspace-root> --artifact-policy metadata
homeboy runner trust <runner-id> --peer <controller-server-id> --fingerprint SHA256:...Persists controller-side trust policy for a runner. Policy is stored in the runner config as policy, not in transient CLI state. Repeated values are appended without duplicates.
Policy fields:
--project <id>allows a project to use the runner.--command <family>allows a command family such astest,bench,lint,audit,trace,cargo, orrunner.exec.--allow-raw-exec <true|false>controls arbitraryrunner execshell access. SSH runner raw exec is denied by default until this is explicitly true.--workspace-root <path>limits execution to one or more approved runner workspace roots.--artifact-policy <label>records artifact behavior;noneanddenyblock patch capture.--peer <id>records accepted peer/controller server IDs for reverse-runner pairing.--fingerprint <value>records expected peer host keys or equivalent fingerprints.
pair
homeboy runner pair <runner-id> --peer <controller-server-id> --accept-project <project-id> --workspace-root <runner-workspace-root>
homeboy runner pair <runner-id> --fingerprint SHA256:... --allow-raw-exec falsePersists runner-side pairing policy for trusted controllers. pair writes the same durable policy object as trust, using runner-side option names for accepted peer IDs, accepted project IDs, peer fingerprints, workspace roots, and raw exec policy.
remove
homeboy runner remove <id>exec
homeboy runner exec <runner-id> -- <command...>
homeboy runner exec <runner-id> --project <project-id> --cwd /runner/workspace/project -- <command...>
homeboy runner exec <runner-id> --ssh --cwd /runner/workspace/project -- <command...>
homeboy runner exec <runner-id> --cwd /runner/workspace/project --require-path /runner/workspace/project -- <command...>
homeboy runner env <runner-id>
homeboy runner env <runner-id> --show-valuesexec submits the command to the connected runner daemon when homeboy runner connect <runner-id> has established a live loopback tunnel. If no daemon session is connected, local runners execute directly and SSH runners require explicit diagnostic --ssh. SSH runner raw exec is policy-denied by default until policy.allow_raw_exec is explicitly true.
Path rules:
- SSH runners require
workspace_rootso local paths are not silently reused remotely. - SSH
--cwdmust be an absolute path under the configuredworkspace_root. - Omitting
--cwdon an SSH runner uses the runnerworkspace_root. --require-path <path>preflights one or more runner-side paths before execution. Use it when a command references a lab worktree path so missing controller-only paths fail with a structuredrequire_patherror instead of an empty command failure.--project <id>feeds the runner trust policy project allowlist check.--sshis the explicit diagnostic fallback whenconnectis unavailable; daemon execution is preferred because it records job metadata and supports artifact-oriented workflows.- Diagnostic SSH output serializes as
mode: "diagnostic_ssh"and does not include job/event evidence. - Raw SSH execution remains intentionally explicit and should not be used as production Lab/offload evidence; use connected daemon or reverse broker execution for job/event/artifact-compatible output.
Runner job environment:
homeboy runner env <runner-id>shows configured public runner env plussecret_envkeys/references for runner jobs. It does not resolve or print secret values.- Public
envvalues are redacted by default because legacy configs may still contain tokens. Use--show-valuesonly in trusted local/operator contexts;secret_envremains references-only even with--show-values. homeboy ssh <server> -- printenv NAMEinspects the server login shell environment. It does not include runner job env unless the variable is also configured on the server shell.- Use
homeboy runner exec <runner-id> -- printenv NAMEfor final execution-time proof when debugging resolved runner job environment.
Runner metrics:
- Local runner execution, connected daemon jobs, and reverse-runner worker results include a
metricsobject withduration_ms,sample_count, and lightweight resource fields when available. - On Linux runners, metrics are sampled from
/procfor the command process tree and includepeak_rss_bytes,child_process_count_peak,cpu_user_ms, andcpu_system_ms. - CPU accounting is sampled and can miss very short-lived child processes between samples; duration is always recorded, and non-Linux runners report
source: "duration_only".
workspace sync
homeboy runner workspace sync <runner-id> --path <local-worktree>
homeboy runner workspace sync <runner-id> --path <local-worktree> --mode snapshot
homeboy runner workspace sync <runner-id> --path <local-worktree> --mode gitworkspace sync materializes a controller-side worktree under the runner’s configured workspace_root so runner execution can run against an explicit remote path while Git operations and canonical edits stay local.
Modes:
snapshotcopies the current local tree, including dirty edits, through a tar stream from the controller. Use this for private or proxy-dependent sources because the runner does not need repository access.gitrequires a clean local tree, then clones or refreshesremote.origin.urlon the runner and checks out localHEADdetached. Use this only when the runner is allowed to fetch the remote directly.
Private/proxied sources:
- Private or proxy-dependent source access stays on the controller machine.
- Materialize those sources with
homeboy runner workspace sync <runner-id> --path <local-worktree> --mode snapshot. - Use the returned
remote_pathfor downstreamrunner exec --cwdor job inputs. - Runner-side Git fetches for configured private/proxied hosts are refused with an actionable diagnostic. The default host list includes
github.a8c.com; override withHOMEBOY_PRIVATE_PROXIED_SOURCE_HOSTSonly when a runner is explicitly allowed to fetch those sources.
Safety rules:
- The remote path is deterministic and lives under
<workspace_root>/_lab_workspaces/. - Snapshot sync excludes dependency directories, build outputs, caches,
.git, and common secret file patterns such as.env*,*.pem, and*.key. - Runner policy can add project-specific generated-state patterns with
snapshot_excludes; configured patterns are merged with the default snapshot safety excludes and affect snapshot hashing, stats, and materialization. - Output includes
local_path,remote_path,sync_mode,snapshot_identity, and snapshotfiles/byteswhen available. - The runner workspace is execution-only; this command does not push branches, commit, or make the runner authoritative for source changes.
workspace apply
homeboy runner workspace apply <runner-apply.json>
homeboy runner workspace apply <runner-apply.json> --forceworkspace apply brings a runner-generated fix artifact back to the local source worktree recorded in the artifact’s source_snapshot.local_path. It is local-only: it does not commit, push, or make the runner canonical. Reviewability stays in normal local Git via git status and git diff.
Safety rules:
- The artifact must identify the local source worktree through
source_snapshot.local_path. - Homeboy recalculates the current local
source_snapshot.snapshot_hashbefore applying. - If the local source worktree drifted since the Lab snapshot, apply is refused unless
--forceis explicit. - Unified diffs are checked with
git apply --checkbefore mutation, so conflicts do not partially apply. - Delta paths must be relative and stay inside the source worktree.
- Output includes
apply_status,modified_files,expected_snapshot_hash, andcurrent_snapshot_hash.
Temporary Wave 4 adapter contract, until the runner fix-capture contract settles:
{
"source_snapshot": {
"runner_id": "lab-a",
"local_path": "/path/to/project@branch",
"remote_path": "/srv/homeboy/_lab_workspaces/project-abc123",
"git_sha": "...",
"dirty": false,
"sync_mode": "snapshot",
"snapshot_hash": "sha256:...",
"synced_at": "2026-05-16T00:00:00Z",
"sync_excludes": [".git/", "node_modules/"]
},
"patch": {
"format": "unified_diff",
"content": "diff --git a/file.txt b/file.txtn..."
}
}Delta form is also accepted for explicit file replacement/deletion:
{
"source_snapshot": { "...": "..." },
"delta": {
"files": [
{ "path": "src/file.txt", "content_base64": "Li4u" },
{ "path": "obsolete.txt", "delete": true }
]
}
}Runner Shape
SSH runner records are stored on their server as runner capability config under ~/.config/homeboy/servers/<id>.json.
{
"id": "runner-a",
"host": "runner.example.internal",
"user": "runner",
"port": 22,
"runner": {
"workspace_root": "/srv/homeboy/workspaces",
"homeboy_path": "/usr/local/bin/homeboy",
"daemon": false,
"concurrency_limit": 4,
"artifact_policy": "copy",
"env": {},
"resources": {}
}
}Standalone local runner records are still stored under ~/.config/homeboy/runners/.
{
"id": "lab-local",
"kind": "local",
"server_id": null,
"workspace_root": "/srv/homeboy/workspaces",
"homeboy_path": "/usr/local/bin/homeboy",
"daemon": false,
"concurrency_limit": 2,
"artifact_policy": "copy",
"env": {},
"resources": {}
}Rules:
kindislocalorssh.sshrunner IDs are server IDs; a single SSH machine does not need a separate runner ID.concurrency_limit, when set, must be greater than zero.envandresourcesare metadata maps for futureconnect,doctor,exec, and Desktop workflows.
JSON Output
All command output is wrapped in the global JSON envelope described in the JSON output contract. The data payload uses the generic entity CRUD shape:
command: action identifier such asrunner.add,runner.enable,runner.list,runner.show,runner.set,runner.remove,runner.doctor,runner.connect,runner.status,runner.disconnect,runner.exec,runner.workspace.sync, orrunner.workspace.applyid: present for single-runner actionsentity: runner configuration for single-runner read/write actionsentities: list forlistupdated_fields: list of updated field names for writesdeleted: list of removed runner IDs