85 lines
1.9 KiB
Go
85 lines
1.9 KiB
Go
package amazonebs
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"github.com/mitchellh/goamz/ec2"
|
|
"github.com/mitchellh/multistep"
|
|
"github.com/mitchellh/packer/packer"
|
|
"log"
|
|
"strconv"
|
|
"text/template"
|
|
"time"
|
|
)
|
|
|
|
type stepCreateAMI struct{}
|
|
|
|
type amiNameData struct {
|
|
CreateTime string
|
|
}
|
|
|
|
func (s *stepCreateAMI) Run(state map[string]interface{}) multistep.StepAction {
|
|
config := state["config"].(config)
|
|
ec2conn := state["ec2"].(*ec2.EC2)
|
|
instance := state["instance"].(*ec2.Instance)
|
|
ui := state["ui"].(packer.Ui)
|
|
|
|
// Parse the name of the AMI
|
|
amiNameBuf := new(bytes.Buffer)
|
|
tData := amiNameData{
|
|
strconv.FormatInt(time.Now().UTC().Unix(), 10),
|
|
}
|
|
|
|
t := template.Must(template.New("ami").Parse(config.AMIName))
|
|
t.Execute(amiNameBuf, tData)
|
|
amiName := amiNameBuf.String()
|
|
|
|
// Create the image
|
|
ui.Say(fmt.Sprintf("Creating the AMI: %s", amiName))
|
|
createOpts := &ec2.CreateImage{
|
|
InstanceId: instance.InstanceId,
|
|
Name: amiName,
|
|
}
|
|
|
|
createResp, err := ec2conn.CreateImage(createOpts)
|
|
if err != nil {
|
|
err := fmt.Errorf("Error creating AMI: %s", err)
|
|
state["error"] = err
|
|
ui.Error(err.Error())
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
// Set the AMI ID in the state
|
|
ui.Say(fmt.Sprintf("AMI: %s", createResp.ImageId))
|
|
amis := make(map[string]string)
|
|
amis[config.Region] = createResp.ImageId
|
|
state["amis"] = amis
|
|
|
|
// Wait for the image to become ready
|
|
ui.Say("Waiting for AMI to become ready...")
|
|
for {
|
|
imageResp, err := ec2conn.Images([]string{createResp.ImageId}, ec2.NewFilter())
|
|
if err != nil {
|
|
err := fmt.Errorf("Error querying images: %s", err)
|
|
state["error"] = err
|
|
ui.Error(err.Error())
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
if imageResp.Images[0].State == "available" {
|
|
break
|
|
}
|
|
|
|
log.Printf("Image in state %s, sleeping 2s before checking again",
|
|
imageResp.Images[0].State)
|
|
|
|
time.Sleep(2 * time.Second)
|
|
}
|
|
|
|
return multistep.ActionContinue
|
|
}
|
|
|
|
func (s *stepCreateAMI) Cleanup(map[string]interface{}) {
|
|
// No cleanup...
|
|
}
|