167 lines
5.1 KiB
Go
167 lines
5.1 KiB
Go
package common
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log"
|
|
"math/rand"
|
|
"sort"
|
|
|
|
"github.com/antihax/optional"
|
|
"github.com/hashicorp/packer/packer-plugin-sdk/multistep"
|
|
packersdk "github.com/hashicorp/packer/packer-plugin-sdk/packer"
|
|
"github.com/outscale/osc-sdk-go/osc"
|
|
)
|
|
|
|
// StepNetworkInfo queries OUTSCALE for information about
|
|
// NET's and Subnets that is used throughout the OMI creation process.
|
|
//
|
|
// Produces (adding them to the state bag):
|
|
// vpc_id string - the NET ID
|
|
// subnet_id string - the Subnet ID
|
|
// availability_zone string - the Subregion name
|
|
type StepNetworkInfo struct {
|
|
NetId string
|
|
NetFilter NetFilterOptions
|
|
SubnetId string
|
|
SubnetFilter SubnetFilterOptions
|
|
SubregionName string
|
|
SecurityGroupIds []string
|
|
SecurityGroupFilter SecurityGroupFilterOptions
|
|
}
|
|
|
|
type subnetsOscSort []osc.Subnet
|
|
|
|
func (a subnetsOscSort) Len() int { return len(a) }
|
|
func (a subnetsOscSort) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|
func (a subnetsOscSort) Less(i, j int) bool {
|
|
return a[i].AvailableIpsCount < a[j].AvailableIpsCount
|
|
}
|
|
|
|
// Returns the most recent OMI out of a slice of images.
|
|
func mostFreeOscSubnet(subnets []osc.Subnet) osc.Subnet {
|
|
sortedSubnets := subnets
|
|
sort.Sort(subnetsOscSort(sortedSubnets))
|
|
return sortedSubnets[len(sortedSubnets)-1]
|
|
}
|
|
|
|
//Run ...
|
|
func (s *StepNetworkInfo) Run(_ context.Context, state multistep.StateBag) multistep.StepAction {
|
|
oscconn := state.Get("osc").(*osc.APIClient)
|
|
ui := state.Get("ui").(packersdk.Ui)
|
|
|
|
// NET
|
|
if s.NetId == "" && !s.NetFilter.Empty() {
|
|
params := osc.ReadNetsRequest{}
|
|
params.Filters = buildOscNetFilters(s.NetFilter.Filters)
|
|
s.NetFilter.Filters["state"] = "available"
|
|
|
|
log.Printf("Using NET Filters %v", params)
|
|
|
|
vpcResp, _, err := oscconn.NetApi.ReadNets(context.Background(), &osc.ReadNetsOpts{
|
|
ReadNetsRequest: optional.NewInterface(params),
|
|
})
|
|
if err != nil {
|
|
err := fmt.Errorf("Error querying NETs: %s", err)
|
|
state.Put("error", err)
|
|
ui.Error(err.Error())
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
if len(vpcResp.Nets) != 1 {
|
|
err := fmt.Errorf("Exactly one NET should match the filter, but %d NET's was found matching filters: %v", len(vpcResp.Nets), params)
|
|
state.Put("error", err)
|
|
ui.Error(err.Error())
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
s.NetId = vpcResp.Nets[0].NetId
|
|
ui.Message(fmt.Sprintf("Found NET ID: %s", s.NetId))
|
|
}
|
|
|
|
// Subnet
|
|
if s.SubnetId == "" && !s.SubnetFilter.Empty() {
|
|
params := osc.ReadSubnetsRequest{}
|
|
s.SubnetFilter.Filters["state"] = "available"
|
|
|
|
if s.NetId != "" {
|
|
s.SubnetFilter.Filters["vpc-id"] = s.NetId
|
|
}
|
|
if s.SubregionName != "" {
|
|
s.SubnetFilter.Filters["availability-zone"] = s.SubregionName
|
|
}
|
|
params.Filters = buildOscSubnetFilters(s.SubnetFilter.Filters)
|
|
log.Printf("Using Subnet Filters %v", params)
|
|
|
|
subnetsResp, _, err := oscconn.SubnetApi.ReadSubnets(context.Background(), &osc.ReadSubnetsOpts{
|
|
ReadSubnetsRequest: optional.NewInterface(params),
|
|
})
|
|
if err != nil {
|
|
err := fmt.Errorf("error querying Subnets: %s", err)
|
|
state.Put("error", err)
|
|
ui.Error(err.Error())
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
if len(subnetsResp.Subnets) == 0 {
|
|
err := fmt.Errorf("No Subnets was found matching filters: %v", params)
|
|
state.Put("error", err)
|
|
ui.Error(err.Error())
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
if len(subnetsResp.Subnets) > 1 && !s.SubnetFilter.Random && !s.SubnetFilter.MostFree {
|
|
err := fmt.Errorf("your filter matched %d Subnets. Please try a more specific search, or set random or most_free to true", len(subnetsResp.Subnets))
|
|
state.Put("error", err)
|
|
ui.Error(err.Error())
|
|
return multistep.ActionHalt
|
|
}
|
|
|
|
var subnet osc.Subnet
|
|
switch {
|
|
case s.SubnetFilter.MostFree:
|
|
subnet = mostFreeOscSubnet(subnetsResp.Subnets)
|
|
case s.SubnetFilter.Random:
|
|
subnet = subnetsResp.Subnets[rand.Intn(len(subnetsResp.Subnets))]
|
|
default:
|
|
subnet = subnetsResp.Subnets[0]
|
|
}
|
|
s.SubnetId = subnet.SubnetId
|
|
ui.Message(fmt.Sprintf("Found Subnet ID: %s", s.SubnetId))
|
|
}
|
|
|
|
// Try to find Subregion and NET Id from Subnet if they are not yet found/given
|
|
if s.SubnetId != "" && (s.SubregionName == "" || s.NetId == "") {
|
|
log.Printf("[INFO] Finding Subregion and NetId for the given subnet '%s'", s.SubnetId)
|
|
resp, _, err := oscconn.SubnetApi.ReadSubnets(context.Background(), &osc.ReadSubnetsOpts{
|
|
ReadSubnetsRequest: optional.NewInterface(osc.ReadSubnetsRequest{
|
|
Filters: osc.FiltersSubnet{
|
|
SubnetIds: []string{s.SubnetId},
|
|
},
|
|
}),
|
|
})
|
|
if err != nil {
|
|
err := fmt.Errorf("describing the subnet: %s returned error: %s", s.SubnetId, err)
|
|
state.Put("error", err)
|
|
ui.Error(err.Error())
|
|
return multistep.ActionHalt
|
|
}
|
|
if s.SubregionName == "" {
|
|
s.SubregionName = resp.Subnets[0].SubregionName
|
|
log.Printf("[INFO] SubregionName found: '%s'", s.SubregionName)
|
|
}
|
|
if s.NetId == "" {
|
|
s.NetId = resp.Subnets[0].NetId
|
|
log.Printf("[INFO] NetId found: '%s'", s.NetId)
|
|
}
|
|
}
|
|
|
|
state.Put("net_id", s.NetId)
|
|
state.Put("subregion_name", s.SubregionName)
|
|
state.Put("subnet_id", s.SubnetId)
|
|
return multistep.ActionContinue
|
|
}
|
|
|
|
//Cleanup ...
|
|
func (s *StepNetworkInfo) Cleanup(multistep.StateBag) {}
|