package vmware import ( "errors" "fmt" "github.com/mitchellh/multistep" "github.com/mitchellh/packer/packer" "log" "path/filepath" "strings" "time" ) // This step shuts down the machine. It first attempts to do so gracefully, // but ultimately forcefully shuts it down if that fails. // // Uses: // communicator packer.Communicator // config *config // driver Driver // ui packer.Ui // vmx_path string // // Produces: // type stepShutdown struct{} func (s *stepShutdown) Run(state map[string]interface{}) multistep.StepAction { comm := state["communicator"].(packer.Communicator) config := state["config"].(*config) driver := state["driver"].(Driver) ui := state["ui"].(packer.Ui) vmxPath := state["vmx_path"].(string) if config.ShutdownCommand != "" { ui.Say("Gracefully halting virtual machine...") log.Printf("Executing shutdown command: %s", config.ShutdownCommand) cmd := &packer.RemoteCmd{Command: config.ShutdownCommand} if err := comm.Start(cmd); err != nil { err := fmt.Errorf("Failed to send shutdown command: %s", err) state["error"] = err ui.Error(err.Error()) return multistep.ActionHalt } // Wait for the command to run cmd.Wait() // Wait for the machine to actually shut down log.Printf("Waiting max %s for shutdown to complete", config.ShutdownTimeout) shutdownTimer := time.After(config.ShutdownTimeout) for { running, _ := driver.IsRunning(vmxPath) if !running { break } select { case <-shutdownTimer: err := errors.New("Timeout while waiting for machine to shut down.") state["error"] = err ui.Error(err.Error()) return multistep.ActionHalt default: time.Sleep(1 * time.Second) } } } else { if err := driver.Stop(vmxPath); err != nil { err := fmt.Errorf("Error stopping VM: %s", err) state["error"] = err ui.Error(err.Error()) return multistep.ActionHalt } } ui.Message("Waiting for VMware to clean up after itself...") lockPattern := filepath.Join(config.OutputDir, "*.lck") timer := time.After(15 * time.Second) LockWaitLoop: for { locks, err := filepath.Glob(lockPattern) if err == nil { if len(locks) == 0 { log.Println("No more lock files found. VMware is clean.") break } if len(locks) == 1 && strings.HasSuffix(locks[0], ".vmx.lck") { log.Println("Only waiting on VMX lock. VMware is clean.") break } log.Printf("Waiting on lock files: %#v", locks) } select { case <-timer: log.Println("Reached timeout on waiting for clean VMware. Assuming clean.") break LockWaitLoop case <-time.After(1 * time.Second): } } log.Println("VM shut down.") return multistep.ActionContinue } func (s *stepShutdown) Cleanup(state map[string]interface{}) {}