Local libvirt/KVM sandbox VM manager for coding agents (Ubuntu 24.04 cloud-image, SSH, optional virtiofs share, optional nftables isolation).
Project description
Read the Docs |
|
Pypi |
A small Python CLI to create and manage a local libvirt/KVM Ubuntu 24.04 VM designed for running coding agents with a stronger boundary than containers.
What it provides
Dedicated libvirt NAT network per aivm configuration
Optional host firewall isolation via nftables
Ubuntu cloud-image VM provisioning via cloud-init
SSH + VS Code Remote-SSH workflows
Optional virtiofs folder sharing (explicit trust extension)
Optional settings sync into the guest user profile
A single config store for defaults, VMs, networks, and attachments
Install
uv pip install .
Fast Start
Recommended for new repos:
Currently aivm config init is required, but we will make that implicit in a future version.
aivm code .
aivm status
aivm status --sudo # optional deeper privileged checks
aivm code . auto-selects/bootstraps VM context from the global config store (~/.config/aivm/config.toml), attaches the current folder if needed, and opens VS Code.
During setup and reconcile flows, subprocess logging is now organized around user-meaningful steps instead of isolated commands. aivm shows the current step, why it exists, a semantic summary for each planned command, and the exact command line that will run before it executes the step. Full raw commands still appear at higher verbosity.
If you prefer an explicit flow, aivm config init is required before aivm vm create.
See also:
Status and sudo behavior
By default, aivm status avoids privileged probes. Use --sudo for network/firewall/libvirt/image checks.
Command manager defaults:
subprocess execution is centralized through a command manager
logs are grouped into step/plan previews with nested context
read-only sudo probes (inspect/query/status) are auto-approved by default
state-changing sudo steps still prompt unless --yes/--yes-sudo is set
approval usually happens once per grouped step, not once per command
Grouped approval does not widen privilege beyond the commands shown in the step preview. The preview is the approval boundary.
Use:
--yes to auto-approve all prompts
--yes-sudo to auto-approve only sudo prompts
When running interactively, expect step previews such as:
current context / breadcrumb
current step title
why the step exists
semantic summaries plus exact commands for the current step
a single approval prompt for the whole step when required
Interactive approval semantics:
y approves the current step only
a approves the current step and all later steps
s shows the full exact commands for the current step, then reprompts
For example, the default shared-root path used by aivm ssh . / aivm code . now groups attachment reconciliation into named steps such as inspecting host bind state, preparing host bind targets, ensuring the VM virtiofs mapping, and mounting/verifying the bind inside the guest.
Readable previews may abbreviate long shell payloads, but the full exact commands are still available on demand in the approval prompt and are always logged when they actually run.
Config defaults:
[behavior]
yes_sudo = false
auto_approve_readonly_sudo = true # set false for strict "prompt every sudo" mode
Common Workflows
VS Code and SSH
aivm vm ssh_config
aivm code . --sync_settings
aivm vm code --host_src . --sync_settings
aivm vm code . --sync_settings
aivm vm ssh .
Folder attachment
aivm attach .
aivm detach .
aivm vm attach --vm aivm-2404 --host_src .
aivm attach . --mode git
Attachment modes:
shared-root (default for new attachments): one VM-level virtiofs mapping exports /var/lib/libvirt/aivm/<vm>/shared-root; each attached folder is bind-mounted under that root on host and then bind-mounted to guest_dst in guest.
shared: direct per-folder virtiofs mapping from host source to guest. This is simpler but consumes one VM virtiofs device slot per folder.
git: guest-local Git clone, synced via host/guest remotes.
In shared and shared-root modes, attached folders mount to the same absolute path inside the guest by default. In git mode, default guest paths are placed under /home/<vm-user>/... so Git sync can run without guest root privileges. Use --guest_dst to override in any mode. Running VMs are live-attached when possible. aivm code and aivm ssh remount the selected folder and best-effort restore other folders already saved for that VM after guest startup.
Config-store lifecycle (explicit flow)
aivm config init
aivm vm create
aivm vm sync_settings
aivm vm update
aivm config discover
aivm config show
aivm config edit
aivm help plan
aivm help tree
aivm help completion
aivm host doctor
Settings sync configuration
[sync]
enabled = true
overwrite = true
paths = [
"~/.gitconfig",
"~/.gitignore",
"~/.config/Code/User/settings.json",
"~/.config/Code/User/keybindings.json",
"~/.tmux.conf",
"~/.bashrc",
]
Ad hoc override:
aivm vm sync-settings \
--paths "~/.gitconfig,~/.config/Code/User/settings.json,~/.tmux.conf"
When [sync].enabled=true, aivm vm code ... syncs before launching VS Code.
Command Groups
aivm config --help
aivm host --help
aivm host image_fetch --help
aivm help --help
aivm host net --help
aivm host fw --help
aivm vm --help
Safety Notes
This tool assumes Linux + libvirt. It focuses on Debian/Ubuntu hosts for dependency installation.
Security model and threat model details: the Security Model.
NAT alone does not prevent VM -> LAN. Enable firewall isolation if you want “internet-only” access.
To allow specific VM->host or VM->blocked-LAN service ports while firewall isolation is enabled, set [firewall].allow_tcp_ports / allow_udp_ports in config (for example allow_tcp_ports = [22, 5432]).
virtiofs sharing is optional; it’s powerful, but it intentionally exposes that host directory to the VM.
aivm vm code requires VS Code’s code CLI and the Remote - SSH extension.
Project details
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 aivm-0.4.0.tar.gz.
File metadata
- Download URL: aivm-0.4.0.tar.gz
- Upload date:
- Size: 147.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b9b938e704ba7653dbce305ff4a4a3fedfe6b0e48a2c8007ad600e9a647110ef
|
|
| MD5 |
dc570ce8b7c3c3f0117784fabe3358c8
|
|
| BLAKE2b-256 |
71a022fbd87b95a825aa3fd069d09670ea5c34a98028e2004d0f02f9d61377e4
|
File details
Details for the file aivm-0.4.0-py3-none-any.whl.
File metadata
- Download URL: aivm-0.4.0-py3-none-any.whl
- Upload date:
- Size: 113.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
dc3cc0c1cc5e38ee0fd27d7100a13f1ae07c168df1254955872570dfa254fb38
|
|
| MD5 |
f615e7c8476eaa1d1c61333ff6495daa
|
|
| BLAKE2b-256 |
89c961897feed79a422165cb8dff605ff3a75ac0d58f208110487b76b9b4411c
|