tailscale/tstest/tailmac/Swift/Common/Config.swift
Jonathan Nobels 8fad8c4b9b
tstest/tailmac: add customized macOS virtualization tooling (#13146)
updates tailcale/corp#22371

Adds custom macOS vm tooling.  See the README for
the general gist, but this will spin up VMs with unixgram
capable network interfaces listening to a named socket,
and with a virtio socket device for host-guest communication.

We can add other devices like consoles, serial, etc as needed.

The whole things is buildable with a single make command, and
everything is controllable via the command line using the TailMac
utility.

This should all be generally functional but takes a few shortcuts
with error handling and the like.  The virtio socket device support
has not been tested and may require some refinement.

Signed-off-by: Jonathan Nobels <jonathan@tailscale.com>
2024-08-19 15:01:19 -04:00

126 lines
3.9 KiB
Swift

// Copyright (c) Tailscale Inc & AUTHORS
// SPDX-License-Identifier: BSD-3-Clause
import Foundation
let kDefaultDiskSizeGb: Int64 = 72
let kDefaultMemSizeGb: UInt64 = 72
/// Represents a configuration for a virtual machine
class Config: Codable {
var serverSocket = "/tmp/qemu-dgram.sock"
var memorySize = (kDefaultMemSizeGb * 1024 * 1024 * 1024) as UInt64
var mac = "52:cc:cc:cc:cc:01"
var ethermac = "52:cc:cc:cc:ce:01"
var port: UInt32 = 51009
// The virtual machines ID. Also double as the directory name under which
// we will store configuration, block device, etc.
let vmID: String
required init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
if let ethermac = try container.decodeIfPresent(String.self, forKey: .ethermac) {
self.ethermac = ethermac
}
if let serverSocket = try container.decodeIfPresent(String.self, forKey: .serverSocket) {
self.serverSocket = serverSocket
}
if let memorySize = try container.decodeIfPresent(UInt64.self, forKey: .memorySize) {
self.memorySize = memorySize
}
if let port = try container.decodeIfPresent(UInt32.self, forKey: .port) {
self.port = port
}
if let mac = try container.decodeIfPresent(String.self, forKey: .mac) {
self.mac = mac
}
if let vmID = try container.decodeIfPresent(String.self, forKey: .vmID) {
self.vmID = vmID
} else {
self.vmID = "default"
}
}
init(_ vmID: String = "default") {
self.vmID = vmID
let configFile = vmDataURL.appendingPathComponent("config.json")
if FileManager.default.fileExists(atPath: configFile.path()) {
print("Using config file at path \(configFile)")
if let jsonData = try? Data(contentsOf: configFile) {
let config = try! JSONDecoder().decode(Config.self, from: jsonData)
self.serverSocket = config.serverSocket
self.memorySize = config.memorySize
self.mac = config.mac
self.port = config.port
self.ethermac = config.ethermac
}
}
}
func persist() {
let configFile = vmDataURL.appendingPathComponent("config.json")
let data = try! JSONEncoder().encode(self)
try! data.write(to: configFile)
}
lazy var restoreImageURL: URL = {
vmBundleURL.appendingPathComponent("RestoreImage.ipsw")
}()
// The VM Data URL holds the specific files composing a unique VM guest instance
// By default, VM's are persisted at ~/VM.bundle/<vmID>
lazy var vmDataURL = {
let dataURL = vmBundleURL.appendingPathComponent(vmID)
return dataURL
}()
lazy var auxiliaryStorageURL = {
vmDataURL.appendingPathComponent("AuxiliaryStorage")
}()
lazy var diskImageURL = {
vmDataURL.appendingPathComponent("Disk.img")
}()
lazy var diskSize: Int64 = {
kDefaultDiskSizeGb * 1024 * 1024 * 1024
}()
lazy var hardwareModelURL = {
vmDataURL.appendingPathComponent("HardwareModel")
}()
lazy var machineIdentifierURL = {
vmDataURL.appendingPathComponent("MachineIdentifier")
}()
lazy var saveFileURL = {
vmDataURL.appendingPathComponent("SaveFile.vzvmsave")
}()
}
// The VM Bundle URL holds the restore image and a set of VM images
// By default, VM's are persisted at ~/VM.bundle
var vmBundleURL: URL = {
let vmBundlePath = NSHomeDirectory() + "/VM.bundle/"
createDir(vmBundlePath)
let bundleURL = URL(fileURLWithPath: vmBundlePath)
return bundleURL
}()
func createDir(_ path: String) {
do {
try FileManager.default.createDirectory(atPath: path, withIntermediateDirectories: true)
} catch {
fatalError("Unable to create dir at \(path) \(error)")
}
}