Self-hosted execution runner for IceFold nodes (reverse-connects to an IceFold server, like a self-hosted CI runner).
Project description
icefold-runner
A self-hosted execution runner for IceFold nodes — like a GitHub self-hosted CI runner. You start it on your own machine; it reverse-connects to an IceFold server (so it works behind NAT with no inbound ports, no public IP, no tunnel), receives node-execution jobs, runs them locally, and streams results back.
It is the place where your uploaded node code runs — on your hardware, with
full subprocess / ffmpeg / GPU / any-dependency access — instead of inside
the server's restricted sandbox.
How it works
your machine (private, behind NAT) IceFold server (public)
┌──────────────────────────────────┐ reverse WSS ┌───────────────────────────┐
│ icefold-runner │ ───────────► │ /v1/ws/worker?token │
│ • dials out, token auth │ node_exec ◄─│ routes node runs (per user) │
│ • reconnect + keepalive │ node_done ─►│ │
│ • bundle runner: │ │ │
│ GET /v1/bundles/<hash> │ HTTP pull │ /upload /download │
│ import bundle + preflight deps │ ◄──────────► │ /v1/workers/output │
│ await __icefold_run__ │ │ │
└──────────────────────────────────┘ HTTP push └───────────────────────────┘
- Control plane rides the reverse WebSocket (
node_exec/cancel→node_status/node_done/missing_dep), JSON frames XOR-obfuscated with the token (TLS still does the real protection). Eachnode_execframe only carries abundle_hashand a single already-sliced variant — no source. - Bulk media + bundles ride plain HTTP: the runner GETs inputs from the
server's
/upload&/downloadand node bundles from/v1/bundles/<hash>(sha256-addressed, cached locally asrunner_work_dir/bundles/<hash>.py, re-hashed on every download), runs the bundle, POSTs products back to/v1/workers/output(which returns server-canonical paths). - The runner ships no node implementations and never compiles user source.
The IceFold server renders every node (your custom ones and the platform's
built-in ones) into a self-contained
.pybundle, withpython_deps/binary_depsdeclared in the bundle header. The runner imports the bundle, pre-flights the deps (sending back a structuredmissing_depreply with platform-aware install hints if anything is absent), and awaits__icefold_run__(inputs, ctx_dict). So when the server adds or upgrades nodes, you never have to upgrade the runner. - Variant planning / dimension & provider resolution all stay on the server; each job is a single already-sliced leaf call.
Install
Requires Python ≥ 3.11, ffmpeg/ffprobe on PATH (for media nodes), and
whatever third-party packages your custom nodes import.
pip install icefold-runner # pulls in icefold-sdk
From source:
git clone <this-repo> icefold-runner
cd icefold-runner
python -m venv .venv && . .venv/bin/activate
pip install -e .
Run
Generate a token in the IceFold app (Nodes ▸ Connect a runner), then:
icefold-runner --token <your-token>
That's it — the token (GitHub-CI style) encodes + signs your IceFold user id, so there's no server URL or user id to pass. The server is built in.
Every flag also reads an env var (see .env.example):
| flag | env | meaning |
|---|---|---|
--token |
ICEFOLD_RUNNER_TOKEN |
runner token from the IceFold app |
--runner-id |
ICEFOLD_RUNNER_ID |
stable id (default: hostname) |
--work-dir |
ICEFOLD_RUNNER_DIR |
scratch for staged inputs + products |
The runner honors standard proxy env vars (HTTPS_PROXY, …) for reaching the
server. It reconnects automatically with backoff; an auth rejection is fatal.
Self-hosting / dev: point the runner at a different server with the
ICEFOLD_RUNNER_SERVERenv var (e.g.ws://127.0.0.1:7000).
Run as a service (systemd)
# /etc/systemd/system/icefold-runner.service
[Unit]
Description=IceFold runner
After=network-online.target
[Service]
EnvironmentFile=/etc/icefold-runner.env
ExecStart=/opt/icefold-runner/.venv/bin/icefold-runner
Restart=always
RestartSec=5
[Install]
WantedBy=multi-user.target
Layout
icefold_runner/ the runner agent (connection, file staging, bundle exec)
client.py reverse-WS client: dial / auth / reconnect / keepalive
runner.py fetch /v1/bundles/<hash>, preflight deps, await __icefold_run__
__main__.py CLI entrypoint (icefold-runner)
The runner imports the bundle on demand; the bundle is self-contained and
already inlines whatever it needs (the author's function body, the
Inputs / Output dataclasses, and a minimal NodeContext shim). The only
runtime dependency on icefold-sdk is the wire protocol + a small helper kit
(get_file_id / run_blocking / write_text), used by the runner agent
itself, not by node code.
Security model
- Node code runs unsandboxed here — it's your machine, your risk. That's the point: code the server sandbox forbids (subprocess/ffmpeg/native deps) runs on the runner instead. The runner downloads each bundle from the server and executes it; it verifies the bundle's sha256 matches the requested hash, but the bundle itself is whatever the server you authenticated to sends. Only point a runner at a server you trust.
- The runner only talks to the one server you point it at, authenticated by the shared token; it pulls input files and pushes products over HTTP to that host.
Self-check
A no-network sanity check that icefold is importable and the bundle execution
path (fetch + import + run __icefold_run__) works against a locally-rendered
bundle:
python selfcheck.py
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file icefold_runner-0.1.0.tar.gz.
File metadata
- Download URL: icefold_runner-0.1.0.tar.gz
- Upload date:
- Size: 16.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
272184c3e95d1074d64ed71148b2f0ce3f86003b8d01b7467e3e109990f94799
|
|
| MD5 |
d58192338d5330c2828716b2fcbb1a8b
|
|
| BLAKE2b-256 |
1f014b12a57829dff8d6df741e43cf36d9a11162d918bcee78a77deb607e463d
|
File details
Details for the file icefold_runner-0.1.0-py3-none-any.whl.
File metadata
- Download URL: icefold_runner-0.1.0-py3-none-any.whl
- Upload date:
- Size: 15.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ede79b01123d657d04c5fbfc70b3b7b242f15cd86ddcce6a540c72c6c33b4571
|
|
| MD5 |
37cb1ff1884eac1da452864fac74824e
|
|
| BLAKE2b-256 |
6999a554773338d1464d8ba1c16ade00ae7db277fcbce087e80e2be48b210033
|