An Incus/LXC-based platform for self-hosting persistent AI coding agent sandboxes with Caddy reverse proxy and direct SSH routing to containers (suitable for VS Code remote ssh).
Create and host your vibe-coded apps on a single VPS/server.
Vibebin TUI Manager - Container detail view with quick actions:
AI Tools Admin Web App - Toggle AI coding tools on/off:
AI Tools Admin Web App - One-click update for all tools:
This project is 99.9% vibe-coded on the exe.dev platform using their Shelley Web AI Coding Agent and Claude Opus 4.5. Take that as you will.
With that said, I am a huge proponent of the exe.dev platform, and if you can, you should definitely try it out and use their service. The love and care for that project/service is extremely evident... AND it is incredibly awesome (and I think it's in its infancy stages, so should only get better).
This is a very new project. Feel free to test and experiment but it's likely to have bugs and may not be ready for production. Use at your own risk.
This project provides the infrastructure to self-host your own AI coding environment on virtually any Linux server -- a VPS, cloud VM (EC2, GCP, Azure), or dedicated hardware. Because it uses Incus/LXC (container-based virtualization rather than nested VMs), it runs efficiently on KVM, VMware, Xen, Hyper-V, and most other hypervisors.
Each container is a fully persistent Linux sandbox running Ubuntu 24.04 LTS (Noble) or Debian 13 (Trixie), with:
- opencode, nanocode, and openhands - AI coding agents with terminal and web UI interfaces
- AI coding web UI accessible via HTTPS at
code.yourdomain.com(Basic Auth protected) - AI Tools Admin web app at
admin.code.yourdomain.comfor managing AI coding tools - Your app/site accessible via HTTPS at
yourdomain.com - SSH access for direct terminal access to your sandbox (VS Code Remote SSH compatible)
- Persistent filesystem that survives container restarts
- Pre-installed development tools: Docker, Go, Node.js, Bun, Deno, uv
- AI-assisted development: Use opencode/nanocode/openhands as your AI pair programmer with full system access
- Vibe coding: Spin up isolated sandboxes for experimental projects
- App/site hosting: Deploy and iterate on web applications
- Learning environments: Safe, isolated Linux environments for experimentation
- CI/CD sandboxes: Temporary or persistent build environments
| Component | Purpose |
|---|---|
| Incus (LXC) | Container runtime - lightweight, persistent Linux containers |
| Caddy | Reverse proxy with automatic HTTPS (Let's Encrypt) |
| SSHPiper | SSH routing - access any container via ssh -p 2222 container-name@host |
| SQLite | Local database for container metadata (/var/lib/vibebin/containers.db) |
| Ubuntu/Debian | Native Incus images (user choice during creation) |
| opencode | Open source AI coding agent with terminal and web UI |
| nanocode | NanoGPT-powered AI coding agent (fork of opencode) |
| openhands | Full-featured AI coding agent from All-Hands-AI |
This project installs opencode, nanocode, and openhands in each container:
OpenCode is a 100% open source AI coding agent that works with multiple LLM providers:
- Anthropic (Claude)
- OpenAI (GPT)
- Google (Gemini)
- OpenCode Zen
- Local models via custom endpoints
NanoCode is a fork of OpenCode configured to work with NanoGPT as the default provider. It includes:
- Automatic model loading from NanoGPT API
- Built-in NanoGPT MCP server
- Interleaved thinking for reasoning models
Note: The nanocode web UI requires LLM configuration before it will work. SSH into the container and run nanocode (CLI mode) first to configure your LLM provider/API keys.
OpenHands (formerly OpenDevin) is a powerful AI coding agent with:
- Full agentic capabilities with file editing, terminal access, and web browsing
- Support for multiple LLM providers (Anthropic, OpenAI, etc.)
- Persistent workspace and history
- Best used with Python 3.12 (installed automatically via uv)
All tools support terminal and web UI modes. Configure your LLM credentials on first run.
vibebin- Interactive TUI for container managementvibebin_sync_daemon- Background service for config synchronizationcontainers.db- SQLite database storing container metadata (at/var/lib/vibebin/containers.db)
- Fresh/minimal Linux installation: Ubuntu 22.04+ or Debian 12+ (amd64 or arm64)
- VPS or VM: Works on most virtualization platforms (KVM, VMware, Xen, EC2, GCP, Azure, etc.)
- Memory: Minimum 4GB RAM; 8GB+ recommended for best performance
- Go 1.21+: Required to build the tools (see Quick Start for installation)
- A domain name with DNS you control
- A regular user with sudo access (avoid running as root)
Any KVM-based VPS will work well with vibebin. Here are some recommended options:
- HostBRR Epyc Turin VPS - AMD Epyc Turin processors, great price/performance
- HostBRR Threadripper Performance VPS - High-performance Threadripper line
Disclosure: The HostBRR links above are affiliate links.
Before installing, ensure your host SSH is properly secured:
# In /etc/ssh/sshd_config, verify these settings:
PermitRootLogin no
PasswordAuthentication noAll administrative tasks should be performed as a regular user with sudo privileges, not as root directly.
# Install wget, curl & git
sudo apt update && sudo apt install wget curl git
# Ubuntu/Debian - install from official Go downloads
wget https://go.dev/dl/go1.23.5.linux-amd64.tar.gz
sudo rm -rf /usr/local/go && sudo tar -C /usr/local -xzf go1.23.5.linux-amd64.tar.gz
echo 'export PATH=$PATH:/usr/local/go/bin' >> ~/.bashrc
source ~/.bashrc
# Verify installation
go versionNote: For ARM64 systems, use
go1.23.5.linux-arm64.tar.gzinstead. Check https://go.dev/dl/ for the latest version.
Use the install script for a one-liner install or upgrade:
curl -fsSL https://raw.githubusercontent.com/jgbrwn/vibebin/main/install-upgrade.sh | bashThe script automatically:
- Clones and builds the project
- Installs binaries to
/usr/local/bin/ - Detects fresh install vs upgrade
- Handles the
vibebin-syncservice accordingly
If you prefer to build manually:
# Clone and build
git clone https://github.com/jgbrwn/vibebin.git
cd vibebin
go build -o vibebin vibebin.go
go build -o vibebin_sync_daemon vibebin_sync_daemon.go
# Install binaries
sudo cp vibebin vibebin_sync_daemon /usr/local/bin/# This auto-installs Incus, Caddy, and SSHPiper
sudo vibebinSee the SSHPiper Manual Setup section below.
sudo vibebinThe creation wizard will guide you through:
- Enter domain name
- Select base image (Ubuntu or Debian)
- Configure DNS (optional auto-creation via Cloudflare/deSEC)
- Set app port
- Provide SSH public key
- Set basic auth credentials for web UI protection
The first run automatically installs:
- Incus 6.20+ from Zabbly stable repository
- Caddy web server with automatic HTTPS
- SSHPiper SSH routing proxy
During container creation, the following is automatically installed:
- Docker (via official get.docker.com script)
- Go (latest version, architecture auto-detected)
- Node.js (latest LTS via NodeSource)
- Bun (JavaScript runtime and package manager)
- Deno (JavaScript/TypeScript runtime)
- uv (Python package manager from Astral)
- opencode (open source AI coding agent)
- nanocode (NanoGPT-powered AI coding agent)
- openhands (full-featured AI coding agent via uv with Python 3.12)
- Project directory (
~/projectsfor AI coding tool workspaces) - Custom MOTD (shows container info, URLs, and tool versions on SSH login)
Before creating containers, verify SSHPiper is running:
# Check SSHPiper status (should be active)
sudo systemctl status sshpiperd
# If not running, start it
sudo systemctl enable --now sshpiperdSSHPiper listens on port 2222 for container SSH access. Host SSH remains on port 22.
Ensure your host SSH is properly secured in /etc/ssh/sshd_config:
PermitRootLogin no
PasswordAuthentication nosudo vibebinList View:
n- Create new containerEnter- View container detailsd- Delete containeru- Show untracked containers (import existing)D- Manage DNS API tokens (Cloudflare/deSEC)i- View Incus logsl- View sync daemon logsq- Quit
Detail View:
s- Start/Stop containerr- Restart containerp- Change app porta- Change auth credentialsS- Snapshot managementu- Update AI coding tools (opencode/nanocode/openhands)Esc- Back to list
Snapshot View:
n- Create new snapshotEnter/r- Restore selected snapshotd- Delete selected snapshot↑/↓orj/k- Navigate snapshotsEsc- Back to container details
- Native Incus Images: Choose Ubuntu 24.04 LTS or Debian 13
- Persistent Sandboxes: Full filesystem persistence across restarts
- Boot Behavior: Containers respect their last state on host reboot
- Resource Monitoring: Live CPU and memory usage in TUI
- Untracked Import: Detect and adopt existing Incus containers
- Snapshots: Create, restore, and delete container snapshots
- Automatic HTTPS: Caddy handles Let's Encrypt certificates
- Reverse Proxy: Each container gets
https://domain.com→ Container's app (port 8000, configurable) - SSH Routing: SSHPiper on port 2222 enables
ssh -p 2222 container-name@hostaccess - Auto DNS: Cloudflare and deSEC API integration (tokens saved securely for reuse)
- opencode: Open source, supports multiple LLM providers
- nanocode: NanoGPT-optimized fork with built-in features
- openhands: Full-featured AI agent with file editing, terminal, and web browsing
- Easy Configuration: All tools prompt for API keys on first run
- Web UI Access: Start any tool in serve mode on port 9999
After creating a container, SSH in and run any of the tools. A project directory is pre-created at ~/projects for your coding workspaces.
# SSH to your container
ssh -p 2222 container-name@host.example.com
# cd to projects directory first
cd ~/projects
# Run opencode
opencode
# Or run nanocode
nanocode
# Or run openhands CLI
openhandsAll tools will prompt you to configure your LLM provider and API key on first run.
To access the AI coding agent via web browser:
# SSH to your container
ssh -p 2222 container-name@host.example.comopencode (cd to project directory first):
cd ~/projects
screen -S code
opencode serve --port 9999 --hostname 0.0.0.0nanocode (cd to project directory first):
cd ~/projects
screen -S code
nanocode serve --port 9999 --hostname 0.0.0.0Note: nanocode web UI requires LLM configuration first. Run
nanocode(CLI) to configure your provider/API keys before starting the web UI.
openhands (uses Docker, mounts ~/projects as workspace):
screen -S code
docker run -it --rm --pull=always \
-v /var/run/docker.sock:/var/run/docker.sock \
-v ~/.openhands:/.openhands \
-p 9999:3000 \
--add-host host.docker.internal:host-gateway \
-e SANDBOX_VOLUMES=~/projects:/workspace:rw \
-e SANDBOX_USER_ID=$(id -u) \
--name openhands-app \
docker.all-hands.dev/all-hands-ai/openhands:latestThen access via https://code.yourdomain.com (protected by Basic Auth credentials you set during container creation).
Press Ctrl+A, D to detach from screen. Reattach with screen -x code.
Note: Only one web UI can run on port 9999 at a time. Stop the current one before starting another.
Each container includes an AI Tools Admin web app accessible at https://admin.code.yourdomain.com. This provides:
MANAGE View:
- Toggle AI coding tools (OpenCode, NanoCode, OpenHands) on/off
- Only one tool can run at a time on port 9999
- View real-time output logs
- Links to App URL and Code UI
- DNS health check indicators
UPDATE View:
- One-click update for all three AI tools
- Automatically stops running processes before updating
- Shows live progress of updates
The admin app runs as a systemd service (admin-app) and is protected by the same Basic Auth credentials as the Code UI.
Note: For OpenHands, the first start may take 2-5 minutes while Docker images are downloaded.
From the container detail view in the TUI, press u to update opencode and nanocode to their latest versions.
ssh -p 2222 container-name@host.example.com
# You'll be logged in as 'ubuntu' (or 'debian') with sudo accessssh user@host.example.comFor HTTPS to work, DNS must point to the host server:
domain.com→ Host IPcode.domain.com→ Host IP (for AI coding web UI)admin.code.domain.com→ Host IP (for AI tools admin app)
Caddy will automatically obtain Let's Encrypt certificates for both domains.
Internet
│
┌────────────┴────────────┐
│ │
▼ ▼
HTTPS (:443) SSH (:2222 via SSHPiper)
│ │
┌───────────────────▼─────────────────────▼─────────────────┐
│ Host System │
├─────────────────────────────────────────────────────────────┤
│ │
│ ┌─────────────┐ ┌─────────────┐ │
│ │ Caddy │ │ SSHPiper │ │
│ │ (reverse │ │ (SSH │ │
│ │ proxy) │ │ router) │ │
│ └──────┬──────┘ └──────┬──────┘ │
│ │ │ │
│ │ Routes by domain │ Routes by username │
│ ▼ ▼ │
│ ┌─────────────────────────────────────────────────────┐ │
│ │ Incus (LXC) │ │
│ │ │ │
│ │ ┌─────────────────┐ ┌─────────────────┐ │ │
│ │ │ Container │ │ Container │ ... │ │
│ │ │ (Ubuntu/Debian)│ │ (Ubuntu/Debian)│ │ │
│ │ │ │ │ │ │ │
│ │ │ ┌───────────┐ │ │ ┌───────────┐ │ │ │
│ │ │ │ opencode/ │ │ │ │ opencode/ │ │ │ │
│ │ │ │ nanocode/ │ │ │ │ nanocode/ │ │ │ │
│ │ │ │ openhands │ │ │ │ openhands │ │ │ │
│ │ │ └───────────┘ │ │ └───────────┘ │ │ │
│ │ │ ┌───────────┐ │ │ ┌───────────┐ │ │ │
│ │ │ │ Your App │ │ │ │ Your App │ │ │ │
│ │ │ │ (:8000) │ │ │ │ (:8000) │ │ │ │
│ │ │ └───────────┘ │ │ └───────────┘ │ │ │
│ │ └─────────────────┘ └─────────────────┘ │ │
│ └─────────────────────────────────────────────────────┘ │
│ │
│ ┌───────────────────┐ │
│ │ vibebin │ TUI for container management │
│ │ (this tool) │ - Create/delete containers │
│ │ │ - Configure domains & auth │
│ └───────────────────┘ │
│ │
└─────────────────────────────────────────────────────────────┘
- HTTPS requests to
myapp.example.com→ Caddy → Container's app (port 8000) - HTTPS requests to
code.myapp.example.com→ Caddy (with Basic Auth) → AI coding web UI (port 9999) - SSH connections to port 2222 as
myapp-example-com@host→ SSHPiper → Container's SSH asubuntu/debian
Routes are managed via Caddy's Admin API (localhost:2019), not config files:
- Routes use
@idfor identification (e.g.,container-name-app,container-name-code) - Changes are atomic and immediate (no reload required)
- Query current routes:
curl http://localhost:2019/config/apps/http/servers/srv0/routes
Containers use Incus's default "last-state" behavior (by not setting boot.autostart):
- Running containers will restart when the host reboots
- Stopped containers will stay stopped
Incus automatically tracks each container's power state and restores it when the daemon starts.
Snapshots allow you to save and restore the complete state of a container.
From the container detail view, press S to access snapshot management, then n to create.
- Before risky changes: Snapshot before major updates or experiments
- Known-good states: Save working configurations you can restore to
- Quick rollback: Instantly revert if something breaks
Note: Snapshots are stored by Incus and consume disk space. Delete old snapshots to free space.
Container won't start:
journalctl -u incus -f
incus info container-nameCaddy certificate errors:
- Ensure DNS is configured and pointing to host IP before creating container
- Check Caddy logs:
journalctl -u caddy -f - Check current routes:
curl -s http://localhost:2019/config/apps/http/servers/srv0/routes | jq .
SSH to containers not working:
- Verify SSHPiper is running:
systemctl status sshpiperd - Ensure you're using port 2222:
ssh -p 2222 container-name@host - Verify upstream config:
cat /var/lib/sshpiper/container-name/sshpiper_upstream
AI coding tools not working:
- SSH to the container and run
opencode,nanocode, oropenhandsinteractively - All tools will prompt for API key configuration on first run
- Check tool versions:
opencode --version,nanocode --version,openhands --version
Sync daemon issues:
journalctl -u vibebin-sync -fYou can use subdomains for your containers:
app.example.com- works correctlystaging.app.example.com- works correctlymy-app.example.com- works correctly
Domains with two-part TLDs (like .co.uk, .com.au) are not fully supported
for automatic DNS creation. The zone detection assumes a single-part TLD.
For example:
- ✅
app.example.com→ zone:example.com(correct) - ❌
app.example.co.uk→ zone:co.uk(incorrect, should beexample.co.uk)
Workaround: For two-part TLDs, select "No" for auto DNS creation and configure DNS records manually.
Each container created by vibebin is fully isolated by default:
- Each container has its own complete, persistent filesystem (Ubuntu 24.04 or Debian 13)
- Containers cannot access each other's filesystems
- Uses Incus/LXC which provides OS-level virtualization with separate root filesystems
- Data persists across container restarts
- Each container gets its own private IP on an internal bridge network (typically
10.x.x.x) - Containers cannot directly communicate with each other by default
- All external access is routed through:
- Caddy (HTTPS reverse proxy) for web traffic
- SSHPiper (port 2222) for SSH access
- Each container only exposes what's explicitly configured (app port, code UI port)
- Containers use Linux namespaces (PID, network, mount, user, etc.)
- Processes in one container cannot see or interact with processes in another
- Each container has its own init system and process tree
security.nesting=true is enabled to allow Docker-in-container (needed for OpenHands). This is standard for development environments but worth noting for security-sensitive deployments.
If you run 3 containers (app1.example.com, app2.example.com, app3.example.com):
- Each has its own isolated Linux environment
- Each has its own AI tools installation (opencode, nanocode, openhands)
- An agent in app1 cannot access files or processes in app2 or app3
- They share the host's resources (CPU, RAM) but are otherwise independent
It's essentially like having separate lightweight VMs, but with much lower overhead than full virtualization.
Currently, this implementation uses the Incus DIR storage driver for proof of concept. The DIR driver uses basic filesystem-level storage and is:
- Simple to set up (no additional dependencies)
- Compatible with any filesystem
- Slow for snapshots (full copy-on-write not available)
Planned: During the dependencies/installation phase, users will be able to choose between:
| Driver | Pros | Cons |
|---|---|---|
| DIR | Simple, works everywhere | Slow snapshots, no CoW |
| Btrfs | Fast snapshots, CoW, compression | Requires Btrfs filesystem |
| ZFS | Fast snapshots, CoW, excellent features | Requires significant RAM (1GB+ per TB of storage) |
Btrfs and ZFS provide instant snapshots via copy-on-write, making them much more suitable for production use.
This project is licensed under the MIT License. See LICENSE for details.
- opencode: anomalyco/opencode - MIT License
- nanocode: nanogpt-community/nanocode - Fork of opencode
- openhands: All-Hands-AI/OpenHands - MIT License
- Incus: linuxcontainers/incus - Apache 2.0 License
- Caddy: caddyserver/caddy - Apache 2.0 License
- SSHPiper: tg123/sshpiper - MIT License


