2013-07-15 02:02:18 -04:00
|
|
|
package ebs
|
2013-05-21 03:55:32 -04:00
|
|
|
|
|
|
|
import (
|
2013-06-04 15:52:52 -04:00
|
|
|
"bytes"
|
2013-05-27 18:15:42 -04:00
|
|
|
"fmt"
|
2013-05-21 03:55:32 -04:00
|
|
|
"github.com/mitchellh/goamz/ec2"
|
2013-06-04 13:00:06 -04:00
|
|
|
"github.com/mitchellh/multistep"
|
2013-07-25 01:56:37 -04:00
|
|
|
awscommon "github.com/mitchellh/packer/builder/amazon/common"
|
2013-05-21 03:55:32 -04:00
|
|
|
"github.com/mitchellh/packer/packer"
|
2013-06-04 16:12:04 -04:00
|
|
|
"strconv"
|
2013-06-04 15:52:52 -04:00
|
|
|
"text/template"
|
|
|
|
"time"
|
2013-05-21 03:55:32 -04:00
|
|
|
)
|
|
|
|
|
2013-08-01 19:31:07 -04:00
|
|
|
type stepCreateAMI struct {
|
|
|
|
Tags []ec2.Tag
|
|
|
|
}
|
2013-05-21 03:55:32 -04:00
|
|
|
|
2013-06-04 15:52:52 -04:00
|
|
|
type amiNameData struct {
|
|
|
|
CreateTime string
|
|
|
|
}
|
|
|
|
|
2013-06-04 13:00:06 -04:00
|
|
|
func (s *stepCreateAMI) Run(state map[string]interface{}) multistep.StepAction {
|
2013-05-21 03:55:32 -04:00
|
|
|
config := state["config"].(config)
|
|
|
|
ec2conn := state["ec2"].(*ec2.EC2)
|
|
|
|
instance := state["instance"].(*ec2.Instance)
|
|
|
|
ui := state["ui"].(packer.Ui)
|
|
|
|
|
2013-06-04 15:52:52 -04:00
|
|
|
// Parse the name of the AMI
|
|
|
|
amiNameBuf := new(bytes.Buffer)
|
|
|
|
tData := amiNameData{
|
2013-06-04 16:12:04 -04:00
|
|
|
strconv.FormatInt(time.Now().UTC().Unix(), 10),
|
2013-06-04 15:52:52 -04:00
|
|
|
}
|
|
|
|
|
2013-06-17 18:24:33 -04:00
|
|
|
t := template.Must(template.New("ami").Parse(config.AMIName))
|
2013-06-04 15:52:52 -04:00
|
|
|
t.Execute(amiNameBuf, tData)
|
|
|
|
amiName := amiNameBuf.String()
|
|
|
|
|
2013-05-21 03:55:32 -04:00
|
|
|
// Create the image
|
2013-06-04 15:52:52 -04:00
|
|
|
ui.Say(fmt.Sprintf("Creating the AMI: %s", amiName))
|
2013-05-21 03:55:32 -04:00
|
|
|
createOpts := &ec2.CreateImage{
|
|
|
|
InstanceId: instance.InstanceId,
|
2013-06-04 15:52:52 -04:00
|
|
|
Name: amiName,
|
2013-05-21 03:55:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
createResp, err := ec2conn.CreateImage(createOpts)
|
|
|
|
if err != nil {
|
2013-06-19 23:54:02 -04:00
|
|
|
err := fmt.Errorf("Error creating AMI: %s", err)
|
|
|
|
state["error"] = err
|
2013-05-21 03:55:32 -04:00
|
|
|
ui.Error(err.Error())
|
2013-06-04 13:00:06 -04:00
|
|
|
return multistep.ActionHalt
|
2013-05-21 03:55:32 -04:00
|
|
|
}
|
|
|
|
|
2013-05-22 01:28:41 -04:00
|
|
|
// Set the AMI ID in the state
|
2013-05-27 18:15:42 -04:00
|
|
|
ui.Say(fmt.Sprintf("AMI: %s", createResp.ImageId))
|
2013-05-22 01:28:41 -04:00
|
|
|
amis := make(map[string]string)
|
2013-07-29 19:42:35 -04:00
|
|
|
amis[ec2conn.Region.Name] = createResp.ImageId
|
2013-05-22 01:28:41 -04:00
|
|
|
state["amis"] = amis
|
2013-05-21 03:55:32 -04:00
|
|
|
|
|
|
|
// Wait for the image to become ready
|
|
|
|
ui.Say("Waiting for AMI to become ready...")
|
2013-07-25 01:56:37 -04:00
|
|
|
if err := awscommon.WaitForAMI(ec2conn, createResp.ImageId); err != nil {
|
|
|
|
err := fmt.Errorf("Error waiting for AMI: %s", err)
|
|
|
|
state["error"] = err
|
|
|
|
ui.Error(err.Error())
|
|
|
|
return multistep.ActionHalt
|
2013-05-21 03:55:32 -04:00
|
|
|
}
|
|
|
|
|
2013-08-01 19:31:07 -04:00
|
|
|
// Add tags to AMI
|
|
|
|
if s.Tags != nil {
|
2013-08-01 19:38:22 -04:00
|
|
|
ui.Say(fmt.Sprintf("Adding tags to AMI (%s)...", createResp.ImageId))
|
2013-08-01 21:35:43 -04:00
|
|
|
_, err := ec2conn.CreateTags([]string{createResp.ImageId}, s.Tags)
|
2013-08-01 19:31:07 -04:00
|
|
|
if err != nil {
|
|
|
|
err := fmt.Errorf("Error adding tags to AMI (%s): %s", createResp.ImageId, err)
|
|
|
|
state["error"] = err
|
|
|
|
ui.Error(err.Error())
|
|
|
|
return multistep.ActionHalt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-04 13:00:06 -04:00
|
|
|
return multistep.ActionContinue
|
2013-05-21 03:55:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stepCreateAMI) Cleanup(map[string]interface{}) {
|
|
|
|
// No cleanup...
|
|
|
|
}
|