The universal remote for your Linux multimedia server.
Part of the odio project — full documentation.
odio is an ultra-lightweight Go daemon that exposes a single clean REST API over your Linux user session's D-Bus: MPRIS players (Spotify, VLC, Firefox, MPD, Kodi), PulseAudio/PipeWire, systemd user services, and power management. No root. No hacks. Just Linux primitives.
Building a Linux multimedia setup is easy. Integrating it cleanly into Home Assistant always felt hacky, scattered integrations, SSH scripts, and fragile glue.
Tested on Fedora 43 Gnome, Debian 13 KDE, Raspbian 13, Openmediavault 8 Raspberry Pi B through Pi 5. Works without any system tweak.
# 1. Install
curl -fsSL https://apt.odio.love/key.gpg | sudo gpg --dearmor -o /usr/share/keyrings/odio.gpg
echo "deb [signed-by=/usr/share/keyrings/odio.gpg] https://apt.odio.love stable main" | sudo tee /etc/apt/sources.list.d/odio.list
sudo apt update && sudo apt install odio-api
# 2. Start
systemctl --user enable --now odio-api.service
# 3. Test (start any MPRIS player first — Spotify, VLC, MPD…)
curl http://localhost:8018/players
curl http://localhost:8018/audio/server→ See Installation for RPM, Docker, or building from source.
The built-in Odio UI is accessible at:
http://localhost:8018/ui (or http://your-host.local:8018/ui if zeroconf/mDNS is enabled)
It's a 100% local, responsive (mobile + desktop), web interface designed to control your entire Linux multimedia setup from one place: MPRIS players, per-app/global volume, systemd user services, PipeWire/PulseAudio server, and more.
There's also an installable PWA to install on your phone/desktop to easily access your remote and navigate between several instances.
odio-ha is the official Home Assistant integration for Odio.
Install via HACS → Custom Repositories → https://github.com/b0bbywan/odio-ha
What it exposes as HA entities:
media_player— global PulseAudio/PipeWire audio receiver (volume, mute)media_playerper systemd service — power on/off, volume, state tracking (MPD, Kodi, shairport-sync, etc.)- MPRIS players — auto-discovered players with full playback control and metadata (in progress)
Odio becomes the hub that makes all your HA integrations point to the correct machine. MPD service lifecycle managed by Odio, rich playback via HA's existing MPD integration — the two work together.
| Setup | What Odio gives you |
|---|---|
| RPi music server (MPD + shairport-sync) | MPRIS control + restart services from HA |
| HTPC / Kodi | Start/stop Kodi, MPRIS control via odio-ha |
| Firefox kiosk (Netflix, YouTube) | Start/stop fake Netflix and Youtube app, MPRIS control via odio-ha |
| Headless Spotify (spotifyd) | MPRIS playback + service lifecycle |
| Any PulseAudio/PipeWire setup | Per-client and global volume/mute control |
Auto-discovers all MPRIS-compatible players in real time — Spotify, VLC, Firefox, MPD, Kodi, etc. Add a new player and it appears immediately, zero config.
- Full playback control: play, pause, stop, next, previous
- Volume, seek, and position control
- Shuffle and loop mode management
- Real-time state updates via D-Bus signals
- Smart caching with automatic cache invalidation
- Position heartbeat for accurate playback tracking
- Server info and default output
- Global and per-client volume/mute control
- Real-time audio events via native PulseAudio monitoring
- Limited PipeWire support via
pipewire-pulse
Explicit whitelist required — nothing managed unless listed in config.yaml.
- List and monitor whitelisted systemd services (system + user)
- Start, stop, restart, enable, disable user services
- Real-time service state updates via D-Bus signals
- Disabled by default
Odio can act as a Bluetooth audio receiver (A2DP sink) using D-Bus, allowing phones, computers, and other Bluetooth devices to stream audio to it.
Inspired from my own Bluetooth setup since 2020
A few system configuration steps are required to make this work. Since Odio doesn't run as root, it can't do it by itself.
First make sure the user running Odio belongs to bluetooth group
$ groups
pi adm dialout cdrom sudo audio video plugdev games users input render netdev bluetooth gpio i2c spi
# if 'bluetooth' doesn't show in the line above:
$ sudo usermod -a -G bluetooth <username>Some packages are needed to automatically plug PulseAudio or PipeWire to Bluetooth.
Odio doesn't directly support ALSA and never will.
# PulseAudio
$ sudo apt install pulseaudio-module-bluetooth
# PipeWire
$ sudo apt install libspa-0.2-bluetoothTo ensure the device is correctly identified by phones and computers, you must edit /etc/bluetooth/main.conf:
[General]
Name=Odio # Bluetooth name shown during device discovery
Class=0x240428Class of Device (CoD) breakdown:
0x24→ Major Device Class: Audio/Video0x0428→ Minor + services :- Audio Sink
- Loudspeaker
- Rendering device
This configuration makes Odio appear as a standard Bluetooth speaker or audio receiver.
After modifying the configuration file, restart the Bluetooth service:
$ sudo systemctl restart bluetooth
# A new user service should now be running
# It creates an mpris player for each connected device
$ systemctl --user status mpris-proxy.service
● mpris-proxy.service - Bluetooth mpris proxy
Loaded: loaded (/usr/lib/systemd/user/mpris-proxy.service; enabled; preset: enabled)
Active: active (running) since Fri 2026-02-27 13:17:33 CET; 1h 15min ago
Invocation: 4480169b9adb4c239ad81d7345dc1f92
Docs: man:mpris-proxy(1)
Main PID: 674 (mpris-proxy)
Tasks: 1 (limit: 379)
CPU: 791ms
CGroup: /user.slice/user-1000.slice/user@1000.service/app.slice/mpris-proxy.service
└─674 /usr/bin/mpris-proxy
févr. 27 13:17:33 rasponkyold systemd[559]: Started mpris-proxy.service - Bluetooth mpris proxy.Bluetooth is intentionally not left in an automatic or always-on state.
- Power up: Bluetooth is enabled, but the device is not discoverable. You can connect to it if your phone is already paired
- Power Down Default 30min of inactivity (= no connected clients)
- Pairing mode:
The device becomes visible to nearby Bluetooth devices and accepts new pairings.
After a successful pairing (or when the timeout expires), Bluetooth automatically returns to its normal state:
- Not discoverable
- Not pairable
- Audio profile: A2DP (high-quality audio streaming).
This behavior matches how most Bluetooth speakers and audio receivers work.
Odio automatically unblocks soft-blocked Bluetooth rfkill devices on power-up, so a rfkill block bluetooth followed by a power-up via the API will work without manual intervention.
Bonus: You get to control it through /pulseaudio/clients or /players/ and in the UI !
Remote reboot and power-off via the REST API — no SSH needed for day-to-day operations. Disabled by default. Uses org.freedesktop.login1 D-Bus interface.
On desktop systems with a graphical session, logind handles permissions automatically — no extra configuration needed.
On headless systems, you need a polkit rule to allow your user to reboot/power-off via D-Bus. Create /etc/polkit-1/rules.d/10-allow-shutdown.rules:
polkit.addRule(function(action, subject) {
if ((action.id == "org.freedesktop.login1.power-off" ||
action.id == "org.freedesktop.login1.power-off-multiple-sessions" ||
action.id == "org.freedesktop.login1.reboot" ||
action.id == "org.freedesktop.login1.reboot-multiple-sessions") &&
subject.user == "<user>") {
return polkit.Result.YES;
}
});Replace <user> with the username running odio-api.
GET /events streams live state changes to any HTTP client — no polling needed.
Events emitted:
| Event type | Backend | Triggered by |
|---|---|---|
player.updated |
mpris |
Playback state change, volume, metadata |
player.added |
mpris |
New MPRIS player appeared |
player.removed |
mpris |
MPRIS player closed |
player.position |
mpris |
Position tick (periodic, lightweight) |
audio.updated |
audio |
PulseAudio sink-input added or changed (volume, mute, cork) |
audio.removed |
audio |
PulseAudio sink-input removed |
service.updated |
systemd |
systemd unit state change |
bluetooth.updated |
bluetooth |
Bluetooth adapter or device state change (power, pairing, connection) |
power.action |
power |
Reboot or poweroff triggered via the API |
Subscribe to a subset of events using query parameters:
| Parameter | Description | Example |
|---|---|---|
types |
Comma-separated event type names to include | ?types=player.updated,player.added |
backend |
Comma-separated backend names to include | ?backend=mpris,audio |
exclude |
Comma-separated event type names to exclude | ?exclude=player.position |
keepalive |
Keepalive interval in seconds (default 30, min 10, max 120) |
?keepalive=60 |
types and backend can be combined — the union of all matched types is used. Omitting both receives all events. server.info is always delivered and cannot be excluded.
<50msp95 response time,0%CPU on idle — tested on Raspberry Pi B and B+- Localhost binding by default, configurable per network interface
- Zeroconf/mDNS auto-discovery on the LAN (opt-in)
| Architecture | Package | Tested on |
|---|---|---|
| amd64 | deb, rpm | Fedora 43 Gnome, Debian 13 KDE |
| arm64 | deb, rpm | Raspberry Pi 3/4/5 (64-bit) |
| armv7hf | deb, rpm | Raspberry Pi 2/3 (32-bit) |
| armhf (ARMv6) | deb, rpm | Raspberry Pi B / B+ / Zero |
Pre-built packages (amd64, arm64, armv7hf, armhf/ARMv6) and a multi-arch Docker image (amd64, arm64, arm/v7) are available on every build. Docker does not target arm/v6 — Pi B/Zero users should use the armhf package.
- Wayland Remote Control, Authentication, Photos Casting...
curl -fsSL https://apt.odio.love/key.gpg | sudo gpg --dearmor -o /usr/share/keyrings/odio.gpg
echo "deb [signed-by=/usr/share/keyrings/odio.gpg] https://apt.odio.love stable main" | sudo tee /etc/apt/sources.list.d/odio.list
sudo apt update
sudo apt install odio-apiPre-built packages for amd64, arm64, armv7hf, and armhf (ARMv6) are available as artifacts on each build workflow run.
# Debian/Ubuntu/Raspberry Pi OS
sudo dpkg -i odio-api_<version>_amd64.deb
# Fedora/RHEL
sudo rpm -i odio-api-<version>.x86_64.rpmgit clone https://github.com/b0bbywan/go-odio-api.git
cd go-odio-api
task build # builds CSS + Go binary with version from git
./bin/odio-apiCreate ~/.config/systemd/user/odio-api.service:
[Unit]
Description=Dbus api for Odio
Documentation=https://github.com/b0bbywan/go-odio-api
Wants=sound.target
After=sound.target
Wants=network-online.target
After=network-online.target
[Service]
ExecStart=/usr/bin/odio-api
Restart=always
RestartSec=12
TimeoutSec=30
[Install]
WantedBy=default.targetsystemctl --user daemon-reload
systemctl --user enable odio-api.service
systemctl --user start odio-api.serviceHeadless systems: Enable lingering so the user session (PulseAudio/PipeWire, D-Bus, XDG_RUNTIME_DIR) survives without an active login:
sudo loginctl enable-linger <username>A pre-built multi-arch image is available on GHCR (amd64, arm64, arm/v7):
ghcr.io/b0bbywan/go-odio-api:latest
# 1. Prepare configuration (bind: all required for Docker)
cp share/config.yaml config.yaml
# Edit config.yaml: set bind: all
# 3. (Optional) Only needed if docker compose config shows wrong paths
cp .env.example .env
# 4. Start
docker compose up -dThe docker-compose.yml reads UID, XDG_RUNTIME_DIR, HOME and DBUS_SESSION_BUS_ADDRESS
directly from your shell environment — no configuration needed for a standard Linux setup.
See .env.example if your shell doesn't export these automatically (e.g. fish).
Environment variables passed to the container:
| Variable | Source | Purpose |
|---|---|---|
XDG_RUNTIME_DIR |
host env → fallback /run/user/$UID |
D-Bus and PulseAudio runtime directory |
DBUS_SESSION_BUS_ADDRESS |
host env → fallback derived from XDG_RUNTIME_DIR |
User D-Bus session socket |
HOME |
host env → fallback /home/odio |
PulseAudio cookie lookup path |
Volumes mounted (all read-only):
| Volume | Purpose |
|---|---|
./config.yaml |
odio configuration |
$XDG_RUNTIME_DIR/bus |
user D-Bus session socket |
$XDG_RUNTIME_DIR/systemd |
user systemd folder (utmp unavailable) |
/run/utmp |
user systemd monitoring (utmp available) |
/var/run/dbus/system_bus_socket |
system D-Bus socket |
$XDG_RUNTIME_DIR/pulse |
PulseAudio socket |
$HOME/.config/pulse/cookie |
PulseAudio cookie |
Note: bind must be set to all in config.yaml for Docker remote access (bridge network). Zeroconf won't work in bridge network mode. Host network mode is strongly discouraged.
To build locally instead:
docker build -t odio-api .
# or simply: task docker:buildThe Docker build is fully self-contained — Tailwind CSS is downloaded and compiled inside the builder stage.
--config <path>— specify a custom YAML configuration file--version— print version and exit--help— show help message
Configuration file locations (in order of precedence):
- Specified with
--config <path> ~/.config/odio-api/config.yaml(user-specific)/etc/odio-api/config.yaml(system-wide)- A default configuration is available in
share/config.yaml
Disabling a backend disables the backend and all its routes.
bind: lo
logLevel: info
api:
enabled: true
port: 8018
ui:
enabled: true
sse:
enabled: true
cors:
origins: ["https://odio-pwa.vercel.app"] # default for PWA
# origins: ["https://app.example.com"] # specific originsbind: lo # loopback only (default)
# bind: enp2s0 # single LAN interface
# bind: [lo, enp2s0] # loopback + LAN (required for UI access from the network)
# bind: [lo, enp2s0, wlan0] # loopback + ethernet + wifi
# bind: all # all interfaces — 0.0.0.0 (Docker, remote access)Note: The built-in web UI requires lo to be in the bind list. If lo is absent, the UI is automatically disabled.
systemd:
enabled: true
timeout: 90s # fsnotify stable state timeout (default: 90s)
system:
- bluetooth.service
- upmpdcli.service
user:
- pipewire-pulse.service
- pulseaudio.service
- mpd.service # see [1]
- shairport-sync.service # see [2]
- snapclient.service # incompatible with mpris
- spotifyd.service # see [3]
- firefox-kiosk@netflix.com.service # default support for mpris
- firefox-kiosk@youtube.com.service # default support for mpris
- firefox-kiosk@my.home-assistant.io.service
- kodi.service # see [4]
- vlc.service # default support for mpris
- plex.service # see [5][1] Install mpd-mpris or mpDris2 for MPRIS support
[2] Check my article on Medium: Shairport Sync/Airplay with PulseAudio and MPRIS support
[3] Default on desktop; on headless, your spotifyd version must be built with MPRIS support
[4] Install Kodi Add-on: MPRIS D-Bus interface
[5] Maybe supported, untested
bluetooth:
enabled: true
timeout: 5s
pairingTimeout: 60s
idleTimeout: 30m # 0 for no autopoweroffpower:
enabled: true
capabilities:
poweroff: true
reboot: truepulseaudio:
enabled: true
serve_cookie: true # exposes GET /audio/cookie for network audio clientsbind: eno1
zeroconf:
enabled: trueOdio advertises itself via mDNS. Look for _http._tcp.local. → instance odio-api. Disabled on lo binding.
- Localhost binding by default — prevents accidental network exposure
- Systemd disabled by default — service control must be explicitly enabled and configured
- Read-only Docker mounts — all volume mounts are read-only in the provided
docker-compose.yml - Zeroconf opt-in — must be enabled, then mDNS adapts to
bind: disabled onlo, enabled on specific interfaces, orallinterfaces withoutlo
GET /server # {"hostname":"","os_platform":"","os_version":"","api_sw":"","api_version":"","backends":{"mpris":true,"pulseaudio":true,"systemd":false,"zeroconf":false}}
GET /players # List all media players
GET /players/{player}/cover # Cover art (serves file:// or redirects http(s)://)
POST /players/{player}/play # Play
POST /players/{player}/pause # Pause
POST /players/{player}/play_pause # Toggle play/pause
POST /players/{player}/stop # Stop
POST /players/{player}/next # Next track
POST /players/{player}/previous # Previous track
POST /players/{player}/seek # Seek (body: {"offset": 1000000})
POST /players/{player}/position # Set position (body: {"track_id": "...", "position": 0})
POST /players/{player}/volume # Set volume (body: {"volume": 0.5})
POST /players/{player}/loop # Set loop status (body: {"loop": "None|Track|Playlist"})
POST /players/{player}/shuffle # Set shuffle (body: {"shuffle": true})
GET /audio # Combined: server info, outputs, clients
GET /audio/server # Get server info
POST /audio/server/mute # Mute/unmute default output
POST /audio/server/volume # Set default output volume (body: {"volume": 0.5})
GET /audio/clients # List audio clients (sink-inputs)
POST /audio/clients/{sink}/mute # Mute/unmute client
POST /audio/clients/{sink}/volume # Set client volume (body: {"volume": 0.5})
GET /audio/outputs # List all audio outputs (sinks)
POST /audio/outputs/{output}/default # Set default output
POST /audio/outputs/{output}/mute # Mute/unmute output
POST /audio/outputs/{output}/volume # Set output volume (body: {"volume": 0.5})
GET /audio/cookie # Download PulseAudio cookie file (requires pulseaudio.serve_cookie: true)
GET /services # List all monitored services
POST /services/{scope}/{unit}/start # Start service (scope: system|user)
POST /services/{scope}/{unit}/stop # Stop service (scope: system|user)
POST /services/{scope}/{unit}/restart # Restart service
POST /services/{scope}/{unit}/enable # Enable service (scope: system|user)
POST /services/{scope}/{unit}/disable # Disable service
GET /bluetooth # Get Bluetooth status (powered, pairing mode state)
POST /bluetooth/power_up # Turns Bluetooth on and makes the device ready to connect to already paired devices.
POST /bluetooth/power_down # Turns Bluetooth off and disconnects any active Bluetooth connections.
POST /bluetooth/pairing_mode # Enables Bluetooth pairing mode for 60s (configurable).
# Returns to non-discoverable state after timeout or successful pairing.
GET /power/ # Power capabilities {"reboot": true, "power_off": false}
POST /power/power_off # Poweroff (403 if not declared in capabilities)
POST /power/reboot # Reboot (403 if not declared in capabilities)
GET /events # All events (text/event-stream)
GET /events?backend=mpris # Only MPRIS player events
GET /events?backend=mpris,audio # Player + audio events
GET /events?backend=bluetooth # Only Bluetooth state changes
GET /events?backend=power # Only power actions (reboot/poweroff)
GET /events?types=player.updated # Specific event types
GET /events?exclude=player.position # All events except position ticks
GET /events?keepalive=60 # Custom keepalive interval (seconds)
GET /events?types=player.updated,service.updated&backend=audio&exclude=player.position # Mixed
# All events
curl -N http://localhost:8018/events
# Only player events
curl -N "http://localhost:8018/events?backend=mpris"
# Only position ticks lightweight on purpose (e.g. to drive a seek bar)
curl -N "http://localhost:8018/events?types=player.position"Expected output:
event: server.info
data: "connected"
event: player.updated
data: {"bus_name":"org.mpris.MediaPlayer2.spotify","identity":"Spotify",...}
event: audio.updated
data: [{"id":42,"name":"Spotify","volume":0.75,"muted":false,...}]
event: audio.removed
data: [{"id":41,"name":"pactl","volume":1,...}]
event: service.updated
data: {"name":"mpd.service","scope":"user","active_state":"active","running":true,...}
event: bluetooth.updated
data: {"powered":true,"discoverable":false,"pairable":false,"pairing_active":false,"known_devices":[{"address":"AA:BB:CC:DD:EE:FF","name":"My Phone","trusted":true,"connected":true}]}
event: power.action
data: {"action":"reboot"}
<!DOCTYPE html>
<html>
<head><title>Odio live events</title></head>
<body>
<pre id="log"></pre>
<script>
const log = document.getElementById('log');
// Subscribe to all events — add ?backend=mpris or ?types=... to filter
const es = new EventSource('http://localhost:8018/events');
['player.updated', 'player.added', 'player.removed', 'player.position',
'audio.updated', 'audio.removed', 'service.updated', 'bluetooth.updated', 'power.action'].forEach(type => {
es.addEventListener(type, e => {
const entry = `[${type}] ${e.data}\n`;
log.textContent = entry + log.textContent;
});
});
es.onerror = () => log.textContent = '[error] connection lost\n' + log.textContent;
</script>
</body>
</html>Save as events.html, open in a browser — events appear live as they happen. No polling, no page refresh needed.
Systemd control is disabled by default and requires an explicit whitelist. Odio mitigates risks with deliberate security design:
- Disabled by default — must explicitly set
systemd.enabled: trueAND configure units. Empty config → auto-disabled even withenabled: true. - Localhost only — API binds to
loby default. Never expose to untrusted networks or the Internet. - User-only mutations — start/stop/restart/enable/disable only work on user D-Bus. System units are strictly read-only, enforced at the application layer regardless of D-Bus or polkit configuration. This protects against misconfigured or compromised D-Bus setups.
- Root forbidden by design — Odio refuses to run as root.
- No preconfigured units — nothing managed unless explicitly listed.
You must knowingly enable this at your own risk. Odio is free software and comes with no warranty.
All multimedia services run as systemd user units, not system-wide daemons. This unlocks a single, unified D-Bus session bus where PulseAudio/PipeWire, MPRIS players, and user systemd units all coexist. Odio listens to that bus and exposes everything via HTTP. Add a new MPRIS player — it appears immediately, zero code or config change.
- MPRIS Backend — D-Bus communication with media players, smart caching, real-time D-Bus signal updates
- PulseAudio Backend — native PulseAudio protocol (pure Go, no libpulse), real-time event monitoring
- Systemd Backend — D-Bus with filesystem monitoring fallback (
/run/user/{uid}/systemd/units) - Power Backend —
org.freedesktop.login1D-Bus interface
- Caching reduces D-Bus calls by ~90%
- D-Bus signal-based updates instead of polling
- Batch property retrieval
- Automatic heartbeat management for position tracking
- Connection pooling and timeout handling
- Go 1.24 or higher
go test ./...
go test -cover ./...
go test ./backend/mpris/...
go test ./backend/pulseaudio/...
go test ./backend/systemd/...The project uses Task for build automation.
# Install Task (once)
go install github.com/go-task/task/v3/cmd/task@latest
# Build for the current host (CSS + Go binary, version from git)
task build
# Cross-compile for all supported architectures (output: dist/)
task build:all-arch
# Individual targets
task build:linux-amd64 # x86_64
task build:linux-arm64 # RPi 3/4/5 64-bit
task build:linux-armv7hf # RPi 2/3 32-bit (ARMv7)
task build:linux-armhf # RPi B/B+/Zero (ARMv6, RPi OS armhf)
# CSS only
task css # Ensure CSS is available (compile or download from CDN)
task css-local # Compile locally (requires Tailwind CLI)
task css:watch # Watch mode for developmentNote: task build injects the version via -ldflags from git describe. The version is visible via ./bin/odio-api --version.
The UI uses Tailwind CSS with an intelligent multi-architecture build strategy:
- Development (x64/arm64/armv7) —
task buildcompiles CSS locally using Tailwind CLI - Legacy ARM (ARMv6 — Raspberry Pi B/B+) —
task builddownloads pre-built CSS from CDN (https://bobbywan.me/odio-css/)
Tailwind CLI doesn't provide ARMv6 binaries. The CSS is architecture-independent, so it's compiled on x64 and distributed via CDN.
CDN structure:
https://bobbywan.me/odio-css/
main/abc1234.css # commit-specific
main/latest.css # latest for branch
tags/v0.6.0.css # release tags (never cleaned)
CSS files are not committed to the repository.
Packages are built with nfpm via Task.
# Install nfpm (once)
go install github.com/goreleaser/nfpm/v2/cmd/nfpm@latest
# Build all packages for all architectures (output: dist/)
task package:all
# Individual targets
task package:deb:linux-amd64 # .deb amd64
task package:deb:linux-arm64 # .deb arm64
task package:deb:linux-armv7hf # .deb armv7hf
task package:deb:linux-armhf # .deb armhf (ARMv6, RPi OS)
task package:rpm:linux-amd64 # .rpm x86_64
task package:rpm:linux-arm64 # .rpm aarch64
task package:rpm:linux-armv7hf # .rpm armv7hl
task package:rpm:linux-armhf # .rpm armv6hl- spf13/viper — configuration
- godbus/dbus — D-Bus bindings
- coreos/go-systemd — systemd D-Bus bindings
- the-jonsey/pulseaudio — pure-Go PulseAudio native protocol (no libpulse)
- grandcat/zeroconf — mDNS / DNS-SD
- HTMX
- TailwindCSS
Odio was first pushed on January 25, 2026. It's early stage. v0.4 works out of the box, but there's a long road ahead. Expect bugs.
Does it work on your setup? What breaks? What's missing?
Try it. Tell me what works and what doesn't. Show me your setup. If you want to contribute code, even better. Go is a great language for this use case.
- Fork the repository
- Create your feature branch (
git checkout -b feature/amazing-feature) - Commit your changes (
git commit -m 'Add some amazing feature') - Push to the branch (
git push origin feature/amazing-feature) - Open a Pull Request
For issues and questions: GitHub repository
BSD 2-Clause License — see the LICENSE file for details.