builder/vmware: Add support for Workstation 9 (Linux).

Now the VMware builder should automatically pick between Fusion 5 and
Workstation 9, based on which one is installed.
This commit is contained in:
Julian Phillips 2013-07-01 20:17:47 +01:00 committed by Mitchell Hashimoto
parent a179220a84
commit 4846d252a3
5 changed files with 179 additions and 4 deletions

View File

@ -359,7 +359,7 @@ func (b *Builder) Cancel() {
} }
} }
func (b *Builder) newDriver() (Driver, error) { func (b *Builder) newFusionDriver() (Driver, error) {
fusionAppPath := "/Applications/VMware Fusion.app" fusionAppPath := "/Applications/VMware Fusion.app"
driver := &Fusion5Driver{fusionAppPath} driver := &Fusion5Driver{fusionAppPath}
if err := driver.Verify(); err != nil { if err := driver.Verify(); err != nil {
@ -368,3 +368,27 @@ func (b *Builder) newDriver() (Driver, error) {
return driver, nil return driver, nil
} }
func (b *Builder) newWorkstationLinuxDriver() (Driver, error) {
driver := &WS9LnxDriver{}
if err := driver.Verify(); err != nil {
return nil, err
}
return driver, nil
}
func (b *Builder) newDriver() (Driver, error) {
fusion, fusionErr := b.newFusionDriver()
ws, wsErr := b.newWorkstationLinuxDriver()
if fusionErr == nil {
return fusion, nil
}
if wsErr == nil {
return ws, nil
}
return nil, fmt.Errorf("Unable to initialise VMware driver:\nFusion 5: %s\nWorkstation 9: %s", fusionErr, wsErr)
}

View File

@ -30,6 +30,9 @@ type Driver interface {
// Get the path to the VMware ISO for the given flavor. // Get the path to the VMware ISO for the given flavor.
ToolsIsoPath(string) string ToolsIsoPath(string) string
// Get the path to the DHCP leases file for the given device.
DhcpLeasesPath(string) string
// Verify checks to make sure that this driver should function // Verify checks to make sure that this driver should function
// properly. This should check that all the files it will use // properly. This should check that all the files it will use
// appear to exist and so on. If everything is okay, this doesn't // appear to exist and so on. If everything is okay, this doesn't
@ -150,6 +153,10 @@ func (d *Fusion5Driver) ToolsIsoPath(k string) string {
return filepath.Join(d.AppPath, "Contents", "Library", "isoimages", k+".iso") return filepath.Join(d.AppPath, "Contents", "Library", "isoimages", k+".iso")
} }
func (d *Fusion5Driver) DhcpLeasesPath(device string) string {
return "/etc/vmware/vmnet-dhcpd-" + device + ".leases"
}
func (d *Fusion5Driver) runAndLog(cmd *exec.Cmd) (string, string, error) { func (d *Fusion5Driver) runAndLog(cmd *exec.Cmd) (string, string, error) {
var stdout, stderr bytes.Buffer var stdout, stderr bytes.Buffer
@ -163,3 +170,143 @@ func (d *Fusion5Driver) runAndLog(cmd *exec.Cmd) (string, string, error) {
return stdout.String(), stderr.String(), err return stdout.String(), stderr.String(), err
} }
// WS9LnxDriver is a driver that can run VMware Workstation 9 on Linux.
type WS9LnxDriver struct {
// These are paths to useful VMware Workstation binaries
AppPath string
VdiskManagerPath string
VmrunPath string
}
func (d *WS9LnxDriver) 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 *WS9LnxDriver) 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 *WS9LnxDriver) 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 *WS9LnxDriver) 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 *WS9LnxDriver) 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 *WS9LnxDriver) 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("Critical application 'vmrun' not found in path.")
}
if err := d.findVdiskManager(); err != nil {
return fmt.Errorf("Critical application 'vmware-vdiskmanager' not found in path.")
}
return nil
}
func (d *WS9LnxDriver) findApp() error {
path, err := exec.LookPath("vmware")
if err != nil {
return err
}
d.AppPath = path
return nil
}
func (d *WS9LnxDriver) findVdiskManager() error {
path, err := exec.LookPath("vmware-vdiskmanager")
if err != nil {
return err
}
d.VdiskManagerPath = path
return nil
}
func (d *WS9LnxDriver) findVmrun() error {
path, err := exec.LookPath("vmrun")
if err != nil {
return err
}
d.VmrunPath = path
return nil
}
func (d *WS9LnxDriver) ToolsIsoPath(flavor string) string {
return "/usr/lib/vmware/isoimages/" + flavor + ".iso"
}
func (d *WS9LnxDriver) DhcpLeasesPath(device string) string {
return "/etc/vmware/" + device + "/dhcpd/dhcpd.leases"
}
func (d *WS9LnxDriver) 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()
log.Printf("stdout: %s", strings.TrimSpace(stdout.String()))
log.Printf("stderr: %s", strings.TrimSpace(stderr.String()))
return stdout.String(), stderr.String(), err
}

View File

@ -2,7 +2,6 @@ package vmware
import ( import (
"errors" "errors"
"fmt"
"io/ioutil" "io/ioutil"
"os" "os"
"regexp" "regexp"
@ -18,6 +17,9 @@ type GuestIPFinder interface {
// DHCPLeaseGuestLookup looks up the IP address of a guest using DHCP // DHCPLeaseGuestLookup looks up the IP address of a guest using DHCP
// lease information from the VMware network devices. // lease information from the VMware network devices.
type DHCPLeaseGuestLookup struct { type DHCPLeaseGuestLookup struct {
// Driver that is being used (to find leases path)
Driver Driver
// Device that the guest is connected to. // Device that the guest is connected to.
Device string Device string
@ -26,7 +28,7 @@ type DHCPLeaseGuestLookup struct {
} }
func (f *DHCPLeaseGuestLookup) GuestIP() (string, error) { func (f *DHCPLeaseGuestLookup) GuestIP() (string, error) {
fh, err := os.Open(fmt.Sprintf("/var/db/vmware/vmnet-dhcpd-%s.leases", f.Device)) fh, err := os.Open(f.Driver.DhcpLeasesPath(f.Device))
if err != nil { if err != nil {
return "", err return "", err
} }

View File

@ -33,7 +33,7 @@ func (f *IfconfigIPFinder) HostIP() (string, error) {
return "", err return "", err
} }
re := regexp.MustCompile(`inet\s*(.+?)\s`) re := regexp.MustCompile(`inet\s*(?:addr:)?(.+?)\s`)
matches := re.FindStringSubmatch(stdout.String()) matches := re.FindStringSubmatch(stdout.String())
if matches == nil { if matches == nil {
return "", errors.New("IP not found in ifconfig output...") return "", errors.New("IP not found in ifconfig output...")

View File

@ -12,6 +12,7 @@ import (
func sshAddress(state map[string]interface{}) (string, error) { func sshAddress(state map[string]interface{}) (string, error) {
config := state["config"].(*config) config := state["config"].(*config)
driver := state["driver"].(Driver)
vmxPath := state["vmx_path"].(string) vmxPath := state["vmx_path"].(string)
log.Println("Lookup up IP information...") log.Println("Lookup up IP information...")
@ -37,6 +38,7 @@ func sshAddress(state map[string]interface{}) (string, error) {
} }
ipLookup := &DHCPLeaseGuestLookup{ ipLookup := &DHCPLeaseGuestLookup{
Driver: driver,
Device: "vmnet8", Device: "vmnet8",
MACAddress: macAddress, MACAddress: macAddress,
} }