internals/skills/vm-spec/SKILL.md
Go type reference for VmSpec and the discriminated-union source types (VmSource cloud_image | bootc). Documents every field, validation rules, and the adopt-user decision. Source files: ov/vm_spec.go, ov/cloud_init_types.go, ov/libvirt_validate.go. MUST be invoked before editing VmSpec Go code or authoring vm.yml entries.
npx skillsauth add overthinkos/overthink-plugins vm-specInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Go type reference for the VM surface. VmSpec + VmSource + VmChecksum + VmNetwork + VmSSH + VmKeyInjection + VmCloudInit + VmOvInstall are the canonical types that drive ov vm build, ov vm create, and ov deploy add vm:<name>. This skill is the authoritative Go reference — field semantics, defaults, validation rules, migration history. The YAML-authoring companion is /ov-vm:vms-catalog.
| File | Contents |
|---|---|
| ov/vm_spec.go | VmSpec, VmSource, VmChecksum, VmNetwork, VmSSH, VmKeyInjection |
| ov/cloud_init_types.go | VmCloudInit, VmCloudInitUser, VmCloudInitFile, VmCloudInitNetwork, VmCloudInitMirrors, VmOvInstall |
| ov/libvirt_schema.go | LibvirtConfig + 30+ sub-types (features, CPU, clock, devices, etc.) |
| ov/libvirt_validate.go | ValidateVmSpec, ValidateLibvirtConfig |
type VmSpec struct {
Source VmSource // discriminated union: kind = cloud_image | bootc
DiskSize string // "20G", "10 GiB", "1T" — virtual size; the qcow2 is lazily/sparsely allocated
Ram string // "4G", "8192M"
Cpus int
Machine string // q35 | virt | i440fx — default: host-native
Firmware string // bios | uefi-insecure | uefi-secure — default: bios
Backend string // auto | libvirt | qemu — pins the backend for this entity
Autostart bool // libvirt domain autostart (libvirt backend only)
Network *VmNetwork
SSH *VmSSH
CloudInit *VmCloudInit
Libvirt *LibvirtConfig
}
Every field except Source, CloudInit, and Source-branch-specific subfields applies equally to both source kinds — the parity guarantee. Anything configurable for cloud_image VMs is configurable for bootc VMs, and vice versa.
autostart: true sets libvirt's per-domain autostart flag (DomainSetAutostart, in runVmSpecCreate after define). Because VMs run under qemu:///session, that flag only fires at host boot once the session daemon is running — and there is no portable user-level virtqemud.socket to socket-activate it (Arch/CachyOS ships none). So ensureBootAutostartPrereqs (ov/vm.go) (a) runs loginctl enable-linger <user> (idempotent) and (b) writes + enables a per-VM user systemd oneshot ov-autostart-<domain>.service that runs virsh -c qemu:///session start <domain> at boot (WantedBy=default.target); virsh spawns the session daemon on demand and starts the already-defined domain — deterministic and cross-distro. ov vm destroy removes the unit (removeAutostartUserUnit). The libvirt flag is a domain property (not XML), so it survives DomainDefineXML redefinitions; runVmSpecCreate re-asserts both on every create/rebuild. ValidateVmSpec rejects autostart: true with backend: qemu. Additive optional field — no schema-version bump.
DiskSize is the virtual size: the bootstrap path's truncate + qemu-img convert -O qcow2 (no preallocation) produces a sparse qcow2 that grows on demand, so disk_size: 1T costs only the bytes actually written.
A libvirt.devices.filesystems[] entry with driver: virtiofs +
accessmode: passthrough + a host-path source is auto-given a guest-user
<idmap> at render time so the share is owned by the guest's interactive user
(uid 1000), not guest-root. This is what makes source: /home/<you> →
target: workspace usable as the SSH user inside the guest. Mechanism +
the exact id partition live in /ov-internals:libvirt-renderer "virtiofs
guest-user idmap"; shared-memory auto-pairing is in the same skill.
type VmSource struct {
Kind string // "cloud_image" | "bootc"
// cloud_image branch:
URL string
Checksum VmChecksum
Cache string
BaseUser string // adopt-user pattern — see below
// bootc branch:
Image string // kind:image entry name
Transport string // registry | containers-storage | oci | oci-archive
Rootfs string // ext4 | xfs | btrfs
RootSize string // "10G" — caps root partition, rest unpartitioned
KernelArgs string
}
Kind is the discriminator. ValidateVmSpec enforces that exactly one branch's required fields are populated.
BaseUser mirrors the container-side base_user: + user_policy: adopt pattern. When set:
/ov-internals:cloud-init-renderer::composeUsers emits users: [default, {name: <base_user>, ssh_authorized_keys: [...]}] — merge-by-name, no useradd.spec.ssh.user defaults to BaseUser.~<base_user>/.ssh/authorized_keys without touching sudoers/shell/home.Common values: arch (Arch cloud image), ubuntu (Ubuntu Cloud), fedora (Fedora Cloud), debian (Debian Cloud), cloud-user (CentOS Cloud).
Leave empty only when the image has no default account — then declare a custom user in spec.cloud_init.users.
type VmSSH struct {
User string // default: cloud_image → "ov" OR base_user; bootc → "root"
Port int // default: 2222
KeySource string // auto | generate | none | <abs-path>
KeyInjection *VmKeyInjection
}
type VmKeyInjection struct {
SMBIOS string // auto | enabled | disabled
CloudInit string // auto | enabled | disabled
}
Dual-channel key injection is additive. Per-source-kind auto-defaults when KeyInjection is nil:
cloud_image → {smbios: enabled, cloud_init: enabled} — belt + suspenders; cloud-init seed ISO is always emitted anyway.bootc → {smbios: enabled, cloud_init: disabled} — cloud-init seed ISO only emits when the guest has the cloud-init layer.Having both channels on simultaneously is the safe default; there's no duplication cost at the guest (systemd-ssh-generator dedups entries in authorized_keys).
The renderer combines structured fields with renderer defaults:
Packages: prepended with {openssh, curl, tar}.RunCmd: prepended with {systemctl enable --now sshd} so distro-specific setup can assume sshd is running.Users: the VmSSH.User account is auto-injected unless already present; if present, the renderer appends the ssh pubkey to the user's existing entry.Extra is a raw-YAML escape hatch merged after structured fields. Prefer structured fields; Extra exists for long-tail cloud-init options the schema doesn't cover.
type VmOvInstall struct {
Strategy string // auto | scp | url | skip
URL string // when Strategy == "url"
Checksum string // "sha256:<hex>" for url strategy
}
| Strategy | Behavior |
|---|---|
| auto (default) | scp the local ov binary (os.Executable()) into the guest post-boot via VmDeployTarget |
| scp | explicit form of auto |
| url | cloud-init runcmd downloads ov from URL at first boot |
| skip | user manages ov install; VmDeployTarget verifies presence only |
ValidateVmSpec enforces the invariants documented in /ov-vm:vms-catalog. Key checks:
source.kind ∈ {cloud_image, bootc}.url: populated; bootc branch requires image:.firmware: ∈ {bios, uefi-insecure, uefi-secure}.network.mode: ∈ {user, bridge, nat}.ssh.key_source: parses as auto | generate | none | absolute path.ssh.key_injection.{smbios,cloud_init} ∈ {auto, enabled, disabled}.Libvirt structure routed to ValidateLibvirtConfig.Failed validation → hard load-time error with a one-line remediation hint pointing at /ov-vm:vms-catalog or ov migrate.
The legacy VmConfig type + ImageConfig.Vm + ImageConfig.Libvirt + ResolvedImage.Vm + LabelVm + LabelLibvirt were all deleted in the hard cutover. Field mapping for forensic purposes:
| Legacy location | New location |
|---|---|
| image.bootc: true + image.vm.disk_size | vms.<name>.source.kind: bootc + vms.<name>.disk_size |
| image.vm.ssh_port | vms.<name>.ssh.port |
| image.vm.ram, .cpus, .rootfs, .root_size, .kernel_args | vms.<name>.ram, .cpus, source.rootfs, source.root_size, source.kernel_args |
| image.vm.firmware | vms.<name>.firmware |
| image.vm.network (string) | vms.<name>.network.mode |
| image.libvirt: ["<xml>", …] (list of strings) | vms.<name>.libvirt.snippets: […] + structured libvirt.devices.* |
ov migrate performs this mapping idempotently. See /ov-build:migrate for the command and /ov-internals:cutover-policy for the policy.
/ov-vm:vms-catalog — YAML-authoring companion (when to pick cloud_image vs bootc, adopt pattern, step-by-step recipes)/ov-internals:libvirt-renderer — LibvirtConfig rendering + pure render functions/ov-internals:cloud-init-renderer — RenderCloudInit, composeUsers, seed ISO/ov-internals:vm-deploy-target — VmDeployTarget consuming VmSpec via DeployExecutor/ov-internals:ovmf — ResolveOvmfForSpec reads spec.Firmware/ov-internals:cutover-policy — why the legacy surface was deleted in one PR/ov-vm:vm — command-family skill; reads vm.yml through VmSpec/ov-build:migrate — ov migrate command/ov-internals:go — Go CLI development overviewdevelopment
Claude Code multi-agent support in Overthink — sub-agents, dynamic workflows, and agent teams, and how each drives the existing `ov eval` disposable beds to test and verify. MUST be invoked before authoring or invoking an ov sub-agent / dynamic workflow / agent team, wiring agent-lifecycle hooks, or asking "which primitive should drive the R10 beds?".
tools
Mounts a virtiofs share tagged `workspace` at /workspace inside a VM guest via a systemd .mount unit. Use when a kind:vm entity shares a host directory into the guest and you need it auto-mounted (and re-mounted at every boot).
development
MUST be invoked before any work involving: the `kind: android` schema kind, a `target: android` deploy, the `apk:` layer package format (installing Android apps declaratively), AndroidDeployTarget, an in-pod emulator OR a remote/physical adb-endpoint device, or nested `pod → android` deployment. The first-class Android device + app surface that sits above `ov eval adb`/`appium`.
tools
Use when committing, branching, pushing, merging, tagging, creating PRs, or approving/merging PRs with gh — the feat/-branch, R10-gated, never-force-push landing workflow across the main repo + the plugins submodule + image/<distro> submodules. Covers sync-to-upstream, branch/worktree pruning, the fork+PR path for contributors without write access, and cross-repo @github landing order.