package virtualbox import ( "crypto/md5" "encoding/hex" "fmt" "github.com/mitchellh/multistep" "github.com/mitchellh/packer/builder/common" "github.com/mitchellh/packer/packer" "log" "time" ) // This step downloads the ISO specified. // // Uses: // cache packer.Cache // config *config // ui packer.Ui // // Produces: // iso_path string type stepDownloadISO struct{} func (s stepDownloadISO) Run(state map[string]interface{}) multistep.StepAction { cache := state["cache"].(packer.Cache) config := state["config"].(*config) ui := state["ui"].(packer.Ui) checksum, err := hex.DecodeString(config.ISOMD5) if err != nil { ui.Error(fmt.Sprintf("Error parsing checksum: %s", err)) return multistep.ActionHalt } log.Printf("Acquiring lock to download the ISO.") cachePath := cache.Lock(config.ISOUrl) defer cache.Unlock(config.ISOUrl) downloadConfig := &common.DownloadConfig{ Url: config.ISOUrl, TargetPath: cachePath, CopyFile: false, Hash: md5.New(), Checksum: checksum, } download := common.NewDownloadClient(downloadConfig) downloadCompleteCh := make(chan error, 1) go func() { ui.Say("Copying or downloading ISO. Progress will be reported periodically.") cachePath, err = download.Get() downloadCompleteCh <- err }() progressTicker := time.NewTicker(5 * time.Second) defer progressTicker.Stop() DownloadWaitLoop: for { select { case err := <-downloadCompleteCh: if err != nil { ui.Error(fmt.Sprintf("Error downloading ISO: %s", err)) } break DownloadWaitLoop case <-progressTicker.C: ui.Say(fmt.Sprintf("Download progress: %d%%", download.PercentProgress())) case <-time.After(1 * time.Second): if _, ok := state[multistep.StateCancelled]; ok { ui.Say("Interrupt received. Cancelling download...") return multistep.ActionHalt } } } log.Printf("Path to ISO on disk: %s", cachePath) state["iso_path"] = cachePath return multistep.ActionContinue } func (stepDownloadISO) Cleanup(map[string]interface{}) {}