83 lines
2.0 KiB
Go
83 lines
2.0 KiB
Go
|
package common
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"time"
|
||
|
|
||
|
"github.com/mitchellh/multistep"
|
||
|
"github.com/mitchellh/packer/packer"
|
||
|
"github.com/nilshell/xmlrpc"
|
||
|
)
|
||
|
|
||
|
type StepWaitForIP struct {
|
||
|
Chan <-chan string
|
||
|
Timeout time.Duration
|
||
|
}
|
||
|
|
||
|
func (self *StepWaitForIP) Run(state multistep.StateBag) multistep.StepAction {
|
||
|
ui := state.Get("ui").(packer.Ui)
|
||
|
client := state.Get("client").(XenAPIClient)
|
||
|
|
||
|
ui.Say("Step: Wait for VM's IP to become known to us.")
|
||
|
|
||
|
uuid := state.Get("instance_uuid").(string)
|
||
|
instance, err := client.GetVMByUuid(uuid)
|
||
|
if err != nil {
|
||
|
ui.Error(fmt.Sprintf("Unable to get VM from UUID '%s': %s", uuid, err.Error()))
|
||
|
return multistep.ActionHalt
|
||
|
}
|
||
|
|
||
|
var ip string
|
||
|
err = InterruptibleWait{
|
||
|
Timeout: self.Timeout,
|
||
|
PredicateInterval: 5 * time.Second,
|
||
|
Predicate: func() (result bool, err error) {
|
||
|
// first check if we got any HTTP requests
|
||
|
select {
|
||
|
case ip = <-self.Chan:
|
||
|
ui.Message(fmt.Sprintf("Got IP '%s' from HTTP request", ip))
|
||
|
return true, nil
|
||
|
default:
|
||
|
}
|
||
|
|
||
|
// failing that, look for PV IP
|
||
|
metrics, err := instance.GetGuestMetrics()
|
||
|
if err != nil {
|
||
|
return false, err
|
||
|
}
|
||
|
if metrics != nil {
|
||
|
networks := metrics["networks"].(xmlrpc.Struct)
|
||
|
if ipRaw, ok := networks["0/ip"]; ok {
|
||
|
if ip = ipRaw.(string); ip != "" {
|
||
|
ui.Message(fmt.Sprintf("Got IP '%s' from XenServer tools", ip))
|
||
|
return true, nil
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false, nil
|
||
|
},
|
||
|
}.Wait(state)
|
||
|
if err != nil {
|
||
|
ui.Error(fmt.Sprintf("Could not get IP address of VM: %s", err.Error()))
|
||
|
// @todo: give advice on what went wrong (no HTTP server? no PV drivers?)
|
||
|
return multistep.ActionHalt
|
||
|
}
|
||
|
|
||
|
ui.Say(fmt.Sprintf("Got IP address '%s'", ip))
|
||
|
state.Put("instance_ssh_address", ip)
|
||
|
|
||
|
return multistep.ActionContinue
|
||
|
}
|
||
|
|
||
|
func (self *StepWaitForIP) Cleanup(state multistep.StateBag) {}
|
||
|
|
||
|
func InstanceSSHIP(state multistep.StateBag) (string, error) {
|
||
|
ip := state.Get("instance_ssh_address").(string)
|
||
|
return ip, nil
|
||
|
}
|
||
|
|
||
|
func InstanceSSHPort(state multistep.StateBag) (uint, error) {
|
||
|
return 22, nil
|
||
|
}
|