2017-09-12 11:30:39 -04:00
|
|
|
package oci
|
2017-02-13 05:35:14 -05:00
|
|
|
|
|
|
|
import (
|
2018-04-11 05:20:40 -04:00
|
|
|
"context"
|
2017-02-13 05:35:14 -05:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2020-08-25 12:13:34 -04:00
|
|
|
"strconv"
|
2018-04-11 05:20:40 -04:00
|
|
|
"time"
|
2017-02-13 05:35:14 -05:00
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
core "github.com/oracle/oci-go-sdk/core"
|
2017-02-13 05:35:14 -05:00
|
|
|
)
|
|
|
|
|
2017-09-12 11:30:39 -04:00
|
|
|
// driverOCI implements the Driver interface and communicates with Oracle
|
|
|
|
// OCI.
|
|
|
|
type driverOCI struct {
|
2018-04-11 05:20:40 -04:00
|
|
|
computeClient core.ComputeClient
|
|
|
|
vcnClient core.VirtualNetworkClient
|
|
|
|
cfg *Config
|
2018-06-26 05:05:56 -04:00
|
|
|
context context.Context
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
// NewDriverOCI Creates a new driverOCI with a connected compute client and a connected vcn client.
|
2017-09-12 11:30:39 -04:00
|
|
|
func NewDriverOCI(cfg *Config) (Driver, error) {
|
2019-10-14 10:08:16 -04:00
|
|
|
coreClient, err := core.NewComputeClientWithConfigurationProvider(cfg.configProvider)
|
2017-02-13 05:35:14 -05:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-04-11 05:20:40 -04:00
|
|
|
|
2019-10-14 10:08:16 -04:00
|
|
|
vcnClient, err := core.NewVirtualNetworkClientWithConfigurationProvider(cfg.configProvider)
|
2018-04-11 05:20:40 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &driverOCI{
|
|
|
|
computeClient: coreClient,
|
|
|
|
vcnClient: vcnClient,
|
|
|
|
cfg: cfg,
|
|
|
|
}, nil
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// CreateInstance creates a new compute instance.
|
2018-06-26 05:05:56 -04:00
|
|
|
func (d *driverOCI) CreateInstance(ctx context.Context, publicKey string) (string, error) {
|
2018-04-11 05:20:40 -04:00
|
|
|
metadata := map[string]string{
|
|
|
|
"ssh_authorized_keys": publicKey,
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
2018-07-17 11:41:19 -04:00
|
|
|
if d.cfg.Metadata != nil {
|
|
|
|
for key, value := range d.cfg.Metadata {
|
2018-07-02 04:48:08 -04:00
|
|
|
metadata[key] = value
|
|
|
|
}
|
|
|
|
}
|
2018-03-28 16:27:41 -04:00
|
|
|
if d.cfg.UserData != "" {
|
2018-04-11 05:20:40 -04:00
|
|
|
metadata["user_data"] = d.cfg.UserData
|
2018-03-28 16:27:41 -04:00
|
|
|
}
|
2018-04-11 05:20:40 -04:00
|
|
|
|
2018-06-22 07:38:25 -04:00
|
|
|
instanceDetails := core.LaunchInstanceDetails{
|
2018-04-11 05:20:40 -04:00
|
|
|
AvailabilityDomain: &d.cfg.AvailabilityDomain,
|
|
|
|
CompartmentId: &d.cfg.CompartmentID,
|
2020-08-31 08:36:09 -04:00
|
|
|
DefinedTags: d.cfg.InstanceDefinedTags,
|
|
|
|
FreeformTags: d.cfg.InstanceTags,
|
2018-04-11 05:20:40 -04:00
|
|
|
ImageId: &d.cfg.BaseImageID,
|
|
|
|
Shape: &d.cfg.Shape,
|
|
|
|
SubnetId: &d.cfg.SubnetID,
|
|
|
|
Metadata: metadata,
|
2018-06-22 07:38:25 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// When empty, the default display name is used.
|
|
|
|
if d.cfg.InstanceName != "" {
|
|
|
|
instanceDetails.DisplayName = &d.cfg.InstanceName
|
|
|
|
}
|
|
|
|
|
2020-08-25 12:13:34 -04:00
|
|
|
// Pass VNIC details, if specified, to the instance
|
|
|
|
if len(d.cfg.CreateVnicDetails) > 0 {
|
|
|
|
CreateVnicDetails, err := mapToCreateVnicDetails(d.cfg.CreateVnicDetails)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
instanceDetails.CreateVnicDetails = &CreateVnicDetails
|
|
|
|
}
|
|
|
|
|
2018-06-22 07:38:25 -04:00
|
|
|
instance, err := d.computeClient.LaunchInstance(context.TODO(), core.LaunchInstanceRequest{LaunchInstanceDetails: instanceDetails})
|
2018-04-11 05:20:40 -04:00
|
|
|
|
2017-02-13 05:35:14 -05:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
return *instance.Id, nil
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// CreateImage creates a new custom image.
|
2018-06-26 05:05:56 -04:00
|
|
|
func (d *driverOCI) CreateImage(ctx context.Context, id string) (core.Image, error) {
|
|
|
|
res, err := d.computeClient.CreateImage(ctx, core.CreateImageRequest{CreateImageDetails: core.CreateImageDetails{
|
2018-04-11 05:20:40 -04:00
|
|
|
CompartmentId: &d.cfg.CompartmentID,
|
|
|
|
InstanceId: &id,
|
|
|
|
DisplayName: &d.cfg.ImageName,
|
2018-06-04 08:14:59 -04:00
|
|
|
FreeformTags: d.cfg.Tags,
|
2019-09-27 08:49:37 -04:00
|
|
|
DefinedTags: d.cfg.DefinedTags,
|
2018-04-11 05:20:40 -04:00
|
|
|
}})
|
|
|
|
|
2017-02-13 05:35:14 -05:00
|
|
|
if err != nil {
|
2018-04-11 05:20:40 -04:00
|
|
|
return core.Image{}, err
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
return res.Image, nil
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteImage deletes a custom image.
|
2018-06-26 05:05:56 -04:00
|
|
|
func (d *driverOCI) DeleteImage(ctx context.Context, id string) error {
|
|
|
|
_, err := d.computeClient.DeleteImage(ctx, core.DeleteImageRequest{ImageId: &id})
|
2018-04-11 05:20:40 -04:00
|
|
|
return err
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
2018-02-13 08:20:26 -05:00
|
|
|
// GetInstanceIP returns the public or private IP corresponding to the given instance id.
|
2018-06-26 05:05:56 -04:00
|
|
|
func (d *driverOCI) GetInstanceIP(ctx context.Context, id string) (string, error) {
|
|
|
|
vnics, err := d.computeClient.ListVnicAttachments(ctx, core.ListVnicAttachmentsRequest{
|
2018-04-11 05:20:40 -04:00
|
|
|
InstanceId: &id,
|
|
|
|
CompartmentId: &d.cfg.CompartmentID,
|
|
|
|
})
|
2017-02-13 05:35:14 -05:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2018-04-11 05:20:40 -04:00
|
|
|
if len(vnics.Items) == 0 {
|
2017-02-13 05:35:14 -05:00
|
|
|
return "", errors.New("instance has zero VNICs")
|
|
|
|
}
|
|
|
|
|
2018-06-26 05:05:56 -04:00
|
|
|
vnic, err := d.vcnClient.GetVnic(ctx, core.GetVnicRequest{VnicId: vnics.Items[0].VnicId})
|
2017-02-13 05:35:14 -05:00
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("Error getting VNIC details: %s", err)
|
|
|
|
}
|
|
|
|
|
2018-02-13 08:20:26 -05:00
|
|
|
if d.cfg.UsePrivateIP {
|
2018-04-11 05:20:40 -04:00
|
|
|
return *vnic.PrivateIp, nil
|
2018-02-13 08:20:26 -05:00
|
|
|
}
|
2018-04-11 05:20:40 -04:00
|
|
|
|
|
|
|
if vnic.PublicIp == nil {
|
|
|
|
return "", fmt.Errorf("Error getting VNIC Public Ip for: %s", id)
|
|
|
|
}
|
|
|
|
|
|
|
|
return *vnic.PublicIp, nil
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
2018-06-26 05:05:56 -04:00
|
|
|
func (d *driverOCI) GetInstanceInitialCredentials(ctx context.Context, id string) (string, string, error) {
|
|
|
|
credentials, err := d.computeClient.GetWindowsInstanceInitialCredentials(ctx, core.GetWindowsInstanceInitialCredentialsRequest{
|
2018-05-24 13:47:37 -04:00
|
|
|
InstanceId: &id,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return *credentials.InstanceCredentials.Username, *credentials.InstanceCredentials.Password, err
|
|
|
|
}
|
|
|
|
|
2017-02-13 05:35:14 -05:00
|
|
|
// TerminateInstance terminates a compute instance.
|
2018-06-26 05:05:56 -04:00
|
|
|
func (d *driverOCI) TerminateInstance(ctx context.Context, id string) error {
|
|
|
|
_, err := d.computeClient.TerminateInstance(ctx, core.TerminateInstanceRequest{
|
2018-04-11 05:20:40 -04:00
|
|
|
InstanceId: &id,
|
|
|
|
})
|
|
|
|
return err
|
2017-02-13 05:35:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// WaitForImageCreation waits for a provisioning custom image to reach the
|
|
|
|
// "AVAILABLE" state.
|
2018-06-26 05:05:56 -04:00
|
|
|
func (d *driverOCI) WaitForImageCreation(ctx context.Context, id string) error {
|
2018-04-11 05:20:40 -04:00
|
|
|
return waitForResourceToReachState(
|
|
|
|
func(string) (string, error) {
|
2018-06-26 05:05:56 -04:00
|
|
|
image, err := d.computeClient.GetImage(ctx, core.GetImageRequest{ImageId: &id})
|
2018-04-11 05:20:40 -04:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return string(image.LifecycleState), nil
|
|
|
|
},
|
2017-02-13 05:35:14 -05:00
|
|
|
id,
|
|
|
|
[]string{"PROVISIONING"},
|
|
|
|
"AVAILABLE",
|
2018-04-11 05:20:40 -04:00
|
|
|
0, //Unlimited Retries
|
|
|
|
5*time.Second, //5 second wait between retries
|
2017-02-13 05:35:14 -05:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// WaitForInstanceState waits for an instance to reach the a given terminal
|
|
|
|
// state.
|
2018-06-26 05:05:56 -04:00
|
|
|
func (d *driverOCI) WaitForInstanceState(ctx context.Context, id string, waitStates []string, terminalState string) error {
|
2018-04-11 05:20:40 -04:00
|
|
|
return waitForResourceToReachState(
|
|
|
|
func(string) (string, error) {
|
2018-06-26 05:05:56 -04:00
|
|
|
instance, err := d.computeClient.GetInstance(ctx, core.GetInstanceRequest{InstanceId: &id})
|
2018-04-11 05:20:40 -04:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return string(instance.LifecycleState), nil
|
|
|
|
},
|
2017-02-13 05:35:14 -05:00
|
|
|
id,
|
|
|
|
waitStates,
|
|
|
|
terminalState,
|
2018-04-11 05:20:40 -04:00
|
|
|
0, //Unlimited Retries
|
|
|
|
5*time.Second, //5 second wait between retries
|
2017-02-13 05:35:14 -05:00
|
|
|
)
|
|
|
|
}
|
2018-04-11 05:20:40 -04:00
|
|
|
|
|
|
|
// WaitForResourceToReachState checks the response of a request through a
|
|
|
|
// polled get and waits until the desired state or until the max retried has
|
|
|
|
// been reached.
|
|
|
|
func waitForResourceToReachState(getResourceState func(string) (string, error), id string, waitStates []string, terminalState string, maxRetries int, waitDuration time.Duration) error {
|
|
|
|
for i := 0; maxRetries == 0 || i < maxRetries; i++ {
|
|
|
|
state, err := getResourceState(id)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if stringSliceContains(waitStates, state) {
|
|
|
|
time.Sleep(waitDuration)
|
|
|
|
continue
|
|
|
|
} else if state == terminalState {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return fmt.Errorf("Unexpected resource state %q, expecting a waiting state %s or terminal state %q ", state, waitStates, terminalState)
|
|
|
|
}
|
|
|
|
return fmt.Errorf("Maximum number of retries (%d) exceeded; resource did not reach state %q", maxRetries, terminalState)
|
|
|
|
}
|
|
|
|
|
|
|
|
// stringSliceContains loops through a slice of strings returning a boolean
|
|
|
|
// based on whether a given value is contained in the slice.
|
|
|
|
func stringSliceContains(slice []string, value string) bool {
|
|
|
|
for _, elem := range slice {
|
|
|
|
if elem == value {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
2020-08-25 12:13:34 -04:00
|
|
|
|
|
|
|
// interfaceToBool converts a variable of type interface to type bool
|
|
|
|
func interfaceToBool(b interface{}) (bool, error) {
|
|
|
|
var boolVal bool
|
|
|
|
var err error
|
|
|
|
|
|
|
|
switch t := b.(type) {
|
|
|
|
case bool:
|
|
|
|
boolVal = t
|
|
|
|
case string:
|
|
|
|
boolVal, err = strconv.ParseBool(t)
|
|
|
|
default:
|
|
|
|
boolVal, err = false, fmt.Errorf("failed to convert %v to boolean type", b)
|
|
|
|
}
|
|
|
|
|
|
|
|
return boolVal, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// mapToCreateVnicDetails creates variable of type core.CreateVnicDetails from map
|
|
|
|
func mapToCreateVnicDetails(m map[string]interface{}) (core.CreateVnicDetails, error) {
|
|
|
|
result := core.CreateVnicDetails{}
|
|
|
|
|
|
|
|
if val, ok := m["assign_public_ip"]; ok {
|
|
|
|
boolVal, err := interfaceToBool(val)
|
|
|
|
if err != nil {
|
|
|
|
return result, fmt.Errorf("assign_public_ip is incorrect type: %v", err)
|
|
|
|
}
|
|
|
|
result.AssignPublicIp = &boolVal
|
|
|
|
}
|
|
|
|
if val, ok := m["display_name"]; ok {
|
|
|
|
tmp := val.(string)
|
|
|
|
result.DisplayName = &tmp
|
|
|
|
}
|
|
|
|
if val, ok := m["hostname_label"]; ok {
|
|
|
|
tmp := val.(string)
|
|
|
|
result.HostnameLabel = &tmp
|
|
|
|
}
|
|
|
|
if val, ok := m["nsg_ids"]; ok {
|
|
|
|
tmp, tmpok := val.([]interface{})
|
|
|
|
if !tmpok {
|
|
|
|
return result, errors.New("nsg_ids not in correct list format")
|
|
|
|
}
|
|
|
|
valStr := make([]string, len(tmp)) //convert []interface{} to []string
|
|
|
|
for i, v := range tmp {
|
|
|
|
valStr[i] = fmt.Sprint(v)
|
|
|
|
}
|
|
|
|
result.NsgIds = valStr
|
|
|
|
}
|
|
|
|
if val, ok := m["private_ip"]; ok {
|
|
|
|
tmp := val.(string)
|
|
|
|
result.PrivateIp = &tmp
|
|
|
|
}
|
|
|
|
if val, ok := m["skip_source_dest_check"]; ok {
|
|
|
|
boolVal, err := interfaceToBool(val)
|
|
|
|
if err != nil {
|
|
|
|
return result, fmt.Errorf("skip_source_dest_check is incorrect type: %v", err)
|
|
|
|
}
|
|
|
|
result.SkipSourceDestCheck = &boolVal
|
|
|
|
}
|
|
|
|
if val, ok := m["subnet_id"]; ok {
|
|
|
|
tmp := val.(string)
|
|
|
|
result.SubnetId = &tmp
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|