packer-cn/builder/openstack/step_allocate_ip.go

146 lines
4.3 KiB
Go
Raw Normal View History

2014-02-27 03:34:24 -05:00
package openstack
import (
"context"
2014-02-27 03:34:24 -05:00
"fmt"
2015-06-12 00:16:43 -04:00
2016-11-27 19:59:26 -05:00
"github.com/gophercloud/gophercloud/openstack/compute/v2/extensions/floatingips"
"github.com/gophercloud/gophercloud/openstack/compute/v2/servers"
"github.com/gophercloud/gophercloud/pagination"
"github.com/hashicorp/packer/helper/multistep"
2017-04-04 16:39:01 -04:00
"github.com/hashicorp/packer/packer"
2014-02-27 03:34:24 -05:00
)
type StepAllocateIp struct {
FloatingIpPool string
FloatingIp string
ReuseIps bool
2014-02-27 03:34:24 -05:00
}
func (s *StepAllocateIp) Run(_ context.Context, state multistep.StateBag) multistep.StepAction {
2014-02-27 03:34:24 -05:00
ui := state.Get("ui").(packer.Ui)
2015-06-12 00:16:43 -04:00
config := state.Get("config").(Config)
server := state.Get("server").(*servers.Server)
// We need the v2 compute client
client, err := config.computeV2Client()
if err != nil {
err = fmt.Errorf("Error initializing compute client: %s", err)
state.Put("error", err)
return multistep.ActionHalt
}
2014-02-27 03:34:24 -05:00
2016-11-27 19:59:26 -05:00
var instanceIp floatingips.FloatingIP
2015-06-12 10:02:04 -04:00
2014-02-27 03:34:24 -05:00
// This is here in case we error out before putting instanceIp into the
// statebag below, because it is requested by Cleanup()
2015-06-12 10:02:04 -04:00
state.Put("access_ip", &instanceIp)
2014-02-27 03:34:24 -05:00
if s.FloatingIp != "" {
2015-06-12 10:02:04 -04:00
instanceIp.IP = s.FloatingIp
2014-02-27 03:34:24 -05:00
} else if s.FloatingIpPool != "" {
// If ReuseIps is set to true and we have a free floating IP in
// the pool, use it first rather than creating one
if s.ReuseIps {
ui.Say(fmt.Sprintf("Searching for unassociated floating IP in pool %s", s.FloatingIpPool))
pager := floatingips.List(client)
err := pager.EachPage(func(page pagination.Page) (bool, error) {
candidates, err := floatingips.ExtractFloatingIPs(page)
if err != nil {
return false, err // stop and throw error out
}
for _, candidate := range candidates {
if candidate.Pool != s.FloatingIpPool || candidate.InstanceID != "" {
continue // move to next in list
}
// In correct pool and able to be allocated
instanceIp.IP = candidate.IP
ui.Message(fmt.Sprintf("Selected floating IP: %s", instanceIp.IP))
state.Put("floatingip_istemp", false)
return false, nil // stop iterating over pages
}
return true, nil // try the next page
})
if err != nil {
err := fmt.Errorf("Error searching for floating ip from pool '%s'", s.FloatingIpPool)
state.Put("error", err)
ui.Error(err.Error())
return multistep.ActionHalt
}
2014-02-27 03:34:24 -05:00
}
2015-06-12 00:16:43 -04:00
if instanceIp.IP == "" {
ui.Say(fmt.Sprintf("Creating floating IP..."))
ui.Message(fmt.Sprintf("Pool: %s", s.FloatingIpPool))
newIp, err := floatingips.Create(client, floatingips.CreateOpts{
Pool: s.FloatingIpPool,
}).Extract()
if err != nil {
err := fmt.Errorf("Error creating floating ip from pool '%s'", s.FloatingIpPool)
state.Put("error", err)
ui.Error(err.Error())
return multistep.ActionHalt
}
instanceIp = *newIp
ui.Message(fmt.Sprintf("Created floating IP: %s", instanceIp.IP))
state.Put("floatingip_istemp", true)
}
2014-02-27 03:34:24 -05:00
}
2015-06-12 10:02:04 -04:00
if instanceIp.IP != "" {
ui.Say(fmt.Sprintf("Associating floating IP with server..."))
ui.Message(fmt.Sprintf("IP: %s", instanceIp.IP))
2016-11-27 19:59:26 -05:00
err := floatingips.AssociateInstance(client, server.ID, floatingips.AssociateOpts{
FloatingIP: instanceIp.IP,
}).ExtractErr()
2015-06-12 00:16:43 -04:00
if err != nil {
err := fmt.Errorf(
"Error associating floating IP %s with instance: %s",
instanceIp.IP, err)
2014-02-27 03:34:24 -05:00
state.Put("error", err)
ui.Error(err.Error())
return multistep.ActionHalt
}
2015-06-12 00:16:43 -04:00
ui.Message(fmt.Sprintf(
"Added floating IP %s to instance!", instanceIp.IP))
2014-02-27 03:34:24 -05:00
}
2015-06-12 10:02:04 -04:00
state.Put("access_ip", &instanceIp)
2014-02-27 03:34:24 -05:00
return multistep.ActionContinue
}
func (s *StepAllocateIp) Cleanup(state multistep.StateBag) {
2015-06-12 00:16:43 -04:00
config := state.Get("config").(Config)
2014-02-27 03:34:24 -05:00
ui := state.Get("ui").(packer.Ui)
2016-11-27 19:59:26 -05:00
instanceIp := state.Get("access_ip").(*floatingips.FloatingIP)
2015-06-12 00:16:43 -04:00
// Don't delete pool addresses we didn't allocate
if state.Get("floatingip_istemp") == false {
return
}
2015-06-12 00:16:43 -04:00
// We need the v2 compute client
client, err := config.computeV2Client()
if err != nil {
ui.Error(fmt.Sprintf(
2015-06-12 10:02:04 -04:00
"Error deleting temporary floating IP %s", instanceIp.IP))
2015-06-12 00:16:43 -04:00
return
}
if s.FloatingIpPool != "" && instanceIp.ID != "" {
2016-11-27 19:59:26 -05:00
if err := floatingips.Delete(client, instanceIp.ID).ExtractErr(); err != nil {
2015-06-12 00:16:43 -04:00
ui.Error(fmt.Sprintf(
2015-06-12 10:02:04 -04:00
"Error deleting temporary floating IP %s", instanceIp.IP))
2014-02-27 03:34:24 -05:00
return
}
2015-06-12 00:16:43 -04:00
2015-06-12 10:02:04 -04:00
ui.Say(fmt.Sprintf("Deleted temporary floating IP %s", instanceIp.IP))
2014-02-27 03:34:24 -05:00
}
}