2014-12-10 15:20:52 -06:00
|
|
|
package cloudstack
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"regexp"
|
2015-07-16 10:40:04 -05:00
|
|
|
"time"
|
2014-12-10 15:20:52 -06:00
|
|
|
|
2015-04-29 04:21:37 -05:00
|
|
|
"github.com/hashicorp/terraform/helper/schema"
|
2014-12-10 15:20:52 -06:00
|
|
|
"github.com/xanzy/go-cloudstack/cloudstack"
|
|
|
|
)
|
|
|
|
|
2015-09-24 15:49:21 -05:00
|
|
|
// CloudStack uses a "special" ID of -1 to define an unlimited resource
|
|
|
|
const UnlimitedResourceID = "-1"
|
2015-09-24 15:16:12 -05:00
|
|
|
|
2014-12-10 15:20:52 -06:00
|
|
|
type retrieveError struct {
|
|
|
|
name string
|
|
|
|
value string
|
|
|
|
err error
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *retrieveError) Error() error {
|
|
|
|
return fmt.Errorf("Error retrieving UUID of %s %s: %s", e.name, e.value, e.err)
|
|
|
|
}
|
|
|
|
|
2015-04-29 04:21:37 -05:00
|
|
|
func setValueOrUUID(d *schema.ResourceData, key string, value string, uuid string) {
|
|
|
|
if isUUID(d.Get(key).(string)) {
|
|
|
|
d.Set(key, uuid)
|
|
|
|
} else {
|
|
|
|
d.Set(key, value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-10 15:20:52 -06:00
|
|
|
func retrieveUUID(cs *cloudstack.CloudStackClient, name, value string) (uuid string, e *retrieveError) {
|
|
|
|
// If the supplied value isn't a UUID, try to retrieve the UUID ourselves
|
|
|
|
if isUUID(value) {
|
|
|
|
return value, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("[DEBUG] Retrieving UUID of %s: %s", name, value)
|
|
|
|
|
|
|
|
var err error
|
|
|
|
switch name {
|
|
|
|
case "disk_offering":
|
|
|
|
uuid, err = cs.DiskOffering.GetDiskOfferingID(value)
|
|
|
|
case "virtual_machine":
|
|
|
|
uuid, err = cs.VirtualMachine.GetVirtualMachineID(value)
|
|
|
|
case "service_offering":
|
|
|
|
uuid, err = cs.ServiceOffering.GetServiceOfferingID(value)
|
|
|
|
case "network_offering":
|
|
|
|
uuid, err = cs.NetworkOffering.GetNetworkOfferingID(value)
|
|
|
|
case "vpc_offering":
|
|
|
|
uuid, err = cs.VPC.GetVPCOfferingID(value)
|
|
|
|
case "vpc":
|
|
|
|
uuid, err = cs.VPC.GetVPCID(value)
|
|
|
|
case "network":
|
|
|
|
uuid, err = cs.Network.GetNetworkID(value)
|
|
|
|
case "zone":
|
2015-09-24 15:49:21 -05:00
|
|
|
if value == UnlimitedResourceID {
|
2015-09-24 14:57:25 -05:00
|
|
|
return value, nil
|
|
|
|
}
|
2014-12-10 15:20:52 -06:00
|
|
|
uuid, err = cs.Zone.GetZoneID(value)
|
|
|
|
case "ipaddress":
|
|
|
|
p := cs.Address.NewListPublicIpAddressesParams()
|
|
|
|
p.SetIpaddress(value)
|
|
|
|
l, e := cs.Address.ListPublicIpAddresses(p)
|
|
|
|
if e != nil {
|
|
|
|
err = e
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if l.Count == 1 {
|
|
|
|
uuid = l.PublicIpAddresses[0].Id
|
|
|
|
break
|
|
|
|
}
|
|
|
|
err = fmt.Errorf("Could not find UUID of IP address: %s", value)
|
2015-04-11 10:50:06 -05:00
|
|
|
case "os_type":
|
|
|
|
p := cs.GuestOS.NewListOsTypesParams()
|
|
|
|
p.SetDescription(value)
|
|
|
|
l, e := cs.GuestOS.ListOsTypes(p)
|
|
|
|
if e != nil {
|
|
|
|
err = e
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if l.Count == 1 {
|
|
|
|
uuid = l.OsTypes[0].Id
|
|
|
|
break
|
|
|
|
}
|
|
|
|
err = fmt.Errorf("Could not find UUID of OS Type: %s", value)
|
2015-05-28 20:41:58 -05:00
|
|
|
case "project":
|
|
|
|
uuid, err = cs.Project.GetProjectID(value)
|
2014-12-10 15:20:52 -06:00
|
|
|
default:
|
|
|
|
return uuid, &retrieveError{name: name, value: value,
|
|
|
|
err: fmt.Errorf("Unknown request: %s", name)}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return uuid, &retrieveError{name: name, value: value, err: err}
|
|
|
|
}
|
|
|
|
|
|
|
|
return uuid, nil
|
|
|
|
}
|
|
|
|
|
2015-03-09 08:02:18 -05:00
|
|
|
func retrieveTemplateUUID(cs *cloudstack.CloudStackClient, zoneid, value string) (uuid string, e *retrieveError) {
|
|
|
|
// If the supplied value isn't a UUID, try to retrieve the UUID ourselves
|
|
|
|
if isUUID(value) {
|
|
|
|
return value, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("[DEBUG] Retrieving UUID of template: %s", value)
|
|
|
|
|
|
|
|
uuid, err := cs.Template.GetTemplateID(value, "executable", zoneid)
|
|
|
|
if err != nil {
|
|
|
|
return uuid, &retrieveError{name: "template", value: value, err: err}
|
|
|
|
}
|
|
|
|
|
|
|
|
return uuid, nil
|
|
|
|
}
|
|
|
|
|
2014-12-10 15:20:52 -06:00
|
|
|
func isUUID(s string) bool {
|
|
|
|
re := regexp.MustCompile(`^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$`)
|
|
|
|
return re.MatchString(s)
|
|
|
|
}
|
2015-07-16 10:40:04 -05:00
|
|
|
|
|
|
|
// RetryFunc is the function retried n times
|
|
|
|
type RetryFunc func() (interface{}, error)
|
|
|
|
|
|
|
|
// Retry is a wrapper around a RetryFunc that will retry a function
|
|
|
|
// n times or until it succeeds.
|
|
|
|
func Retry(n int, f RetryFunc) (interface{}, error) {
|
|
|
|
var lastErr error
|
|
|
|
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
r, err := f()
|
|
|
|
if err == nil {
|
|
|
|
return r, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
lastErr = err
|
|
|
|
time.Sleep(30 * time.Second)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, lastErr
|
|
|
|
}
|