Commit 422f2574 authored by Mitchell Hashimoto's avatar Mitchell Hashimoto

builder/vmware: refactor the platform-specific stuff out

/cc @rasa - This is more idiomatic Go: to share the platform-independent
stuff and make the platform-specific stuff unexported functions
parent d7140cd9
// +build darwin freebsd linux netbsd openbsd
package vmware package vmware
import ( import (
"bytes" "bytes"
"errors"
"fmt" "fmt"
"log" "log"
"os/exec" "os/exec"
...@@ -88,64 +85,39 @@ func (d *Workstation9Driver) Stop(vmxPath string) error { ...@@ -88,64 +85,39 @@ func (d *Workstation9Driver) Stop(vmxPath string) error {
} }
func (d *Workstation9Driver) Verify() error { func (d *Workstation9Driver) Verify() error {
if err := d.findApp(); err != nil { var err error
return fmt.Errorf("VMware Workstation application ('vmware') not found in path.") if d.AppPath == "" {
if d.AppPath, err = workstationFindVMware(); err != nil {
return err
}
} }
if err := d.findVmrun(); err != nil { if d.VmrunPath == "" {
return fmt.Errorf("Required application 'vmrun' not found in path.") if d.VmrunPath, err = workstationFindVmrun(); err != nil {
return err
}
} }
if err := d.findVdiskManager(); err != nil { if d.VdiskManagerPath == "" {
return fmt.Errorf("Required application 'vmware-vdiskmanager' not found in path.") if d.VdiskManagerPath, err = workstationFindVdiskManager(); err != nil {
return err
}
} }
// Check to see if it APPEARS to be licensed. // Check to see if it APPEARS to be licensed.
matches, err := filepath.Glob("/etc/vmware/license-*") if err := workstationCheckLicense(); err != nil {
if err != nil {
return fmt.Errorf("Error looking for VMware license: %s", err)
}
if len(matches) == 0 {
return errors.New("Workstation does not appear to be licensed. Please license it.")
}
return nil
}
func (d *Workstation9Driver) findApp() error {
path, err := exec.LookPath("vmware")
if err != nil {
return err
}
d.AppPath = path
return nil
}
func (d *Workstation9Driver) findVdiskManager() error {
path, err := exec.LookPath("vmware-vdiskmanager")
if err != nil {
return err return err
} }
d.VdiskManagerPath = path
return nil
}
func (d *Workstation9Driver) findVmrun() error {
path, err := exec.LookPath("vmrun")
if err != nil {
return err
}
d.VmrunPath = path
return nil return nil
} }
func (d *Workstation9Driver) ToolsIsoPath(flavor string) string { func (d *Workstation9Driver) ToolsIsoPath(flavor string) string {
return "/usr/lib/vmware/isoimages/" + flavor + ".iso" return workstationToolsIsoPath(flavor)
} }
func (d *Workstation9Driver) DhcpLeasesPath(device string) string { func (d *Workstation9Driver) DhcpLeasesPath(device string) string {
return "/etc/vmware/" + device + "/dhcpd/dhcpd.leases" return workstationDhcpLeasesPath(device)
} }
func (d *Workstation9Driver) runAndLog(cmd *exec.Cmd) (string, string, error) { func (d *Workstation9Driver) runAndLog(cmd *exec.Cmd) (string, string, error) {
......
// +build !windows
package vmware
import (
"errors"
"fmt"
"os/exec"
"path/filepath"
)
func workstationCheckLicense() error {
matches, err := filepath.Glob("/etc/vmware/license-*")
if err != nil {
return fmt.Errorf("Error looking for VMware license: %s", err)
}
if len(matches) == 0 {
return errors.New("Workstation does not appear to be licensed. Please license it.")
}
return nil
}
func workstationFindVdiskManager() (string, error) {
return exec.LookPath("vmware-vdiskmanager")
}
func workstationFindVMware() (string, error) {
return exec.LookPath("vmware")
}
func workstationFindVmrun() (string, error) {
return exec.LookPath("vmrun")
}
func workstationDhcpLeasesPath(device string) string {
return "/etc/vmware/" + device + "/dhcpd/dhcpd.leases"
}
func workstationToolsIsoPath(flavor string) string {
return "/usr/lib/vmware/isoimages/" + flavor + ".iso"
}
// +build windows // +build windows
// Contributed by Ross Smith II (smithii.com)
package vmware package vmware
import ( import (
"bytes"
"fmt"
"log" "log"
"os" "os"
"os/exec" "os/exec"
...@@ -15,196 +12,74 @@ import ( ...@@ -15,196 +12,74 @@ import (
"unsafe" "unsafe"
) )
// Workstation9Driver is a driver that can run VMware Workstation 9 func workstationCheckLicense() error {
// on Windows. // Not implemented on Windows
type Workstation9Driver struct {
AppPath string
VdiskManagerPath string
VmrunPath string
}
func (d *Workstation9Driver) CompactDisk(diskPath string) error {
defragCmd := exec.Command(d.VdiskManagerPath, "-d", diskPath)
if _, _, err := d.runAndLog(defragCmd); err != nil {
return err
}
shrinkCmd := exec.Command(d.VdiskManagerPath, "-k", diskPath)
if _, _, err := d.runAndLog(shrinkCmd); err != nil {
return err
}
return nil return nil
} }
func (d *Workstation9Driver) CreateDisk(output string, size string) error { func workstationFindVdiskManager() (string, error) {
cmd := exec.Command(d.VdiskManagerPath, "-c", "-s", size, "-a", "lsilogic", "-t", "1", output) path, err := exec.LookPath("vmware-vdiskmanager.exe")
if _, _, err := d.runAndLog(cmd); err != nil { if err == nil {
return err return path, nil
}
return nil
}
func (d *Workstation9Driver) IsRunning(vmxPath string) (bool, error) {
vmxPath, err := filepath.Abs(vmxPath)
if err != nil {
return false, err
} }
cmd := exec.Command(d.VmrunPath, "-T", "ws", "list") path, err = workstationVMwareRoot()
stdout, _, err := d.runAndLog(cmd)
if err != nil { if err != nil {
return false, err return "", err
}
for _, line := range strings.Split(stdout, "\n") {
if line == vmxPath {
return true, nil
}
} }
return false, nil return filepath.Join(path, "vmware-vdiskmanager.exe"), nil
} }
func (d *Workstation9Driver) Start(vmxPath string, headless bool) error { func workstationFindVMware() (string, error) {
guiArgument := "gui" path, err := exec.LookPath("vmware.exe")
if headless { if err == nil {
guiArgument = "nogui" return path, nil
}
cmd := exec.Command(d.VmrunPath, "-T", "ws", "start", vmxPath, guiArgument)
if _, _, err := d.runAndLog(cmd); err != nil {
return err
}
return nil
}
func (d *Workstation9Driver) Stop(vmxPath string) error {
cmd := exec.Command(d.VmrunPath, "-T", "ws", "stop", vmxPath, "hard")
if _, _, err := d.runAndLog(cmd); err != nil {
return err
}
return nil
}
func (d *Workstation9Driver) Verify() error {
if err := d.findApp(); err != nil {
return fmt.Errorf("VMware Workstation application ('vmware') not found in path.")
}
if err := d.findVmrun(); err != nil {
return fmt.Errorf("Required application 'vmrun' not found in path.")
}
if err := d.findVdiskManager(); err != nil {
return fmt.Errorf("Required application 'vmware-vdiskmanager' not found in path.")
} }
// Check to see if it APPEARS to be licensed. path, err = workstationVMwareRoot()
/*
matches, err := filepath.Glob("/etc/vmware/license-*")
if err != nil {
return fmt.Errorf("Error looking for VMware license: %s", err)
}
if len(matches) == 0 {
return errors.New("Workstation does not appear to be licensed. Please license it.")
}
*/
return nil
}
func (d *Workstation9Driver) findApp() error {
path, err := exec.LookPath("vmware.exe")
if err != nil { if err != nil {
path, err := getVmwarePath() return "", err
if err != nil {
return err
}
path += "vmware.exe"
} }
path = strings.Replace(path, "\\", "/", -1)
log.Printf("Using '%s' for vmware path", path)
d.AppPath = path
return nil return filepath.Join(path, "vmware.exe"), nil
} }
func (d *Workstation9Driver) findVdiskManager() error { func workstationFindVmrun() (string, error) {
path, err := exec.LookPath("vmware-vdiskmanager.exe") path, err := exec.LookPath("vmrun.exe")
if err != nil { if err == nil {
path, err := getVmwarePath() return path, nil
if err != nil {
return err
}
path += "vmware-vdiskmanager.exe"
} }
path = strings.Replace(path, "\\", "/", -1)
log.Printf("Using '%s' for vmware-vdiskmanager path", path)
d.VdiskManagerPath = path
return nil
}
func (d *Workstation9Driver) findVmrun() error { path, err = workstationVMwareRoot()
path, err := exec.LookPath("vmrun.exe")
if err != nil { if err != nil {
path, err := getVmwarePath() return "", err
if err != nil {
return err
}
path += "vmrun.exe"
} }
path = strings.Replace(path, "\\", "/", -1)
log.Printf("Using '%s' for vmrun path", path) return filepath.Join(path, "vmrun.exe"), nil
d.VmrunPath = path
return nil
} }
func (d *Workstation9Driver) ToolsIsoPath(flavor string) string { func workstationToolsIsoPath(flavor string) string {
path, err := getVmwarePath() path, err := workstationVMwareRoot()
if err != nil { if err != nil {
return "" return ""
} else {
return path + flavor + ".iso"
} }
return filepath.Join(path, flavor+".iso")
} }
func (d *Workstation9Driver) DhcpLeasesPath(device string) string { func workstationDhcpLeasesPath(device string) string {
programData := os.Getenv("ProgramData") programData := os.Getenv("ProgramData")
rv := programData + "/VMware/vmnetdhcp.leases" if programData == "" {
if _, err := os.Stat(rv); os.IsNotExist(err) {
log.Printf("File not found: '%s' (found '%s' in %%ProgramData%%)", rv, programData)
return "" return ""
} }
return rv
}
func (d *Workstation9Driver) runAndLog(cmd *exec.Cmd) (string, string, error) {
var stdout, stderr bytes.Buffer
log.Printf("Executing: %s %v", cmd.Path, cmd.Args[1:])
cmd.Stdout = &stdout
cmd.Stderr = &stderr
err := cmd.Run()
stdoutString := strings.TrimSpace(stdout.String())
stderrString := strings.TrimSpace(stderr.String())
if _, ok := err.(*exec.ExitError); ok {
err = fmt.Errorf("VMware error: %s", stderrString)
}
log.Printf("stdout: %s", stdoutString) return filepath.Join(programData, "/VMware/vmnetdhcp.leases")
log.Printf("stderr: %s", stderrString)
return stdout.String(), stderr.String(), err
} }
// see http://blog.natefinch.com/2012/11/go-win-stuff.html // See http://blog.natefinch.com/2012/11/go-win-stuff.html
//
// This is used by workstationVMwareRoot in order to read some registry data.
func readRegString(hive syscall.Handle, subKeyPath, valueName string) (value string, err error) { func readRegString(hive syscall.Handle, subKeyPath, valueName string) (value string, err error) {
var h syscall.Handle var h syscall.Handle
err = syscall.RegOpenKeyEx(hive, syscall.StringToUTF16Ptr(subKeyPath), 0, syscall.KEY_READ, &h) err = syscall.RegOpenKeyEx(hive, syscall.StringToUTF16Ptr(subKeyPath), 0, syscall.KEY_READ, &h)
...@@ -215,7 +90,6 @@ func readRegString(hive syscall.Handle, subKeyPath, valueName string) (value str ...@@ -215,7 +90,6 @@ func readRegString(hive syscall.Handle, subKeyPath, valueName string) (value str
var typ uint32 var typ uint32
var bufSize uint32 var bufSize uint32
err = syscall.RegQueryValueEx( err = syscall.RegQueryValueEx(
h, h,
syscall.StringToUTF16Ptr(valueName), syscall.StringToUTF16Ptr(valueName),
...@@ -228,7 +102,6 @@ func readRegString(hive syscall.Handle, subKeyPath, valueName string) (value str ...@@ -228,7 +102,6 @@ func readRegString(hive syscall.Handle, subKeyPath, valueName string) (value str
} }
data := make([]uint16, bufSize/2+1) data := make([]uint16, bufSize/2+1)
err = syscall.RegQueryValueEx( err = syscall.RegQueryValueEx(
h, h,
syscall.StringToUTF16Ptr(valueName), syscall.StringToUTF16Ptr(valueName),
...@@ -243,15 +116,16 @@ func readRegString(hive syscall.Handle, subKeyPath, valueName string) (value str ...@@ -243,15 +116,16 @@ func readRegString(hive syscall.Handle, subKeyPath, valueName string) (value str
return syscall.UTF16ToString(data), nil return syscall.UTF16ToString(data), nil
} }
func getVmwarePath() (s string, e error) { // This reads the VMware installation path from the Windows registry.
key := "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\App Paths\\vmware.exe" func workstationVMwareRoot() (s string, err error) {
key := `SOFTWARE\Microsoft\Windows\CurrentVersion\App Paths\vmware.exe`
subkey := "Path" subkey := "Path"
s, e = readRegString(syscall.HKEY_LOCAL_MACHINE, key, subkey) s, err = readRegString(syscall.HKEY_LOCAL_MACHINE, key, subkey)
if e != nil { if err != nil {
log.Printf("Unable to read registry key %s\\%s", key, subkey) log.Printf(`Unable to read registry key %s\%s`, key, subkey)
return "", e return
} }
log.Printf("Found '%s' in registry key %s\\%s", s, key, subkey)
s = strings.Replace(s, "\\", "/", -1) s = strings.Replace(s, "\\", "/", -1)
return s, nil return
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment