package main import ( "bytes" "flag" "fmt" "log" "os" "os/exec" "path/filepath" "strings" "time" ) const ( workbench = `build` ) var ( key = "" repo = "" commit = "" task = "" image = "" port = "" ) func init() { flag.StringVar(&key, "key", "", "Path to SSH private key") flag.StringVar(&repo, "repo", "", "Source code as valid Git URL") flag.StringVar(&commit, "commit", "", "Commit or commit-ish reference for checkout") flag.StringVar(&task, "task", "", "Shell code to execute for the build") flag.StringVar(&image, "image", "", "QEMU qcow2 image which contains the VM") flag.StringVar(&port, "port", "", "localhost port which is connected to VM SSH server") } func main() { flag.Parse() log.SetFlags(0) log.Println("Starting craft") defer log.Println("Exiting craft") source, err := os.MkdirTemp("", "*-craft") if err != nil { log.Fatal(err) } defer os.RemoveAll(source) checkout(repo, commit, source) vm := qemu() err = vm.Start() if err != nil { log.Fatal(err) } defer func() { err := vm.Process.Kill() if err != nil { log.Fatal(err) } }() waitBoot() upload(source) craft() } func qemu() *exec.Cmd { return exec.Command( "qemu-system-x86_64", "-enable-kvm", "-m", "4G", "-net", "nic,model=virtio", "-net", fmt.Sprintf( "user,hostfwd=tcp:127.0.0.1:%s-:22", port, ), "-drive", fmt.Sprintf( "file=%s,media=disk,snapshot=on,if=virtio", image, ), "-smp", "cpus=4", "-nographic", ) } func sshCommand(name string, arg ...string) *exec.Cmd { return exec.Command( "ssh", append([]string{ "-p", port, "-o", "UserKnownHostsFile=/dev/null", "-o", "StrictHostKeyChecking=no", "-i", key, "root@localhost", name, }, arg...)..., ) } func waitBoot() { retries := 5 initialDelay := 1 * time.Second log.Println("Waiting for VM to boot") time.Sleep(initialDelay) // without that `ssh` returns too quickly for range retries { cmd := sshCommand("true") err := cmd.Run() if err == nil { log.Println("VM booted") return } } log.Fatalf("Could not reach VM %d times - giving up", retries) } func craft() { script := fmt.Sprintf(`#!/bin/sh cd %s %s `, workbench, task) cmd := sshCommand("sh") cmd.Stdin = bytes.NewBufferString(script) cmd.Stdout = os.Stdout cmd.Stderr = os.Stderr err := cmd.Run() if err != nil { log.Fatal(err) } } func runCommand(name string, args ...string) { log.Printf("Execution: %s %s\n", name, strings.Join(args, " ")) command := exec.Command(name, args...) command.Stderr = os.Stderr command.Stdout = os.Stdout err := command.Run() if err != nil { log.Fatal(err) } } func checkout(repo string, commit string, target string) { runCommand( "git", "clone", repo, target, ) runCommand( "git", "--git-dir", filepath.Join(target, ".git"), "--work-tree", target, "checkout", commit, ) runCommand( "git", "-C", target, "submodule", "update", "--depth=1", "--init", "--recursive", ) log.Printf("Checked source out to '%s'", target) } func upload(directory string) { mkdir := sshCommand("mkdir", "/root/build") err := mkdir.Run() if err != nil { log.Fatal(err) } runCommand( "rsync", "-a", filepath.Clean(directory)+"/", "--rsh", "ssh -p "+port+" -o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no -i "+key, "root@localhost:/root/build/", ) }