255 lines
5.8 KiB
Go
255 lines
5.8 KiB
Go
|
// +build windows
|
||
|
// Contributed by Ross Smith II (smithii.com)
|
||
|
|
||
|
package vmware
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"fmt"
|
||
|
"log"
|
||
|
"os"
|
||
|
"os/exec"
|
||
|
"path/filepath"
|
||
|
"strings"
|
||
|
"unsafe"
|
||
|
"syscall"
|
||
|
)
|
||
|
|
||
|
// Workstation9Driver is a driver that can run VMware Workstation 9
|
||
|
// 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
|
||
|
}
|
||
|
|
||
|
func (d *Workstation9Driver) CreateDisk(output string, size string) error {
|
||
|
cmd := exec.Command(d.VdiskManagerPath, "-c", "-s", size, "-a", "lsilogic", "-t", "1", output)
|
||
|
if _, _, err := d.runAndLog(cmd); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
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")
|
||
|
stdout, _, err := d.runAndLog(cmd)
|
||
|
if err != nil {
|
||
|
return false, err
|
||
|
}
|
||
|
|
||
|
for _, line := range strings.Split(stdout, "\n") {
|
||
|
if line == vmxPath {
|
||
|
return true, nil
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false, nil
|
||
|
}
|
||
|
|
||
|
func (d *Workstation9Driver) Start(vmxPath string, headless bool) error {
|
||
|
guiArgument := "gui"
|
||
|
if headless {
|
||
|
guiArgument = "nogui"
|
||
|
}
|
||
|
|
||
|
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.
|
||
|
/*
|
||
|
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 {
|
||
|
path, err := getVmwarePath()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
path += "vmware.exe"
|
||
|
}
|
||
|
log.Printf("Using '%s' for vmware path", path)
|
||
|
d.AppPath = path
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d *Workstation9Driver) findVdiskManager() error {
|
||
|
path, err := exec.LookPath("vmware-vdiskmanager.exe")
|
||
|
if err != nil {
|
||
|
path, err := getVmwarePath()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
path += "vmware-vdiskmanager.exe"
|
||
|
}
|
||
|
log.Printf("Using '%s' for vmware-vdiskmanager path", path)
|
||
|
d.VdiskManagerPath = path
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d *Workstation9Driver) findVmrun() error {
|
||
|
path, err := exec.LookPath("vmrun.exe")
|
||
|
if err != nil {
|
||
|
path, err := getVmwarePath()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
path += "vmrun.exe"
|
||
|
}
|
||
|
log.Printf("Using '%s' for vmrun path", path)
|
||
|
d.VmrunPath = path
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d *Workstation9Driver) ToolsIsoPath(flavor string) string {
|
||
|
path, err := getVmwarePath()
|
||
|
if err != nil {
|
||
|
return ""
|
||
|
} else {
|
||
|
return path + flavor + ".iso"
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (d *Workstation9Driver) DhcpLeasesPath(device string) string {
|
||
|
programData := os.Getenv("ProgramData")
|
||
|
rv := programData + "/VMware/vmnetdhcp.leases"
|
||
|
if _, err := os.Stat(rv); os.IsNotExist(err) {
|
||
|
log.Printf("File not found: '%s' (found '%s' in %%ProgramData%%)", rv, programData)
|
||
|
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)
|
||
|
log.Printf("stderr: %s", stderrString)
|
||
|
|
||
|
return stdout.String(), stderr.String(), err
|
||
|
}
|
||
|
|
||
|
// see http://blog.natefinch.com/2012/11/go-win-stuff.html
|
||
|
|
||
|
func readRegString(hive syscall.Handle, subKeyPath, valueName string) (value string, err error) {
|
||
|
var h syscall.Handle
|
||
|
err = syscall.RegOpenKeyEx(hive, syscall.StringToUTF16Ptr(subKeyPath), 0, syscall.KEY_READ, &h)
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
defer syscall.RegCloseKey(h)
|
||
|
|
||
|
var typ uint32
|
||
|
var bufSize uint32
|
||
|
|
||
|
err = syscall.RegQueryValueEx(
|
||
|
h,
|
||
|
syscall.StringToUTF16Ptr(valueName),
|
||
|
nil,
|
||
|
&typ,
|
||
|
nil,
|
||
|
&bufSize)
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
data := make([]uint16, bufSize/2+1)
|
||
|
|
||
|
err = syscall.RegQueryValueEx(
|
||
|
h,
|
||
|
syscall.StringToUTF16Ptr(valueName),
|
||
|
nil,
|
||
|
&typ,
|
||
|
(*byte)(unsafe.Pointer(&data[0])),
|
||
|
&bufSize)
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
return syscall.UTF16ToString(data), nil
|
||
|
}
|
||
|
|
||
|
func getVmwarePath() (s string, e error) {
|
||
|
key := "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\App Paths\\vmware.exe"
|
||
|
subkey := "Path"
|
||
|
s, e = readRegString(syscall.HKEY_LOCAL_MACHINE, key, subkey)
|
||
|
if e != nil {
|
||
|
log.Printf("Unable to read registry key %s\\%s", key, subkey)
|
||
|
return "", e
|
||
|
}
|
||
|
log.Printf("Found '%s' in registry key %s\\%s", s, key, subkey)
|
||
|
s = strings.Replace(s, "\\", "/", -1)
|
||
|
return s, nil
|
||
|
}
|