package qemu import ( "errors" "fmt" "github.com/mitchellh/multistep" "github.com/mitchellh/packer/packer" "log" "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 // vmName string // // Produces: // type stepShutdown struct{} func (s *stepShutdown) Run(state multistep.StateBag) multistep.StepAction { comm := state.Get("communicator").(packer.Communicator) config := state.Get("config").(*config) driver := state.Get("driver").(Driver) ui := state.Get("ui").(packer.Ui) vmName := config.VMName 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 := cmd.StartWithUi(comm, ui); err != nil { err := fmt.Errorf("Failed to send shutdown command: %s", err) state.Put("error", err) ui.Error(err.Error()) return multistep.ActionHalt } // 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(vmName) if !running { break } select { case <-shutdownTimer: err := errors.New("Timeout while waiting for machine to shut down.") state.Put("error", err) ui.Error(err.Error()) return multistep.ActionHalt default: time.Sleep(1 * time.Second) } } } else { ui.Say("Halting the virtual machine...") if err := driver.Stop(vmName); err != nil { err := fmt.Errorf("Error stopping VM: %s", err) state.Put("error", err) ui.Error(err.Error()) return multistep.ActionHalt } } log.Println("VM shut down.") return multistep.ActionContinue } func (s *stepShutdown) Cleanup(state multistep.StateBag) {}