Agent Safety Kit command-line utilities
Project description
Agent Safety Kit
A toolkit for running AI agents in an isolated environment inside a Multipass virtual machine.
Why this matters
Some stories (you can find plenty more):
- Qwen Coder agent destroys working builds
- Codex keeps deleting unrelated and uncommitted files! even ignoring rejected requests
- comment: qwen-code CLI destroyed my entire project, deleted important files
- Claude Code deleted my entire workspace, here's the proof
- I Asked Claude Code to Fix All Bugs, and It Deleted the Whole Repo
- Codex has twice deleted and corrupted my files (r/ClaudeAI comment)
Everyone says "you should have backups" and "everything must live in git", but console AI agents still lack built-in snapshots to roll back after every change they make. Until sandboxes catch up, this toolkit helps you manage that yourself.
Key ideas
- Agents run only inside a virtual machine.
- The VM is launched via Multipass (a simple Canonical tool to start Ubuntu VMs with a single command).
- Project folders from the host are mounted into the VM; an automatic backup job runs in parallel to a sibling directory at a configurable interval (defaults to every five minutes and only when changes are detected), using
rsyncwith hardlinks to save space. - VM, mount, and cloud-init settings are stored in a YAML config.
- You can run the agent without entering the guest via
multipass shell—it still executes inside the VM. - Multipass commands and agent runs can be wrapped in proxychains: set a proxy URL per VM or override it once with
--proxychainsto generate a temporary proxychains config automatically.
Working agents
Currently confirmed working agent types are:
- qwen
- codex
- codex-glibc (built dynamically)
Quick start
-
Install the package with pip (requires Python 3.9 or newer):
python3 -m venv ./venv source ./venv/bin/activate pip install agsekit
This makes the
agsekitcommand available inside the virtual environment. -
Alternatively, clone the repository and install from sources:
git clone https://github.com/MihanEntalpo/agent-safety-kit/ cd agent-safety-kit pip install .
-
Create a YAML configuration (the CLI checks
--config, thenCONFIG_PATH, then~/.config/agsekit/config.yaml):agsekit config-example # edit vms/mounts/cloud-init to your needs
When working from a cloned repository, you can also copy the file directly:
mkdir -p ~/.config/agsekit cp config-example.yaml ~/.config/agsekit/config.yaml
You can also run
agsekit config-gento answer a few questions and save the config (defaults to~/.config/agsekit/config.yaml; use--overwriteto replace an existing file). -
Install required system dependencies (in particular, Multipass; requires sudo and currently works only on Debian-based systems):
agsekit prepare -
Create the virtual machines defined in YAML:
agsekit create-vmsTo launch just one VM, use
agsekit create-vm <name>. If the config contains only one VM, you can omit<name>and it will be used automatically. If a VM already exists, the command compares the desired resources with the current ones and reports any differences. Changing resources of an existing VM is not supported yet. -
Mount your folders (assuming mounts are already configured in the YAML file):
agsekit mount --all
-
Install all configured agents into their default VMs:
agsekit install-agents --all-agents
-
Launch an agent inside its VM (example runs
qwenin the folder where/host/path/projectis mounted, with backups enabled by default):agsekit run qwen /host/path/project --vm agent-ubuntu
On the very first run with backups enabled, the CLI creates an initial snapshot with progress output before launching the agent, so wait for it to complete.
agsekit commands
Setup and VM lifecycle
agsekit prepare— installs required system dependencies (including Multipass; requires sudo and currently works only on Debian-based systems).agsekit config-gen [--config <path>] [--overwrite]— interactive wizard that asks about VMs, mounts, and agents, then writes a YAML config to the chosen path (defaults to~/.config/agsekit/config.yaml). Without--overwrite, the command warns if the file already exists.agsekit config-example [<path>]— copiesconfig-example.yamlto the target path (defaults to~/.config/agsekit/config.yaml). If the default config already exists, the command skips copying.agsekit create-vms— creates every VM defined in the YAML configuration.agsekit create-vm <name>— launches just one VM. If the config contains only one VM, you can omit<name>and it will be used automatically. If a VM already exists, the command compares the desired resources with the current ones and reports any differences. Changing resources of an existing VM is not supported yet.agsekit shell [<vm_name>] [--config <path>]— opens an interactivemultipass shellsession inside the chosen VM, applying any configured port forwarding. If only one VM is defined in the config, the CLI connects there even withoutvm_name. When multiple VMs exist and the command runs in a TTY, the CLI prompts you to pick one; in non-interactive mode, an explicitvm_nameis required.agsekit ssh <vm_name> [--config <path>] [<ssh_args...>]— connects to the VM over SSH using~/.config/agsekit/ssh/id_rsaand forwards any extra arguments directly to thesshcommand (for example,-L,-R,-N).agsekit portforward [--config <path>]— starts a dedicatedagsekit sshtunnel for each VM that definesport-forwardingrules, monitoring the child processes and restarting them if they exit. Stop with Ctrl+C to gracefully terminate the tunnels.agsekit start-vm <vm_name> [--config <path>]— starts the specified VM from the configuration. If only one VM is configured, the name can be omitted.agsekit start-vm --all-vms [--config <path>]— starts every VM declared in the config file.agsekit stop-vm <vm_name> [--config <path>]— stops the specified VM from the configuration. If only one VM is configured, the name can be omitted.agsekit stop-vm --all-vms [--config <path>]— stops every VM declared in the config file.agsekit destroy-vm <vm_name> [--config <path>] [-y]— deletes the specified VM from Multipass. Without-y, the CLI asks for interactive confirmation.agsekit destroy-vm --all [--config <path>] [-y]— deletes every VM from the configuration, with the same confirmation requirement.agsekit systemd install [--config <path>]— writes~/.config/agsekit/systemd.envwith absolute paths toagsekit, the config, and the current project directory, then registers and starts the user unit fromsystemd/agsekit-portforward.serviceviasystemctl --user(link, daemon-reload, start, enable).agsekit systemd uninstall— stops and disables the user unit, then removes the linkedsystemd/agsekit-portforward.servicefrom systemd viasystemctl --user.
Mount management
agsekit mount --source-dir <path> [--config <path>]— mounts the directory described bysourcein the configuration file (default search:--config,CONFIG_PATH,~/.config/agsekit/config.yaml) into its VM usingmultipass mount. Use--allto mount every entry from the config. When there is only one mount in the config, the command can be run without--source-diror--all.agsekit umount --source-dir <path> [--config <path>]— unmounts the directory described bysourcein the config (orCONFIG_PATH/--config);--allunmounts every configured path. If only one mount is configured, the command will unmount it even without explicit flags.
Backups
One-off backup
agsekit backup-once --source-dir <path> --dest-dir <path> [--exclude <pattern> ...] [--progress] — runs a single backup of the source directory into the specified destination using rsync.
The command creates a timestamped directory with a -partial suffix, supports incremental copies via --link-dest to the previous backup, and honors exclusions from .backupignore and --exclude arguments. When finished, the temporary folder is renamed to a final timestamp without the suffix. If nothing changed relative to the last backup, no new snapshot is created and the tool reports the absence of updates.
Pass --progress to forward rsync progress flags and show a console progress bar while files are copied.
.backupignore examples:
# exclude virtual environments and dependencies
venv/
node_modules/
# ignore temporary and log files by pattern
*.log
*.tmp
# include a specific file inside an excluded folder
!logs/important.log
# skip documentation build artifacts
docs/build/
Backups use rsync with incremental links (--link-dest) to the previous copy: if only a small set of files changed, the new snapshot stores just the updated data, while unchanged files are hardlinked to the prior snapshot. This keeps a chain of dated directories while consuming minimal space when changes are rare.
Repeated backups
agsekit backup-repeated --source-dir <path> --dest-dir <path> [--exclude <pattern> ...] [--interval <minutes>] [--skip-first]— runs an immediate backup and then repeats it everyintervalminutes (defaults to five minutes). With--skip-first, the loop waits for the first interval before performing the initial run. After each backup it printsDone, waiting N minuteswith the actual interval value.agsekit backup-repeated-mount --mount <path> [--config <path>]— looks up the mount by itssourcepath in the configuration file (default search:--config,CONFIG_PATH,~/.config/agsekit/config.yaml) and launches repeated backups using the paths and interval from the config. When only one mount is present,--mountcan be omitted; with multiple mounts, an explicit choice is required.agsekit backup-repeated-all [--config <path>]— reads all mounts from the config (default search:--config,CONFIG_PATH,~/.config/agsekit/config.yaml) and starts concurrent repeated backups for each entry within a single process. Use Ctrl+C to stop the loops.
Agent installation
agsekit install-agents <agent_name> [<vm>|--all-vms] [--config <path>] [--proxychains <value>]— runs the prepared installation script for the chosen agent type inside the specified VM (or the agent's default VM if none is provided). If the config defines only one agent, you can skip<agent_name>and it will be picked automatically. Use--proxychains <scheme://host:port>to override the VM proxy for this installation or--proxychains ""to ignore it once.agsekit install-agents --all-agents [--all-vms] [--config <path>] [--proxychains <value>]— installs every configured agent either into their default VM or into every VM when--all-vmsis set.
The installation scripts live in agsekit_cli/agent_scripts/: codex installs the npm CLI, codex-glibc builds the Rust sources with the glibc target and installs the binary as codex-glibc, and qwen/claude-code follow their upstream steps (the qwen script installs the qwen-code CLI). Other agent types are not supported yet.
Running agents
agsekit run <agent_name> [<source_dir>|--vm <vm_name>] [--config <path>] [--proxychains <value>] [--disable-backups] [--skip-default-args] [--debug] -- <agent_args...>— starts an interactive agent command inside Multipass. Environment variables from the config are passed to the process. If asource_dirfrom the mounts list is provided, the agent starts inside the mounted target path in the matching VM; otherwise it launches in the home directory of the default VM. Unless--disable-backupsis set, background repeated backups for the selected mount are started for the duration of the run. When no backups exist yet, the CLI first creates an initial snapshot with progress output before launching the agent and then starts the repeated loop with the initial run skipped. Arguments fromagents.<name>.default-argsare added unless--skip-default-argsis set; if the user already passed an option with the same name (for example--openai-api-key), the default value is skipped. With--debug, the CLI prints every external command before executing it to help troubleshoot agent launches. Use--proxychains <scheme://host:port>to override the VM setting for one run; pass an empty string to disable it temporarily.
Interactive mode
In a TTY you don’t have to type full commands every time: the CLI can guide you through an interactive menu that fills in parameters for you.
- Run
agsekitwithout arguments to open the interactive menu, choose a command, and select options such as the config path, mounts, or agent parameters. - Start a command without mandatory arguments (for example,
agsekit run) to automatically fall back to the interactive flow after the CLI prints a “not enough parameters” hint. Use--non-interactiveif you prefer the usual help output instead of prompts.
Localization
The CLI reads the system locale and falls back to English if it cannot detect a supported language. You can override this behavior with the AGSEKIT_LANG environment variable:
AGSEKIT_LANG=ru agsekit --help
YAML configuration
The configuration file (looked up via --config, CONFIG_PATH, or ~/.config/agsekit/config.yaml) describes VM parameters, mounted directories, and any cloud-init settings. A base example lives in config-example.yaml:
vms: # VM parameters for Multipass (you can define multiple)
agent-ubuntu: # VM name
cpu: 2 # number of vCPUs
ram: 4G # RAM size (supports 2G, 4096M, etc.)
disk: 20G # disk size
proxychains: "" # optional proxy URL (scheme://host:port); agsekit writes a temporary proxychains.conf and wraps Multipass commands automatically
cloud-init: {} # place your standard cloud-init config here if needed
port-forwarding: # Port forwarding config
- type: remote # Open port inside VM and pass connections to Host machine's port
host-addr: 127.0.0.1:80
vm-addr: 127.0.0.1:8080
- type: local # Open port on Host machine, and pass connections to VM's port
host-addr: 0.0.0.0:15432
vm-addr: 127.0.0.1:5432
- type: socks5 # Open socks5-proxy port inside VM, directing traffic to Host machine's network
vm-addr: 127.0.0.1:8088
mounts:
- source: /host/path/project # path to the source folder on the host
target: /home/ubuntu/project # mount point inside the VM; defaults to /home/ubuntu/<source_basename>
backup: /host/backups/project # backup directory; defaults to backups-<source_basename> next to source
interval: 5 # backup interval in minutes; defaults to 5 if omitted
vm: agent-ubuntu # VM name; defaults to the first VM in the configuration
agents:
qwen: # agent name; add as many as you need
type: qwen # agent type: qwen (installs and uses the `qwen` binary), codex, codex-glibc (installs the `codex-glibc` binary), or claude-code (other types are not supported yet)
env: # arbitrary environment variables passed to the agent process
OPENAI_API_KEY: "my_local_key"
OPENAI_BASE_URL: "https://127.0.0.1:11556/v1"
OPENAI_MODEL: "Qwen/Qwen3-Coder-30B-A3B-Instruct-FP8"
default-args: # arguments passed to the agent unless the user overrides them
- "--openai-api-key=my_local_key"
- "--openai-base-url=https://127.0.0.1:11556/v1"
vm: qwen-ubuntu # default VM for this agent; falls back to the mount VM or the first VM in the list
codex:
type: codex
claude:
type: claude-code
codex2:
type: codex-glibc
Note: Prefer ASCII-only paths for both
sourceandtargetmount points: AppArmor may refuse to mount directories whose paths contain non-ASCII characters.
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 agsekit-0.9.5.tar.gz.
File metadata
- Download URL: agsekit-0.9.5.tar.gz
- Upload date:
- Size: 70.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.9.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b7cead2ad89c4fea7bf11c29b92485ec64afa9c0fbcb834273b66993aef311eb
|
|
| MD5 |
6739151d463602e5b1c6876e6218a2e3
|
|
| BLAKE2b-256 |
6d2ff729c40dac56bf93cc7d58868eeb456e4493f750b03f6a33f5650d9386e8
|
File details
Details for the file agsekit-0.9.5-py3-none-any.whl.
File metadata
- Download URL: agsekit-0.9.5-py3-none-any.whl
- Upload date:
- Size: 75.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.9.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
23ed083810ad92c68173e6c7c526cc2e87b78f879b6845475013d97b0f42bfb4
|
|
| MD5 |
b05ceac8ec69b6bb3081632bfe14366d
|
|
| BLAKE2b-256 |
a3ce9e52aa4c9f51dca276850279223099b08dc0f50762092ee970c588164fcf
|