package oci import ( "context" "errors" "fmt" "time" core "github.com/oracle/oci-go-sdk/core" ) // driverOCI implements the Driver interface and communicates with Oracle // OCI. type driverOCI struct { computeClient core.ComputeClient vcnClient core.VirtualNetworkClient cfg *Config context context.Context } // NewDriverOCI Creates a new driverOCI with a connected compute client and a connected vcn client. func NewDriverOCI(cfg *Config) (Driver, error) { coreClient, err := core.NewComputeClientWithConfigurationProvider(cfg.ConfigProvider) if err != nil { return nil, err } vcnClient, err := core.NewVirtualNetworkClientWithConfigurationProvider(cfg.ConfigProvider) if err != nil { return nil, err } return &driverOCI{ computeClient: coreClient, vcnClient: vcnClient, cfg: cfg, }, nil } // CreateInstance creates a new compute instance. func (d *driverOCI) CreateInstance(ctx context.Context, publicKey string) (string, error) { metadata := map[string]string{ "ssh_authorized_keys": publicKey, } if d.cfg.Metadata != nil { for key, value := range d.cfg.Metadata { metadata[key] = value } } if d.cfg.UserData != "" { metadata["user_data"] = d.cfg.UserData } instanceDetails := core.LaunchInstanceDetails{ AvailabilityDomain: &d.cfg.AvailabilityDomain, CompartmentId: &d.cfg.CompartmentID, ImageId: &d.cfg.BaseImageID, Shape: &d.cfg.Shape, SubnetId: &d.cfg.SubnetID, Metadata: metadata, } // When empty, the default display name is used. if d.cfg.InstanceName != "" { instanceDetails.DisplayName = &d.cfg.InstanceName } instance, err := d.computeClient.LaunchInstance(context.TODO(), core.LaunchInstanceRequest{LaunchInstanceDetails: instanceDetails}) if err != nil { return "", err } return *instance.Id, nil } // CreateImage creates a new custom image. func (d *driverOCI) CreateImage(ctx context.Context, id string) (core.Image, error) { res, err := d.computeClient.CreateImage(ctx, core.CreateImageRequest{CreateImageDetails: core.CreateImageDetails{ CompartmentId: &d.cfg.CompartmentID, InstanceId: &id, DisplayName: &d.cfg.ImageName, FreeformTags: d.cfg.Tags, }}) if err != nil { return core.Image{}, err } return res.Image, nil } // DeleteImage deletes a custom image. func (d *driverOCI) DeleteImage(ctx context.Context, id string) error { _, err := d.computeClient.DeleteImage(ctx, core.DeleteImageRequest{ImageId: &id}) return err } // GetInstanceIP returns the public or private IP corresponding to the given instance id. func (d *driverOCI) GetInstanceIP(ctx context.Context, id string) (string, error) { vnics, err := d.computeClient.ListVnicAttachments(ctx, core.ListVnicAttachmentsRequest{ InstanceId: &id, CompartmentId: &d.cfg.CompartmentID, }) if err != nil { return "", err } if len(vnics.Items) == 0 { return "", errors.New("instance has zero VNICs") } vnic, err := d.vcnClient.GetVnic(ctx, core.GetVnicRequest{VnicId: vnics.Items[0].VnicId}) if err != nil { return "", fmt.Errorf("Error getting VNIC details: %s", err) } if d.cfg.UsePrivateIP { return *vnic.PrivateIp, nil } if vnic.PublicIp == nil { return "", fmt.Errorf("Error getting VNIC Public Ip for: %s", id) } return *vnic.PublicIp, nil } func (d *driverOCI) GetInstanceInitialCredentials(ctx context.Context, id string) (string, string, error) { credentials, err := d.computeClient.GetWindowsInstanceInitialCredentials(ctx, core.GetWindowsInstanceInitialCredentialsRequest{ InstanceId: &id, }) if err != nil { return "", "", err } return *credentials.InstanceCredentials.Username, *credentials.InstanceCredentials.Password, err } // TerminateInstance terminates a compute instance. func (d *driverOCI) TerminateInstance(ctx context.Context, id string) error { _, err := d.computeClient.TerminateInstance(ctx, core.TerminateInstanceRequest{ InstanceId: &id, }) return err } // WaitForImageCreation waits for a provisioning custom image to reach the // "AVAILABLE" state. func (d *driverOCI) WaitForImageCreation(ctx context.Context, id string) error { return waitForResourceToReachState( func(string) (string, error) { image, err := d.computeClient.GetImage(ctx, core.GetImageRequest{ImageId: &id}) if err != nil { return "", err } return string(image.LifecycleState), nil }, id, []string{"PROVISIONING"}, "AVAILABLE", 0, //Unlimited Retries 5*time.Second, //5 second wait between retries ) } // WaitForInstanceState waits for an instance to reach the a given terminal // state. func (d *driverOCI) WaitForInstanceState(ctx context.Context, id string, waitStates []string, terminalState string) error { return waitForResourceToReachState( func(string) (string, error) { instance, err := d.computeClient.GetInstance(ctx, core.GetInstanceRequest{InstanceId: &id}) if err != nil { return "", err } return string(instance.LifecycleState), nil }, id, waitStates, terminalState, 0, //Unlimited Retries 5*time.Second, //5 second wait between retries ) } // 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 }