Skip to content

Commit

Permalink
Enable linked multiple flexible GPUs to VM
Browse files Browse the repository at this point in the history
  • Loading branch information
outscale-toa committed Dec 15, 2023
1 parent e1448b0 commit 0a7f43c
Show file tree
Hide file tree
Showing 2 changed files with 140 additions and 115 deletions.
239 changes: 130 additions & 109 deletions outscale/resource_outscale_flexible_gpu_link.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,15 +7,16 @@ import (

oscgo "github.com/outscale/osc-sdk-go/v2"

"github.com/hashicorp/terraform-plugin-sdk/helper/resource"
"github.com/hashicorp/terraform-plugin-sdk/helper/schema"
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/resource"
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema"
"github.com/terraform-providers/terraform-provider-outscale/utils"
)

func resourceOutscaleOAPIFlexibleGpuLink() *schema.Resource {
return &schema.Resource{
Create: resourceOutscaleOAPIFlexibleGpuLinkCreate,
Read: resourceOutscaleOAPIFlexibleGpuLinkRead,
Update: resourceFlexibleGpuLinkUpdate,
Delete: resourceOutscaleOAPIFlexibleGpuLinkDelete,
Importer: &schema.ResourceImporter{
State: schema.ImportStatePassthrough,
Expand All @@ -28,10 +29,13 @@ func resourceOutscaleOAPIFlexibleGpuLink() *schema.Resource {
},

Schema: map[string]*schema.Schema{
"flexible_gpu_id": {
Type: schema.TypeString,
"flexible_gpu_ids": {
Type: schema.TypeSet,
Required: true,
ForceNew: true,
MinItems: 1,
Elem: &schema.Schema{
Type: schema.TypeString,
},
},
"vm_id": {
Type: schema.TypeString,
Expand All @@ -48,61 +52,31 @@ func resourceOutscaleOAPIFlexibleGpuLink() *schema.Resource {

func resourceOutscaleOAPIFlexibleGpuLinkCreate(d *schema.ResourceData, meta interface{}) error {
conn := meta.(*OutscaleClient).OSCAPI

flexGpuID := d.Get("flexible_gpu_id").(string)
vmId := d.Get("vm_id").(string)
GpuIdsList := utils.SetToStringSlice(d.Get("flexible_gpu_ids").(*schema.Set))

filter := &oscgo.FiltersFlexibleGpu{
FlexibleGpuIds: &[]string{flexGpuID},
}
reqFlex := &oscgo.ReadFlexibleGpusRequest{
Filters: filter,
}
reqLink := oscgo.LinkFlexibleGpuRequest{
FlexibleGpuId: flexGpuID,
VmId: vmId,
}
var resp oscgo.LinkFlexibleGpuResponse
err := resource.Retry(d.Timeout(schema.TimeoutCreate), func() *resource.RetryError {
var err error
rp, httpResp, err := conn.FlexibleGpuApi.LinkFlexibleGpu(
context.Background()).LinkFlexibleGpuRequest(reqLink).Execute()
if err != nil {
return utils.CheckThrottling(httpResp, err)
for _, flexGpuID := range GpuIdsList {
var resp oscgo.LinkFlexibleGpuResponse
reqLink := oscgo.LinkFlexibleGpuRequest{
FlexibleGpuId: flexGpuID,
VmId: vmId,
}
resp = rp
return nil
})

if err != nil {
return fmt.Errorf("Error Link flexibe gpu: %s", err.Error())
}

if !resp.HasResponseContext() {
return fmt.Errorf("Error there is not Link flexible gpu (%s)", err)
}

var respV oscgo.ReadFlexibleGpusResponse
err = resource.Retry(d.Timeout(schema.TimeoutRead), func() *resource.RetryError {
rp, httpResp, err := conn.FlexibleGpuApi.ReadFlexibleGpus(context.Background()).
ReadFlexibleGpusRequest(*reqFlex).Execute()
err := resource.Retry(d.Timeout(schema.TimeoutCreate), func() *resource.RetryError {
var err error
rp, httpResp, err := conn.FlexibleGpuApi.LinkFlexibleGpu(
context.Background()).LinkFlexibleGpuRequest(reqLink).Execute()
if err != nil {
return utils.CheckThrottling(httpResp, err)
}
resp = rp
return nil
})
if err != nil {
return utils.CheckThrottling(httpResp, err)
return fmt.Errorf("Error Link flexibe gpu: %s", err.Error())
}
if !resp.HasResponseContext() {
return fmt.Errorf("Error there is not Link flexible gpu (%s)", err)
}
respV = rp
return nil
})

if err != nil {
return fmt.Errorf("error reading the FlexibleGpu %s", err)
}

if err := utils.IsResponseEmptyOrMutiple(len(respV.GetFlexibleGpus()), "FlexibleGpu"); err != nil {
return err
}

if (*respV.FlexibleGpus)[0].GetState() != "attaching" {
return fmt.Errorf("Unable to link Flexible GPU")
}

if err := changeShutdownBehavior(conn, vmId, d.Timeout(schema.TimeoutDelete)); err != nil {
Expand All @@ -114,17 +88,12 @@ func resourceOutscaleOAPIFlexibleGpuLinkCreate(d *schema.ResourceData, meta inte

func resourceOutscaleOAPIFlexibleGpuLinkRead(d *schema.ResourceData, meta interface{}) error {
conn := meta.(*OutscaleClient).OSCAPI

flexGpuID := d.Get("flexible_gpu_id").(string)
if flexGpuID == "" {
flexGpuID = d.Id()
}
vmId := d.Get("vm_id").(string)
req := &oscgo.ReadFlexibleGpusRequest{
Filters: &oscgo.FiltersFlexibleGpu{
FlexibleGpuIds: &[]string{flexGpuID},
VmIds: &[]string{vmId},
},
}

var resp oscgo.ReadFlexibleGpusResponse
var err error
err = resource.Retry(d.Timeout(schema.TimeoutRead), func() *resource.RetryError {
Expand All @@ -145,69 +114,69 @@ func resourceOutscaleOAPIFlexibleGpuLinkRead(d *schema.ResourceData, meta interf
d.SetId("")
return nil
}

fg := (*resp.FlexibleGpus)[0]
if err := d.Set("flexible_gpu_id", fg.GetFlexibleGpuId()); err != nil {
flexGpus := resp.GetFlexibleGpus()[:]
readGpuIdsLink := make([]string, len(flexGpus))
for k, flexGpu := range flexGpus {
readGpuIdsLink[k] = flexGpu.GetFlexibleGpuId()
}
if err := d.Set("flexible_gpu_ids", readGpuIdsLink); err != nil {
return err
}
if err := d.Set("vm_id", fg.GetVmId()); err != nil {
if err := d.Set("vm_id", vmId); err != nil {
return err
}
d.SetId(fg.GetFlexibleGpuId())

d.SetId(resource.UniqueId())
return nil
}

func resourceOutscaleOAPIFlexibleGpuLinkDelete(d *schema.ResourceData, meta interface{}) error {
conn := meta.(*OutscaleClient).OSCAPI

flexGpuID := d.Get("flexible_gpu_id").(string)
GpuIdsList := utils.SetToStringSlice(d.Get("flexible_gpu_ids").(*schema.Set))
vmId := d.Get("vm_id").(string)

filter := &oscgo.FiltersFlexibleGpu{
FlexibleGpuIds: &[]string{flexGpuID},
}
reqFlex := &oscgo.ReadFlexibleGpusRequest{
Filters: filter,
}

req := &oscgo.UnlinkFlexibleGpuRequest{
FlexibleGpuId: flexGpuID,
}

var err error
err = resource.Retry(d.Timeout(schema.TimeoutDelete), func() *resource.RetryError {
_, httpResp, err := conn.FlexibleGpuApi.UnlinkFlexibleGpu(
context.Background()).UnlinkFlexibleGpuRequest(*req).Execute()

for _, flexGpuID := range GpuIdsList {
req := &oscgo.UnlinkFlexibleGpuRequest{
FlexibleGpuId: flexGpuID,
}
err = resource.Retry(d.Timeout(schema.TimeoutDelete), func() *resource.RetryError {
_, httpResp, err := conn.FlexibleGpuApi.UnlinkFlexibleGpu(
context.Background()).UnlinkFlexibleGpuRequest(*req).Execute()
if err != nil {
return utils.CheckThrottling(httpResp, err)
}
return nil
})
if err != nil {
return utils.CheckThrottling(httpResp, err)
return err
}
return nil
})

if err != nil {
return err
}

var resp oscgo.ReadFlexibleGpusResponse
err = resource.Retry(d.Timeout(schema.TimeoutDelete), func() *resource.RetryError {
rp, httpResp, err := conn.FlexibleGpuApi.ReadFlexibleGpus(context.Background()).
ReadFlexibleGpusRequest(*reqFlex).Execute()
var resp oscgo.ReadFlexibleGpusResponse
reqFlex := &oscgo.ReadFlexibleGpusRequest{
Filters: &oscgo.FiltersFlexibleGpu{
FlexibleGpuIds: &[]string{flexGpuID},
},
}
err = resource.Retry(d.Timeout(schema.TimeoutDelete), func() *resource.RetryError {
rp, httpResp, err := conn.FlexibleGpuApi.ReadFlexibleGpus(context.Background()).
ReadFlexibleGpusRequest(*reqFlex).Execute()

if err != nil {
return utils.CheckThrottling(httpResp, err)
}
resp = rp
return nil
})
if err != nil {
return utils.CheckThrottling(httpResp, err)
return fmt.Errorf("error reading the FlexibleGpu %s", err)
}
resp = rp
return nil
})
if err != nil {
return fmt.Errorf("error reading the FlexibleGpu %s", err)
}

if len(*resp.FlexibleGpus) != 1 {
return fmt.Errorf("Unable to find Flexible GPU")
}
if (*resp.FlexibleGpus)[0].GetState() != "detaching" {
return fmt.Errorf("Unable to unlink Flexible GPU")
if len(*resp.FlexibleGpus) != 1 {
return fmt.Errorf("Unable to find Flexible GPU")
}
if (*resp.FlexibleGpus)[0].GetState() != "detaching" {
return fmt.Errorf("Unable to unlink Flexible GPU")
}
}

if err := changeShutdownBehavior(conn, vmId, d.Timeout(schema.TimeoutDelete)); err != nil {
Expand All @@ -216,11 +185,63 @@ func resourceOutscaleOAPIFlexibleGpuLinkDelete(d *schema.ResourceData, meta inte

d.SetId("")
return nil
}

func resourceFlexibleGpuLinkUpdate(d *schema.ResourceData, meta interface{}) error {
conn := meta.(*OutscaleClient).OSCAPI
vmId := d.Get("vm_id").(string)
oldIds, newIds := d.GetChange("flexible_gpu_ids")

interIds := oldIds.(*schema.Set).Intersection(newIds.(*schema.Set))
toCreate := newIds.(*schema.Set).Difference(interIds)
toRemove := oldIds.(*schema.Set).Difference(interIds)
var err error

if toRemove.Len() > 0 {
for _, flexGpuID := range utils.SetToStringSlice(toRemove) {
req := &oscgo.UnlinkFlexibleGpuRequest{
FlexibleGpuId: flexGpuID,
}
err = resource.Retry(20*time.Second, func() *resource.RetryError {
_, httpResp, err := conn.FlexibleGpuApi.UnlinkFlexibleGpu(
context.Background()).UnlinkFlexibleGpuRequest(*req).Execute()
if err != nil {
return utils.CheckThrottling(httpResp, err)
}
return nil
})
if err != nil {
return err
}
}
}
if toCreate.Len() > 0 {
for _, flexGpuID := range utils.SetToStringSlice(toCreate) {
req := &oscgo.LinkFlexibleGpuRequest{
FlexibleGpuId: flexGpuID,
VmId: vmId,
}
err = resource.Retry(20*time.Second, func() *resource.RetryError {
_, httpResp, err := conn.FlexibleGpuApi.LinkFlexibleGpu(
context.Background()).LinkFlexibleGpuRequest(*req).Execute()
if err != nil {
return utils.CheckThrottling(httpResp, err)
}
return nil
})
if err != nil {
return err
}
}
}
if err := changeShutdownBehavior(conn, vmId, d.Timeout(schema.TimeoutUpdate)); err != nil {
return fmt.Errorf("Unable to change ShutdownBehavior: %s\n", err)
}

return resourceOutscaleOAPIFlexibleGpuLinkRead(d, meta)
}

func changeShutdownBehavior(conn *oscgo.APIClient, vmId string, timeOut time.Duration) error {

var resp oscgo.ReadVmsResponse
err := resource.Retry(timeOut, func() *resource.RetryError {
rp, httpResp, err := conn.VmApi.ReadVms(context.Background()).ReadVmsRequest(oscgo.ReadVmsRequest{
Expand Down
16 changes: 10 additions & 6 deletions outscale/resource_outscale_flexible_gpu_link_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import (
"os"
"testing"

"github.com/hashicorp/terraform-plugin-sdk/helper/resource"
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/resource"
"github.com/terraform-providers/terraform-provider-outscale/utils"
)

Expand Down Expand Up @@ -35,17 +35,21 @@ func testAccOutscaleOAPIFlexibleGpuLinkConfig(omi, vmType, region string) string
}
resource "outscale_flexible_gpu" "fGPU-1" {
resource "outscale_flexible_gpu" "fGPU-1" {
model_name = "nvidia-p6"
generation = "v5"
subregion_name = "%[3]sa"
subregion_name = "%[3]s"
delete_on_vm_deletion = true
}
resource "outscale_flexible_gpu" "fGPU-2" {
model_name = "nvidia-p6"
generation = "v5"
subregion_name = "%[3]s"
delete_on_vm_deletion = true
}
resource "outscale_flexible_gpu_link" "link_fGPU" {
flexible_gpu_id = outscale_flexible_gpu.fGPU-1.flexible_gpu_id
flexible_gpu_ids = [outscale_flexible_gpu.fGPU-1.flexible_gpu_id,outscale_flexible_gpu.fGPU-2.flexible_gpu_id]
vm_id = outscale_vm.basic.vm_id
}
`, omi, vmType, region)
}

0 comments on commit 0a7f43c

Please sign in to comment.