2013-07-16 17:23:01 -04:00
|
|
|
package vmware
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
2013-07-31 15:39:49 -04:00
|
|
|
"os"
|
2013-07-16 17:23:01 -04:00
|
|
|
"os/exec"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
// Workstation9Driver is a driver that can run VMware Workstation 9
|
|
|
|
// on non-Windows platforms.
|
|
|
|
type Workstation9Driver struct {
|
2013-07-16 17:23:01 -04:00
|
|
|
AppPath string
|
|
|
|
VdiskManagerPath string
|
|
|
|
VmrunPath string
|
|
|
|
}
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
func (d *Workstation9Driver) CompactDisk(diskPath string) error {
|
2013-07-16 17:23:01 -04:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
func (d *Workstation9Driver) CreateDisk(output string, size string) error {
|
2013-07-16 17:23:01 -04:00
|
|
|
cmd := exec.Command(d.VdiskManagerPath, "-c", "-s", size, "-a", "lsilogic", "-t", "1", output)
|
|
|
|
if _, _, err := d.runAndLog(cmd); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
func (d *Workstation9Driver) IsRunning(vmxPath string) (bool, error) {
|
2013-07-16 17:23:01 -04:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
func (d *Workstation9Driver) Start(vmxPath string, headless bool) error {
|
2013-07-16 17:23:01 -04:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
func (d *Workstation9Driver) Stop(vmxPath string) error {
|
2013-07-16 17:23:01 -04:00
|
|
|
cmd := exec.Command(d.VmrunPath, "-T", "ws", "stop", vmxPath, "hard")
|
|
|
|
if _, _, err := d.runAndLog(cmd); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
func (d *Workstation9Driver) Verify() error {
|
2013-07-31 15:36:17 -04:00
|
|
|
var err error
|
|
|
|
if d.AppPath == "" {
|
|
|
|
if d.AppPath, err = workstationFindVMware(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2013-07-16 17:23:01 -04:00
|
|
|
}
|
|
|
|
|
2013-07-31 15:36:17 -04:00
|
|
|
if d.VmrunPath == "" {
|
|
|
|
if d.VmrunPath, err = workstationFindVmrun(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2013-07-16 17:23:01 -04:00
|
|
|
}
|
|
|
|
|
2013-07-31 15:36:17 -04:00
|
|
|
if d.VdiskManagerPath == "" {
|
|
|
|
if d.VdiskManagerPath, err = workstationFindVdiskManager(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2013-07-16 17:23:01 -04:00
|
|
|
}
|
|
|
|
|
2013-07-31 16:50:49 -04:00
|
|
|
log.Printf("VMware app path: %s", d.AppPath)
|
|
|
|
log.Printf("vmrun path: %s", d.VmrunPath)
|
|
|
|
log.Printf("vdisk-manager path: %s", d.VdiskManagerPath)
|
|
|
|
|
2013-07-31 15:39:49 -04:00
|
|
|
if _, err := os.Stat(d.AppPath); err != nil {
|
|
|
|
return fmt.Errorf("VMware application not found: %s", d.AppPath)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := os.Stat(d.VmrunPath); err != nil {
|
|
|
|
return fmt.Errorf("'vmrun' application not found: %s", d.VmrunPath)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := os.Stat(d.VdiskManagerPath); err != nil {
|
|
|
|
return fmt.Errorf("'vmrun' application not found: %s", d.VdiskManagerPath)
|
|
|
|
}
|
|
|
|
|
2013-07-16 17:58:16 -04:00
|
|
|
// Check to see if it APPEARS to be licensed.
|
2013-07-31 15:36:17 -04:00
|
|
|
if err := workstationCheckLicense(); err != nil {
|
2013-07-16 17:23:01 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
func (d *Workstation9Driver) ToolsIsoPath(flavor string) string {
|
2013-07-31 15:36:17 -04:00
|
|
|
return workstationToolsIsoPath(flavor)
|
2013-07-16 17:23:01 -04:00
|
|
|
}
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
func (d *Workstation9Driver) DhcpLeasesPath(device string) string {
|
2013-07-31 15:36:17 -04:00
|
|
|
return workstationDhcpLeasesPath(device)
|
2013-07-16 17:23:01 -04:00
|
|
|
}
|
|
|
|
|
2013-07-27 16:00:21 -04:00
|
|
|
func (d *Workstation9Driver) runAndLog(cmd *exec.Cmd) (string, string, error) {
|
2013-07-16 17:23:01 -04:00
|
|
|
var stdout, stderr bytes.Buffer
|
|
|
|
|
|
|
|
log.Printf("Executing: %s %v", cmd.Path, cmd.Args[1:])
|
|
|
|
cmd.Stdout = &stdout
|
|
|
|
cmd.Stderr = &stderr
|
|
|
|
err := cmd.Run()
|
|
|
|
|
2013-07-16 17:46:34 -04:00
|
|
|
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)
|
2013-07-16 17:23:01 -04:00
|
|
|
|
2013-07-31 16:58:57 -04:00
|
|
|
// Replace these for Windows, we only want to deal with Unix
|
|
|
|
// style line endings.
|
|
|
|
returnStdout := strings.Replace(stdout.String(), "\r\n", "\n", -1)
|
|
|
|
returnStderr := strings.Replace(stderr.String(), "\r\n", "\n", -1)
|
|
|
|
|
|
|
|
return returnStdout, returnStderr, err
|
2013-07-16 17:23:01 -04:00
|
|
|
}
|