package chroot import ( "errors" "fmt" "github.com/mitchellh/goamz/ec2" "github.com/mitchellh/multistep" awscommon "github.com/mitchellh/packer/builder/amazon/common" "github.com/mitchellh/packer/packer" "strings" ) // StepAttachVolume attaches the previously created volume to an // available device location. // // Produces: // device string - The location where the volume was attached. // attach_cleanup CleanupFunc type StepAttachVolume struct { attached bool volumeId string } func (s *StepAttachVolume) Run(state map[string]interface{}) multistep.StepAction { ec2conn := state["ec2"].(*ec2.EC2) device := state["device"].(string) instance := state["instance"].(*ec2.Instance) ui := state["ui"].(packer.Ui) volumeId := state["volume_id"].(string) // For the API call, it expects "sd" prefixed devices. attachVolume := strings.Replace(device, "/xvd", "/sd", 1) ui.Say(fmt.Sprintf("Attaching the root volume to %s", attachVolume)) _, err := ec2conn.AttachVolume(volumeId, instance.InstanceId, attachVolume) if err != nil { err := fmt.Errorf("Error attaching volume: %s", err) state["error"] = err ui.Error(err.Error()) return multistep.ActionHalt } // Mark that we attached it so we can detach it later s.attached = true s.volumeId = volumeId // Wait for the volume to become attached stateChange := awscommon.StateChangeConf{ Conn: ec2conn, Pending: []string{"attaching"}, StepState: state, Target: "attached", Refresh: func() (interface{}, string, error) { resp, err := ec2conn.Volumes([]string{volumeId}, ec2.NewFilter()) if err != nil { return nil, "", err } if len(resp.Volumes[0].Attachments) == 0 { return nil, "", errors.New("No attachments on volume.") } return nil, resp.Volumes[0].Attachments[0].Status, nil }, } _, err = awscommon.WaitForState(&stateChange) if err != nil { err := fmt.Errorf("Error waiting for volume: %s", err) state["error"] = err ui.Error(err.Error()) return multistep.ActionHalt } state["attach_cleanup"] = s return multistep.ActionContinue } func (s *StepAttachVolume) Cleanup(state map[string]interface{}) { ui := state["ui"].(packer.Ui) if err := s.CleanupFunc(state); err != nil { ui.Error(err.Error()) } } func (s *StepAttachVolume) CleanupFunc(state map[string]interface{}) error { if !s.attached { return nil } ec2conn := state["ec2"].(*ec2.EC2) ui := state["ui"].(packer.Ui) ui.Say("Detaching EBS volume...") _, err := ec2conn.DetachVolume(s.volumeId) if err != nil { return fmt.Errorf("Error detaching EBS volume: %s", err) } s.attached = false // Wait for the volume to detach stateChange := awscommon.StateChangeConf{ Conn: ec2conn, Pending: []string{"attaching", "attached", "detaching"}, StepState: state, Target: "detached", Refresh: func() (interface{}, string, error) { resp, err := ec2conn.Volumes([]string{s.volumeId}, ec2.NewFilter()) if err != nil { return nil, "", err } state := "detached" if len(resp.Volumes[0].Attachments) > 0 { state = resp.Volumes[0].Attachments[0].Status } return nil, state, nil }, } _, err = awscommon.WaitForState(&stateChange) if err != nil { return fmt.Errorf("Error waiting for volume: %s", err) } return nil }