`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. Both storage shapes share the same runner capability contract: workspace root, settings, environment, secret references, resources, and policy.
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, fuzz 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.
Commands launched through non-local homeboy runner exec run with HOMEBOY_RUNNER_HOSTED_EXEC=1. That marker is Homeboy’s first-class runner-side dispatch signal: nested runner commands such as homeboy agent-task cook are allowed to pass the non-interactive resource preflight without adding --force-hot, because the work is already intentionally hosted on the selected runner.
Lab offload support is intentionally command-specific:
| Command | Auto offload | Explicit --runner | Decision |
|---|---|---|---|
agent-task cook / agent-task run-plan | Yes | Yes | Portable agent-task execution when the command has deterministic gates where required. |
agent-task controller from-spec --resume / agent-task controller materialize / agent-task controller resume | No | Yes | Runner-hosted controller lifecycle work. |
agent-task retry --run | Yes | Yes | Retry plus execution follows the portable agent-task run path. |
agent-task run / run-next / status / logs / artifacts / review / providers | No | Yes | Runner-resident inspection and queue interaction. |
agent-task auth status | No | Yes | Runner-resident auth diagnostics. |
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. |
build | No | No | Local quality gate; extension-owned build commands are not represented in the portable Lab contract yet. |
fuzz run / default fuzz run | Yes | Yes | Safe single-workspace replay with fuzz-specific case/result artifacts. |
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. |
refactor --from audit / refactor --all | No | Yes | Source refactor runs with command-owned mutation flags and runner-side workspace controls. |
review | Yes | Yes | Scoped audit/lint/test umbrella using the same portable quality gates. |
rig check | No | Yes | Runner-side rig validation without starting local services. |
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. |
tunnel preview-consumer run | No | Yes | Runner-hosted preview consumer execution. |
tunnel service expose / tunnel service start | No | Yes | Runner-hosted private service tunnel lifecycle. |
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 unzip
homeboy runner doctor <runner-id> --scope lab-offload
homeboy runner doctor <runner-id> --scope lab-offload --repairDiagnoses 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.
Use --scope lab-offload before serious Lab evidence runs. It adds checks for
the configured runner Homeboy command, bare homeboy PATH resolution, preferred
runner binaries, connected daemon exec readiness, and Sample Runtime runner-path
freshness signals. When Homeboy can identify a safe exact recovery command, the
check includes that remediation instead of leaving the operator to infer it from
logs.
Use --scope lab-offload --repair for the narrow self-healing path. Today this
reconnects a failing direct Lab runner daemon and reruns the daemon exec probe.
It does not upgrade binaries, rewrite runner paths, or refresh Sample Runtime caches;
those remain explicit operator actions because they can be expensive or depend
on environment-specific paths.
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.
Runner-managed dependency sources
Portable Lab/evidence runs should declare runner dependency sources instead of
smuggling local checkout paths through workflow-specific environment variables.
Homeboy’s generic contract is the extension/component setting
validation_dependencies: each entry is a component id or explicit checkout
path that the runner workspace sync treats as managed source input.
For each declared dependency, homeboy runner workspace sync:
- resolves a sibling checkout, registered component, or deterministic clone;
- rejects dirty, stale, divergent, missing-upstream, or ambiguous Git state;
- runs the dependency install/build lifecycle in a prepared copy;
- materializes the prepared dependency beside the primary runner workspace; and
The JSON output includes
validation_dependencies, with each dependency’sid,role, controllerlocal_path, runnerremote_path, andevidence_path. Bench, trace, eval, and provider layers can consume that generic output to populate their own path settings without knowing Lab filesystem layouts or dependency-specific environment variable names.
Example manifest fragment:
{
"extensions": {
"provider-id": {
"settings": {
"validation_dependencies": ["runtime-component"]
}
}
}
}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.
Broker authentication and pairing
Every /runner/* broker route requires an authenticated, scoped bearer token
(#2990). The trust model is:
- Pairing. On the broker host, mint a credential with
homeboy runner broker pair <id> --runner-id <runner-id> [--work] [--submit]. Pairing prints a one-time bearer token and stores only its SHA-256 hash in~/.config/homeboy/broker_auth.json(0600). The plaintext token is shown once and never re-displayed or logged. - Scopes.
--workauthorizes worker routes (register, claim, events, finish, heartbeat);--submitauthorizes controller job submission and cancellation (POST /runner/jobs,POST /runner/jobs/<id>/cancel). A credential may carry both. When neither flag is given,--workis assumed. - Runner-id binding. Worker routes carry a
runner_id; the presented token must belong to a credential paired to that exactrunner_id, so a paired runner can never claim, progress, or finish another runner’s jobs. - Revocation.
homeboy runner broker revoke <id>disables a credential.homeboy runner broker listshows non-secret credential metadata (never tokens or hashes). - Secure by default. With no credentials configured, the broker rejects all
/runner/*traffic with a structuredbroker.auth_denied(401) error. To keep an existing loopback-only smoke setup working without auth, set"allow_unauthenticated_loopback": trueinbroker_auth.json; this is only honored when the broker is bound to loopback.
Workers send the token via --broker-token <token> or the
HOMEBOY_BROKER_TOKEN environment variable; controllers read their submit token
from HOMEBOY_BROKER_TOKEN. Tokens travel as both X-Homeboy-Broker-Token and
Authorization: Bearer <token> so they survive proxies that strip one form.
Unauthenticated, mismatched-runner, or wrong-scope requests are rejected with a
structured broker.auth_denied error and never expose the stored secret.
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.
Runner lifecycle output is transport-neutral even when the underlying route is a direct daemon tunnel or a reverse broker. JSON consumers should prefer these generic contracts:
RunnerSession: persisted connection/session ownership, including mode, role, controller/broker references, process IDs, version identity, and heartbeat.RunnerJob: active or inspected job state with lifecycle owner, claim lease, command, status, durable run linkage, and artifact refs.RunnerWorkspaceLease: controller-owned local-to-runner workspace materialization with source ref/commit/dirty metadata.RunnerWorkspaceLeaseSet: named workspace leases for jobs that materialize more than one tree.primarynames the default execution workspace; entries inleasesbind names such assource,rig, orruntimeto their ownRunnerWorkspaceLease. Existing single-workspace output keepsworkspace_leasewhile migrations addworkspace_leases.RunnerResult: terminal command result with exit status, stream sizes, mirrored run linkage, and artifact refs.RunnerArtifactRef: stable artifact identity plus path/URL/hash/transport metadata for later retrieval.RunnerMutationArtifacts: transport-independent mutation return shape. When a runner command captures source changes,mutation_artifacts.patch_refpoints at the patch artifact; future file-bundle and operation-log returns usefile_bundle_refandoperation_log_refwithout changing caller logic.RunnerHandoff: the controller-to-runner handoff envelope tying transport, owner, job, workspace lease, and result together.
Compatibility fields such as active_jobs, job, job_id, job_events, and
workspace_lease remain present for existing callers. New integrations should
read active_runner_jobs, runner_job, workspace_leases, runner_result,
mutation_artifacts, artifact_ref, and handoff so direct SSH and reverse
broker implementations stay hidden behind the same lifecycle vocabulary.
{
"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> --broker-token <token>
homeboy runner work <runner-id> --broker-url <url> --project <project-id> --lease-ms 30000
HOMEBOY_BROKER_TOKEN=<token> homeboy runner work <runner-id> --broker-url <url> --loopPass the paired token with --broker-token or the HOMEBOY_BROKER_TOKEN
environment variable. The token is required whenever the broker enforces auth;
omit it only for a loopback-open smoke broker.
Claims 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.
broker
homeboy runner broker pair lab-1 --runner-id homeboy-lab --work
homeboy runner broker pair ctrl-1 --runner-id homeboy-lab --submit
homeboy runner broker revoke lab-1
homeboy runner broker listManage reverse runner broker authentication and pairing on the broker host.
pair mints a one-time scoped bearer token (printed once; only its SHA-256 hash
is stored under ~/.config/homeboy/broker_auth.json). revoke disables a
credential by id, and list reports non-secret credential metadata. See
Broker authentication and pairing for the
full trust model.
job
homeboy runner job logs <runner-id> <job-id>
homeboy runner job logs <runner-id> <job-id> --follow --poll-ms 1000
homeboy runner job cancel <runner-id> <job-id>Inspects or cancels durable runner daemon jobs after runner exec or Lab offload
has submitted work to a connected runner. logs fetches the persisted job plus
its event stream; --follow keeps polling until the job reaches a terminal state
and prints newly observed events as they arrive. Use this when a controller exits
after dispatching runner work and you need to inspect the already-started job.
cancel requests cancellation for a queued or running durable runner daemon job
through the connected runner daemon.
Minimal Homeboy Lab systemd unit:
[Unit]
Description=Homeboy reverse runner worker
After=network-online.target
Wants=network-online.target
[Service]
Type=simple
User=deploy
WorkingDirectory=/home/user
Environment=HOMEBOY_BROKER_TOKEN=<paired-token>
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 worker authenticates with the broker using the paired bearer token. Prefer
delivering it through HOMEBOY_BROKER_TOKEN — via a systemd
EnvironmentFile=/etc/homeboy/broker.env with 0600 permissions rather than an
inline Environment= line — so the secret never lands in systemctl show
output. Pair the credential on the broker host with homeboy runner broker pair
(see Broker authentication and pairing).
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.
Status output also carries the former Lab readiness handoff. extra includes
preferred_lab_runner, selected_lab_runner.runner_homeboy, and
managed_followups with copyable commands for recent run discovery, evidence
inspection, runner doctor --scope lab-offload, runner environment inspection,
daemon refresh, binary upgrade, managed runner exec, and workspace sync. Use
this output before assuming a runner has picked up a merged Homeboy CLI fix.
Runner-side extension updates use the generic runner/extension primitives instead of a Lab-specific sync command:
homeboy runner exec <runner-id> -- homeboy extension install <source> --id <extension-id> --ref <ref> --replace
homeboy runner exec <runner-id> -- homeboy extension refresh <source> --id <extension-id> --ref <ref>Status output is also the operator handoff for active runner jobs. When the
connected daemon or reverse broker can report active work, each active job
includes lifecycle timing fields such as updated_at_ms, heartbeat_age_ms,
claim_id, claimed_by_runner_id, claimed_at_ms, claim_expires_at_ms, and
claim_expires_in_ms. The top-level operator_commands array gives copyable
next actions for the current session, including job log following, cancellation,
broker claim reconciliation for reverse runners, and artifact lookup/fetch
commands when the status payload has enough context.
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.
- Mutation-capable executions expose returned source changes through
mutation_artifacts.patch_refwhen a patch artifact is available. Direct daemon and reverse broker results use the same field; callers should not branch on transport mode to find returned mutations.
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.example.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. - Git sync refuses to overwrite an existing dirty runner-side checkout by default. Use
--allow-dirty-lab-workspaceonly for noisy investigation where discarding runner-side changes is intentional; Lab metadata records that override. - 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