package vmware import ( "fmt" "github.com/mitchellh/multistep" "github.com/mitchellh/packer/packer" "time" ) // This step runs the created virtual machine. // // Uses: // config *config // driver Driver // ui packer.Ui // vmx_path string // // Produces: // type stepRun struct { bootTime time.Time vmxPath string registered bool } func (s *stepRun) Run(state multistep.StateBag) multistep.StepAction { config := state.Get("config").(*config) driver := state.Get("driver").(Driver) ui := state.Get("ui").(packer.Ui) vmxPath := state.Get("vmx_path").(string) vncIp := state.Get("vnc_ip").(string) vncPort := state.Get("vnc_port").(uint) // Set the VMX path so that we know we started the machine s.bootTime = time.Now() s.vmxPath = vmxPath ui.Say("Starting virtual machine...") if config.Headless { ui.Message(fmt.Sprintf( "The VM will be run headless, without a GUI. If you want to\n"+ "view the screen of the VM, connect via VNC without a password to\n"+ "%s:%d", vncIp, vncPort)) } if remoteDriver, ok := driver.(RemoteDriver); ok { if err := remoteDriver.Register(vmxPath); err != nil { err := fmt.Errorf("Error registering VM: %s", err) state.Put("error", err) ui.Error(err.Error()) return multistep.ActionHalt } s.registered = true } if err := driver.Start(vmxPath, config.Headless); err != nil { err := fmt.Errorf("Error starting VM: %s", err) state.Put("error", err) ui.Error(err.Error()) return multistep.ActionHalt } // Wait the wait amount if int64(config.bootWait) > 0 { ui.Say(fmt.Sprintf("Waiting %s for boot...", config.bootWait.String())) wait := time.After(config.bootWait) WAITLOOP: for { select { case <-wait: break WAITLOOP case <-time.After(1 * time.Second): if _, ok := state.GetOk(multistep.StateCancelled); ok { return multistep.ActionHalt } } } } return multistep.ActionContinue } func (s *stepRun) Cleanup(state multistep.StateBag) { driver := state.Get("driver").(Driver) ui := state.Get("ui").(packer.Ui) // If we started the machine... stop it. if s.vmxPath != "" { // If we started it less than 5 seconds ago... wait. sinceBootTime := time.Since(s.bootTime) waitBootTime := 5 * time.Second if sinceBootTime < waitBootTime { sleepTime := waitBootTime - sinceBootTime ui.Say(fmt.Sprintf("Waiting %s to give VMware time to clean up...", sleepTime.String())) time.Sleep(sleepTime) } // See if it is running running, _ := driver.IsRunning(s.vmxPath) if running { ui.Say("Stopping virtual machine...") if err := driver.Stop(s.vmxPath); err != nil { ui.Error(fmt.Sprintf("Error stopping VM: %s", err)) } } if remoteDriver, ok := driver.(RemoteDriver); ok && s.registered { ui.Say("Unregistering virtual machine...") if err := remoteDriver.Unregister(s.vmxPath); err != nil { ui.Error(fmt.Sprintf("Error unregistering VM: %s", err)) } s.registered = false } } }