Make iterating on a Firecracker-friendly Void guest practical without replacing the Debian default image path. Add local Void rootfs build/register/verify plumbing, a language-agnostic dev package baseline, and guest SSH/work-disk hardening so new images use the runtime bundle key, keep a normal root bash environment, and repair stale nested /root layouts on restart. Replace the guest PING/PONG responder with an HTTP /healthz agent over vsock, rename the runtime bundle and config surface from ping helper to agent while still accepting the legacy keys, and route the post-SSH reminder through the new vm.health path. Validated with GOCACHE=/tmp/banger-gocache go test ./..., make build, bash -n customize.sh make-rootfs-void.sh, and git diff --check.
93 lines
2.5 KiB
Go
93 lines
2.5 KiB
Go
package paths
|
|
|
|
import (
|
|
"encoding/json"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"banger/internal/runtimebundle"
|
|
)
|
|
|
|
func TestResolveRuntimeDirPrefersEnv(t *testing.T) {
|
|
t.Setenv("BANGER_RUNTIME_DIR", "/env/runtime")
|
|
|
|
if got := ResolveRuntimeDir("/config/runtime", "/deprecated/repo"); got != "/env/runtime" {
|
|
t.Fatalf("ResolveRuntimeDir() = %q, want /env/runtime", got)
|
|
}
|
|
}
|
|
|
|
func TestResolveRuntimeDirUsesInstalledLayout(t *testing.T) {
|
|
root := t.TempDir()
|
|
runtimeDir := filepath.Join(root, "lib", "banger")
|
|
createRuntimeBundle(t, runtimeDir)
|
|
|
|
origExecutablePath := executablePath
|
|
executablePath = func() (string, error) {
|
|
return filepath.Join(root, "bin", "banger"), nil
|
|
}
|
|
t.Cleanup(func() {
|
|
executablePath = origExecutablePath
|
|
})
|
|
|
|
if got := ResolveRuntimeDir("", ""); got != runtimeDir {
|
|
t.Fatalf("ResolveRuntimeDir() = %q, want %q", got, runtimeDir)
|
|
}
|
|
}
|
|
|
|
func TestResolveRuntimeDirUsesSourceCheckoutRuntimeSubdir(t *testing.T) {
|
|
root := t.TempDir()
|
|
runtimeDir := filepath.Join(root, "runtime")
|
|
createRuntimeBundle(t, runtimeDir)
|
|
|
|
origExecutablePath := executablePath
|
|
executablePath = func() (string, error) {
|
|
return filepath.Join(root, "banger"), nil
|
|
}
|
|
t.Cleanup(func() {
|
|
executablePath = origExecutablePath
|
|
})
|
|
|
|
if got := ResolveRuntimeDir("", ""); got != runtimeDir {
|
|
t.Fatalf("ResolveRuntimeDir() = %q, want %q", got, runtimeDir)
|
|
}
|
|
}
|
|
|
|
func createRuntimeBundle(t *testing.T, runtimeDir string) {
|
|
t.Helper()
|
|
metadata := runtimebundle.BundleMetadata{
|
|
FirecrackerBin: "bin/firecracker",
|
|
SSHKeyPath: "keys/id_ed25519",
|
|
NamegenPath: "bin/namegen",
|
|
CustomizeScript: "scripts/customize.sh",
|
|
VSockAgentPath: "bin/banger-vsock-agent",
|
|
DefaultPackages: "config/packages.apt",
|
|
DefaultRootfs: "images/rootfs-docker.ext4",
|
|
DefaultKernel: "kernels/vmlinux",
|
|
}
|
|
for _, rel := range []string{
|
|
metadata.FirecrackerBin,
|
|
metadata.SSHKeyPath,
|
|
metadata.NamegenPath,
|
|
metadata.CustomizeScript,
|
|
metadata.VSockAgentPath,
|
|
metadata.DefaultPackages,
|
|
metadata.DefaultRootfs,
|
|
metadata.DefaultKernel,
|
|
} {
|
|
path := filepath.Join(runtimeDir, rel)
|
|
if err := os.MkdirAll(filepath.Dir(path), 0o755); err != nil {
|
|
t.Fatalf("mkdir %s: %v", filepath.Dir(path), err)
|
|
}
|
|
if err := os.WriteFile(path, []byte("test"), 0o644); err != nil {
|
|
t.Fatalf("write %s: %v", path, err)
|
|
}
|
|
}
|
|
data, err := json.Marshal(metadata)
|
|
if err != nil {
|
|
t.Fatalf("Marshal: %v", err)
|
|
}
|
|
if err := os.WriteFile(filepath.Join(runtimeDir, runtimebundle.BundleMetadataFile), data, 0o644); err != nil {
|
|
t.Fatalf("write bundle metadata: %v", err)
|
|
}
|
|
}
|