Commit c7b88d65 authored by Mitchell Hashimoto's avatar Mitchell Hashimoto

builder/amazon/common: generic wait for state to wait for any state

parent 4773b487
...@@ -11,17 +11,38 @@ import ( ...@@ -11,17 +11,38 @@ import (
type StateChangeConf struct { type StateChangeConf struct {
Conn *ec2.EC2 Conn *ec2.EC2
Instance *ec2.Instance
Pending []string Pending []string
Refresh func() (interface{}, string, error)
StepState map[string]interface{} StepState map[string]interface{}
Target string Target string
} }
func WaitForState(conf *StateChangeConf) (i *ec2.Instance, err error) { func InstanceStateRefreshFunc(conn *ec2.EC2, i *ec2.Instance) func() (interface{}, string, error) {
return func() (interface{}, string, error) {
resp, err := conn.Instances([]string{i.InstanceId}, ec2.NewFilter())
if err != nil {
return nil, "", err
}
i = &resp.Reservations[0].Instances[0]
return i, i.State.Name, nil
}
}
func WaitForState(conf *StateChangeConf) (i interface{}, err error) {
log.Printf("Waiting for instance state to become: %s", conf.Target) log.Printf("Waiting for instance state to become: %s", conf.Target)
i = conf.Instance for {
for i.State.Name != conf.Target { var currentState string
i, currentState, err = conf.Refresh()
if err != nil {
return
}
if currentState == conf.Target {
return
}
if conf.StepState != nil { if conf.StepState != nil {
if _, ok := conf.StepState[multistep.StateCancelled]; ok { if _, ok := conf.StepState[multistep.StateCancelled]; ok {
return nil, errors.New("interrupted") return nil, errors.New("interrupted")
...@@ -30,24 +51,17 @@ func WaitForState(conf *StateChangeConf) (i *ec2.Instance, err error) { ...@@ -30,24 +51,17 @@ func WaitForState(conf *StateChangeConf) (i *ec2.Instance, err error) {
found := false found := false
for _, allowed := range conf.Pending { for _, allowed := range conf.Pending {
if i.State.Name == allowed { if currentState == allowed {
found = true found = true
break break
} }
} }
if !found { if !found {
fmt.Errorf("unexpected state '%s', wanted target '%s'", i.State.Name, conf.Target) fmt.Errorf("unexpected state '%s', wanted target '%s'", currentState, conf.Target)
return return
} }
var resp *ec2.InstancesResp
resp, err = conf.Conn.Instances([]string{i.InstanceId}, ec2.NewFilter())
if err != nil {
return
}
i = &resp.Reservations[0].Instances[0]
time.Sleep(2 * time.Second) time.Sleep(2 * time.Second)
} }
......
...@@ -62,12 +62,13 @@ func (s *StepRunSourceInstance) Run(state map[string]interface{}) multistep.Step ...@@ -62,12 +62,13 @@ func (s *StepRunSourceInstance) Run(state map[string]interface{}) multistep.Step
ui.Say(fmt.Sprintf("Waiting for instance (%s) to become ready...", s.instance.InstanceId)) ui.Say(fmt.Sprintf("Waiting for instance (%s) to become ready...", s.instance.InstanceId))
stateChange := StateChangeConf{ stateChange := StateChangeConf{
Conn: ec2conn, Conn: ec2conn,
Instance: s.instance,
Pending: []string{"pending"}, Pending: []string{"pending"},
Target: "running", Target: "running",
Refresh: InstanceStateRefreshFunc(ec2conn, s.instance),
StepState: state, StepState: state,
} }
s.instance, err = WaitForState(&stateChange) latestInstance, err := WaitForState(&stateChange)
s.instance = latestInstance.(*ec2.Instance)
if err != nil { if err != nil {
err := fmt.Errorf("Error waiting for instance (%s) to become ready: %s", s.instance.InstanceId, err) err := fmt.Errorf("Error waiting for instance (%s) to become ready: %s", s.instance.InstanceId, err)
state["error"] = err state["error"] = err
...@@ -96,8 +97,8 @@ func (s *StepRunSourceInstance) Cleanup(state map[string]interface{}) { ...@@ -96,8 +97,8 @@ func (s *StepRunSourceInstance) Cleanup(state map[string]interface{}) {
stateChange := StateChangeConf{ stateChange := StateChangeConf{
Conn: ec2conn, Conn: ec2conn,
Instance: s.instance,
Pending: []string{"pending", "running", "shutting-down", "stopped", "stopping"}, Pending: []string{"pending", "running", "shutting-down", "stopped", "stopping"},
Refresh: InstanceStateRefreshFunc(ec2conn, s.instance),
Target: "running", Target: "running",
} }
......
...@@ -29,12 +29,13 @@ func (s *stepStopInstance) Run(state map[string]interface{}) multistep.StepActio ...@@ -29,12 +29,13 @@ func (s *stepStopInstance) Run(state map[string]interface{}) multistep.StepActio
ui.Say("Waiting for the instance to stop...") ui.Say("Waiting for the instance to stop...")
stateChange := awscommon.StateChangeConf{ stateChange := awscommon.StateChangeConf{
Conn: ec2conn, Conn: ec2conn,
Instance: instance,
Pending: []string{"running", "stopping"}, Pending: []string{"running", "stopping"},
Target: "stopped", Target: "stopped",
Refresh: awscommon.InstanceStateRefreshFunc(ec2conn, instance),
StepState: state, StepState: state,
} }
instance, err = awscommon.WaitForState(&stateChange) instanceRaw, err := awscommon.WaitForState(&stateChange)
instance = instanceRaw.(*ec2.Instance)
if err != nil { if err != nil {
err := fmt.Errorf("Error waiting for instance to stop: %s", err) err := fmt.Errorf("Error waiting for instance to stop: %s", err)
state["error"] = err state["error"] = err
......
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