packer-plugin-xenserver/builder/xenserver/common/step_start_on_himn.go

138 lines
3.5 KiB
Go
Raw Normal View History

package common
import (
2014-12-08 09:34:48 -06:00
"fmt"
2020-09-13 03:21:07 -05:00
"log"
"time"
2014-12-08 09:34:48 -06:00
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/packer"
gossh "golang.org/x/crypto/ssh"
)
type StepStartOnHIMN struct{}
/*
* This step starts the installed guest on the Host Internal Management Network
* as there exists an API to obtain the IP allocated to the VM by XAPI.
* This in turn will allow Packer to SSH into the VM, provided NATing has been
* enabled on the host.
*
*/
func (self *StepStartOnHIMN) Run(state multistep.StateBag) multistep.StepAction {
2014-12-08 09:34:48 -06:00
ui := state.Get("ui").(packer.Ui)
2020-09-13 03:21:07 -05:00
c := state.Get("client").(*Connection)
2014-12-08 09:34:48 -06:00
ui.Say("Step: Start VM on the Host Internal Mangement Network")
uuid := state.Get("instance_uuid").(string)
2020-09-13 03:21:07 -05:00
instance, err := c.client.VM.GetByUUID(c.session, uuid)
if err != nil {
ui.Error(fmt.Sprintf("Unable to get VM from UUID '%s': %s", uuid, err.Error()))
return multistep.ActionHalt
}
2014-12-08 09:34:48 -06:00
// Find the HIMN Ref
2020-09-13 03:21:07 -05:00
networks, err := c.client.Network.GetByNameLabel(c.session, "Host internal management network")
2014-12-08 09:34:48 -06:00
if err != nil || len(networks) == 0 {
ui.Error("Unable to find a host internal management network")
ui.Error(err.Error())
2014-12-08 09:34:48 -06:00
return multistep.ActionHalt
}
2014-12-08 09:34:48 -06:00
himn := networks[0]
2014-12-08 09:34:48 -06:00
// Create a VIF for the HIMN
2020-09-13 03:21:07 -05:00
himn_vif, err := ConnectNetwork(c, himn, instance, "0")
2014-12-08 09:34:48 -06:00
if err != nil {
ui.Error("Error creating VIF")
ui.Error(err.Error())
2014-12-08 09:34:48 -06:00
return multistep.ActionHalt
}
2014-12-08 09:34:48 -06:00
// Start the VM
2020-09-13 03:21:07 -05:00
c.client.VM.Start(c.session, instance, false, false)
2014-12-08 09:34:48 -06:00
var himn_iface_ip string = ""
2014-12-08 09:34:48 -06:00
// Obtain the allocated IP
err = InterruptibleWait{
Predicate: func() (found bool, err error) {
2020-09-13 03:21:07 -05:00
ips, err := c.client.Network.GetAssignedIps(c.session, himn)
if err != nil {
return false, fmt.Errorf("Can't get assigned IPs: %s", err.Error())
}
log.Printf("IPs: %s", ips)
2020-09-13 03:21:07 -05:00
log.Printf("Ref: %s", instance)
//Check for instance.Ref in map
2020-09-13 03:21:07 -05:00
if vm_ip, ok := ips[*himn_vif]; ok && vm_ip != "" {
ui.Say("Found the VM's IP: " + vm_ip)
himn_iface_ip = vm_ip
return true, nil
}
ui.Say("Wait for IP address...")
return false, nil
},
PredicateInterval: 10 * time.Second,
Timeout: 100 * time.Second,
}.Wait(state)
if err != nil {
ui.Error(fmt.Sprintf("Unable to find an IP on the Host-internal management interface: %s", err.Error()))
return multistep.ActionHalt
2014-12-08 09:34:48 -06:00
}
state.Put("himn_ssh_address", himn_iface_ip)
ui.Say("Stored VM's IP " + himn_iface_ip)
2014-12-08 09:34:48 -06:00
// Wait for the VM to boot, and check we can ping this interface
2014-12-08 09:34:48 -06:00
ping_cmd := fmt.Sprintf("ping -c 1 %s", himn_iface_ip)
err = InterruptibleWait{
Predicate: func() (success bool, err error) {
ui.Message(fmt.Sprintf("Attempting to ping interface: %s", ping_cmd))
_, err = ExecuteHostSSHCmd(state, ping_cmd)
switch err.(type) {
case nil:
// ping succeeded
return true, nil
case *gossh.ExitError:
// ping failed, try again
return false, nil
default:
// unknown error
return false, err
}
},
PredicateInterval: 10 * time.Second,
Timeout: 300 * time.Second,
}.Wait(state)
2014-12-08 09:34:48 -06:00
if err != nil {
ui.Error(fmt.Sprintf("Unable to ping interface. (Has the VM not booted?): %s", err.Error()))
2014-12-08 09:34:48 -06:00
return multistep.ActionHalt
}
ui.Message("Ping success! Continuing...")
2014-12-08 09:34:48 -06:00
return multistep.ActionContinue
}
func (self *StepStartOnHIMN) Cleanup(state multistep.StateBag) {}
func HimnSSHIP(state multistep.StateBag) (string, error) {
2014-12-08 09:34:48 -06:00
ip := state.Get("himn_ssh_address").(string)
return ip, nil
}
func HimnSSHPort(state multistep.StateBag) (uint, error) {
2015-01-02 12:19:00 -06:00
config := state.Get("commonconfig").(CommonConfig)
return config.SSHPort, nil
}