Commit 4677f388 authored by Mitchell Hashimoto's avatar Mitchell Hashimoto

builder/amazon: parallelize AMI region copies [GH-495]

parent 98985c2d
...@@ -20,6 +20,8 @@ IMPROVEMENTS: ...@@ -20,6 +20,8 @@ IMPROVEMENTS:
* core: Plugins communicate over a single TCP connection per plugin now, * core: Plugins communicate over a single TCP connection per plugin now,
instead of sometimes dozens. Performance around plugin communication instead of sometimes dozens. Performance around plugin communication
dramatically increased. dramatically increased.
* builder/amazon/all: Copying AMIs to multiple regions now happens
in parallel. [GH-495]
* provisioner/puppet-masterless: Can now specify a `manifest_dir` to * provisioner/puppet-masterless: Can now specify a `manifest_dir` to
upload manifests to the remote machine for imports. [GH-655] upload manifests to the remote machine for imports. [GH-655]
......
...@@ -6,6 +6,7 @@ import ( ...@@ -6,6 +6,7 @@ import (
"github.com/mitchellh/goamz/ec2" "github.com/mitchellh/goamz/ec2"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
"sync"
) )
type StepAMIRegionCopy struct { type StepAMIRegionCopy struct {
...@@ -23,41 +24,37 @@ func (s *StepAMIRegionCopy) Run(state multistep.StateBag) multistep.StepAction { ...@@ -23,41 +24,37 @@ func (s *StepAMIRegionCopy) Run(state multistep.StateBag) multistep.StepAction {
} }
ui.Say(fmt.Sprintf("Copying AMI (%s) to other regions...", ami)) ui.Say(fmt.Sprintf("Copying AMI (%s) to other regions...", ami))
var lock sync.Mutex
var wg sync.WaitGroup
errs := new(packer.MultiError)
for _, region := range s.Regions { for _, region := range s.Regions {
wg.Add(1)
ui.Message(fmt.Sprintf("Copying to: %s", region)) ui.Message(fmt.Sprintf("Copying to: %s", region))
// Connect to the region where the AMI will be copied to go func(region string) {
regionconn := ec2.New(ec2conn.Auth, aws.Regions[region]) defer wg.Done()
resp, err := regionconn.CopyImage(&ec2.CopyImage{ id, err := amiRegionCopy(state, ec2conn.Auth, ami,
SourceRegion: ec2conn.Region.Name, aws.Regions[region], ec2conn.Region)
SourceImageId: ami,
}) lock.Lock()
defer lock.Unlock()
if err != nil { amis[region] = id
err := fmt.Errorf("Error Copying AMI (%s) to region (%s): %s", ami, region, err) if err != nil {
state.Put("error", err) errs = packer.MultiErrorAppend(errs, err)
ui.Error(err.Error()) }
return multistep.ActionHalt }(region)
} }
stateChange := StateChangeConf{ // TODO(mitchellh): Wait but also allow for cancels to go through...
Conn: regionconn, ui.Message("Waiting for all copies to complete...")
Pending: []string{"pending"}, wg.Wait()
Target: "available",
Refresh: AMIStateRefreshFunc(regionconn, resp.ImageId), // If there were errors, show them
StepState: state, if len(errs.Errors) > 0 {
} state.Put("error", errs)
ui.Error(errs.Error())
ui.Say(fmt.Sprintf("Waiting for AMI (%s) in region (%s) to become ready...", return multistep.ActionHalt
resp.ImageId, region))
if _, err := WaitForState(&stateChange); err != nil {
err := fmt.Errorf("Error waiting for AMI (%s) in region (%s): %s", resp.ImageId, region, err)
state.Put("error", err)
ui.Error(err.Error())
return multistep.ActionHalt
}
amis[region] = resp.ImageId
} }
state.Put("amis", amis) state.Put("amis", amis)
...@@ -67,3 +64,36 @@ func (s *StepAMIRegionCopy) Run(state multistep.StateBag) multistep.StepAction { ...@@ -67,3 +64,36 @@ func (s *StepAMIRegionCopy) Run(state multistep.StateBag) multistep.StepAction {
func (s *StepAMIRegionCopy) Cleanup(state multistep.StateBag) { func (s *StepAMIRegionCopy) Cleanup(state multistep.StateBag) {
// No cleanup... // No cleanup...
} }
// amiRegionCopy does a copy for the given AMI to the target region and
// returns the resulting ID or error.
func amiRegionCopy(state multistep.StateBag, auth aws.Auth, imageId string,
target aws.Region, source aws.Region) (string, error) {
// Connect to the region where the AMI will be copied to
regionconn := ec2.New(auth, target)
resp, err := regionconn.CopyImage(&ec2.CopyImage{
SourceRegion: source.Name,
SourceImageId: imageId,
})
if err != nil {
return "", fmt.Errorf("Error Copying AMI (%s) to region (%s): %s",
imageId, target, err)
}
stateChange := StateChangeConf{
Conn: regionconn,
Pending: []string{"pending"},
Target: "available",
Refresh: AMIStateRefreshFunc(regionconn, resp.ImageId),
StepState: state,
}
if _, err := WaitForState(&stateChange); err != nil {
return "", fmt.Errorf("Error waiting for AMI (%s) in region (%s): %s",
resp.ImageId, target, err)
}
return resp.ImageId, nil
}
...@@ -7,11 +7,30 @@ ...@@ -7,11 +7,30 @@
load test_helper load test_helper
fixtures amazon-ebs fixtures amazon-ebs
# This counts how many AMIs were copied to another region
aws_ami_region_copy_count() {
aws ec2 describe-images --region $1 --owners self --output text \
--filters 'Name=tag:packer-id,Values=ami_region_copy' \
--query "Images[*].ImageId" \
| wc -l
}
teardown() { teardown() {
aws_ami_cleanup aws_ami_cleanup 'us-east-1'
aws_ami_cleanup 'us-west-1'
aws_ami_cleanup 'us-west-2'
} }
@test "amazon-ebs: build minimal.json" { @test "amazon-ebs: build minimal.json" {
run packer build $FIXTURE_ROOT/minimal.json run packer build $FIXTURE_ROOT/minimal.json
[ "$status" -eq 0 ] [ "$status" -eq 0 ]
} }
# @unit-testable
@test "amazon-ebs: AMI region copy" {
run packer build $FIXTURE_ROOT/ami_region_copy.json
[ "$status" -eq 0 ]
[ "$(aws_ami_region_copy_count 'us-east-1')" -eq "1" ]
[ "$(aws_ami_region_copy_count 'us-west-1')" -eq "1" ]
[ "$(aws_ami_region_copy_count 'us-west-2')" -eq "1" ]
}
{
"builders": [{
"type": "amazon-ebs",
"ami_name": "packer-test {{timestamp}}",
"instance_type": "m1.small",
"region": "us-east-1",
"ssh_username": "ubuntu",
"source_ami": "ami-0568456c",
"tags": {
"packer-test": "true",
"packer-id": "ami_region_copy"
},
"ami_regions": ["us-west-1", "us-west-2"]
}]
}
...@@ -39,7 +39,9 @@ fixtures() { ...@@ -39,7 +39,9 @@ fixtures() {
# This deletes any AMIs with a tag "packer-test" of "true" # This deletes any AMIs with a tag "packer-test" of "true"
aws_ami_cleanup() { aws_ami_cleanup() {
aws ec2 describe-images --owners self --output json --filters 'Name=tag:packer-test,Values=true' \ local region=${1:-us-east-1}
aws ec2 describe-images --region ${region} --owners self --output json \
--filters 'Name=tag:packer-test,Values=true' \
| jq -r -M '.Images[]["ImageId"]' \ | jq -r -M '.Images[]["ImageId"]' \
| xargs -n1 aws ec2 deregister-image --image-id | xargs -n1 aws ec2 deregister-image --region ${region} --image-id
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment