1.1.0
This commit is contained in:
@@ -0,0 +1,331 @@
|
||||
package utilities
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"strconv"
|
||||
|
||||
"github.com/hashicorp/terraform-plugin-framework/diag"
|
||||
"github.com/hashicorp/terraform-plugin-framework/types"
|
||||
"github.com/hashicorp/terraform-plugin-log/tflog"
|
||||
decort "repository.basistech.ru/BASIS/decort-golang-sdk"
|
||||
"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudbroker/k8ci"
|
||||
"repository.basistech.ru/BASIS/terraform-provider-dynamix/internal/service/cloudbroker/k8ci/models"
|
||||
"repository.basistech.ru/BASIS/terraform-provider-dynamix/internal/status"
|
||||
)
|
||||
|
||||
func CreateRequestResourceK8CI(ctx context.Context, plan *models.ResourceK8CIModel) (k8ci.CreateRequest, diag.Diagnostics) {
|
||||
tflog.Info(ctx, "Start CreateRequestResourceK8CI", map[string]any{
|
||||
"name": plan.Name.ValueString(),
|
||||
})
|
||||
|
||||
// set up required parameters in resource group create request
|
||||
createReq := k8ci.CreateRequest{
|
||||
Name: plan.Name.ValueString(),
|
||||
Version: plan.Version.ValueString(),
|
||||
MasterDriver: plan.MasterDriver.ValueString(),
|
||||
WorkerDriver: plan.WorkerDriver.ValueString(),
|
||||
MaxMasterCount: uint64(plan.MaxMasterCount.ValueInt64()),
|
||||
MaxWorkerCount: uint64(plan.MaxWorkerCount.ValueInt64()),
|
||||
MasterImageID: uint64(plan.MasterImageId.ValueInt64()),
|
||||
WorkerImageID: uint64(plan.WorkerImageId.ValueInt64()),
|
||||
}
|
||||
|
||||
networkPlugins := make([]string, 0, len(plan.NetworkPlugins.Elements()))
|
||||
for _, plugin := range plan.NetworkPlugins.Elements() {
|
||||
networkPlugins = append(networkPlugins, plugin.(types.String).ValueString())
|
||||
}
|
||||
createReq.NetworkPlugins = networkPlugins
|
||||
|
||||
if !plan.Description.IsNull() {
|
||||
createReq.Description = plan.Description.ValueString()
|
||||
}
|
||||
|
||||
if !plan.SharedWith.IsNull() {
|
||||
sharedWith := make([]uint64, 0, len(plan.SharedWith.Elements()))
|
||||
for _, shared := range plan.SharedWith.Elements() {
|
||||
sharedWith = append(sharedWith, uint64(shared.(types.Int64).ValueInt64()))
|
||||
}
|
||||
createReq.SharedWith = sharedWith
|
||||
}
|
||||
|
||||
return createReq, nil
|
||||
}
|
||||
|
||||
func K8CIResourceEnableDisable(ctx context.Context, plan *models.ResourceK8CIModel, c *decort.DecortClient) diag.Diagnostics {
|
||||
tflog.Info(ctx, "EnableDisable k8ci with ID", map[string]any{"k8ci_id": plan.Id.ValueString()})
|
||||
diags := diag.Diagnostics{}
|
||||
k8ciid, err := strconv.ParseUint(plan.Id.ValueString(), 10, 64)
|
||||
if err != nil {
|
||||
diags.AddError("Cannot parsed ID k8ci from state", err.Error())
|
||||
return diags
|
||||
}
|
||||
if plan.Enabled.IsNull() || plan.Enabled.ValueBool() {
|
||||
tflog.Info(ctx, "Enable k8ci with ID", map[string]any{"k8ci_id": plan.Id.ValueString()})
|
||||
_, err := c.CloudBroker().K8CI().Enable(ctx, k8ci.EnableRequest{K8CIID: k8ciid})
|
||||
if err != nil {
|
||||
diags.AddError("K8CIResourceEnableDisable: error to enable k8ci", err.Error())
|
||||
return diags
|
||||
}
|
||||
} else {
|
||||
tflog.Info(ctx, "Disable k8ci with ID", map[string]any{"k8ci_id": plan.Id.ValueString()})
|
||||
_, err := c.CloudBroker().K8CI().Disable(ctx, k8ci.DisableRequest{K8CIID: k8ciid})
|
||||
if err != nil {
|
||||
diags.AddError("K8CIResourceEnableDisable: error to disable k8ci", err.Error())
|
||||
return diags
|
||||
}
|
||||
}
|
||||
return diags
|
||||
}
|
||||
|
||||
func K8CIResourceCheckPresence(ctx context.Context, plan *models.ResourceK8CIModel, c *decort.DecortClient) (*k8ci.RecordK8CI, diag.Diagnostics) {
|
||||
tflog.Info(ctx, fmt.Sprintf("K8CIResourceCheckPresence: Get info about k8ci with ID - %v", plan.Id.ValueString()))
|
||||
|
||||
diags := diag.Diagnostics{}
|
||||
|
||||
k8ciid, err := strconv.ParseUint(plan.Id.ValueString(), 10, 64)
|
||||
if err != nil {
|
||||
diags.AddError("Cannot parsed ID k8ci from state", err.Error())
|
||||
return nil, diags
|
||||
}
|
||||
|
||||
recordK8ci, err := c.CloudBroker().K8CI().Get(ctx, k8ci.GetRequest{K8CIID: k8ciid})
|
||||
if err != nil {
|
||||
diags.AddError(fmt.Sprintf("Cannot get info about k8ci with ID %v", k8ciid), err.Error())
|
||||
return nil, diags
|
||||
}
|
||||
|
||||
tflog.Info(ctx, "K8CIResourceCheckPresence: response from CloudBroker().K8CI().Get", map[string]any{"k8ci_id": k8ciid, "response": recordK8ci})
|
||||
|
||||
return recordK8ci, nil
|
||||
}
|
||||
|
||||
func K8CIReadStatus(ctx context.Context, plan *models.ResourceK8CIModel, c *decort.DecortClient) diag.Diagnostics {
|
||||
tflog.Info(ctx, "Read status k8ci with ID", map[string]any{"k8ci_id": plan.Id.ValueString()})
|
||||
|
||||
diags := diag.Diagnostics{}
|
||||
|
||||
k8ciid, err := strconv.ParseUint(plan.Id.ValueString(), 10, 64)
|
||||
if err != nil {
|
||||
diags.AddError("Cannot parsed ID k8ci from state", err.Error())
|
||||
return diags
|
||||
}
|
||||
|
||||
k8ciItem, err := c.CloudBroker().K8CI().Get(ctx, k8ci.GetRequest{K8CIID: k8ciid})
|
||||
if err != nil {
|
||||
diags.AddError(fmt.Sprintf("Cannot get info about k8ci with ID %v", k8ciItem), err.Error())
|
||||
return diags
|
||||
}
|
||||
|
||||
switch k8ciItem.Status {
|
||||
case status.Modeled:
|
||||
diags.AddError("Error:", fmt.Sprintf("The k8ci is in status: %s, please, contact support for more information", k8ciItem.Status))
|
||||
return diags
|
||||
case status.Deleted:
|
||||
if plan.Restore.ValueBool() || plan.Restore.IsNull() {
|
||||
diags = K8CIRestore(ctx, plan, c)
|
||||
if diags.HasError() {
|
||||
tflog.Error(ctx, "Error restore K8CI", map[string]any{"k8ci_id": plan.Id.ValueString()})
|
||||
return diags
|
||||
}
|
||||
} else {
|
||||
diags.AddError("k8ci in status Deleted:", "please clean state, or restore k8ci")
|
||||
return diags
|
||||
}
|
||||
diags = K8CIResourceEnableDisable(ctx, plan, c)
|
||||
if diags.HasError() {
|
||||
tflog.Error(ctx, "Error enable/disable k8ci", map[string]any{"k8ci_id": plan.Id.ValueString()})
|
||||
return diags
|
||||
}
|
||||
case status.Destroying:
|
||||
diags.AddError("Error:", fmt.Sprintf("The k8ci is in progress with status: %s", k8ciItem.Status))
|
||||
return diags
|
||||
case status.Destroyed:
|
||||
diags.AddError("Error:", "The resource cannot be updated because it has been destroyed")
|
||||
return diags
|
||||
}
|
||||
|
||||
tflog.Info(ctx, "Read status k8ci successfully", map[string]any{"k8ci_id": plan.Id.ValueString()})
|
||||
return diags
|
||||
}
|
||||
|
||||
func K8CIRestore(ctx context.Context, plan *models.ResourceK8CIModel, c *decort.DecortClient) diag.Diagnostics {
|
||||
tflog.Info(ctx, "Restore k8ci with ID", map[string]any{"k8ci_id": plan.Id.ValueString()})
|
||||
|
||||
diags := diag.Diagnostics{}
|
||||
|
||||
k8ciid, err := strconv.ParseUint(plan.Id.ValueString(), 10, 64)
|
||||
if err != nil {
|
||||
diags.AddError("Cannot parsed ID k8ci from state", err.Error())
|
||||
return diags
|
||||
}
|
||||
|
||||
_, err = c.CloudBroker().K8CI().Restore(ctx, k8ci.RestoreRequest{K8CIID: k8ciid})
|
||||
if err != nil {
|
||||
diags.AddError(fmt.Sprintf("Cannot restore k8ci with ID - %s", plan.Id.ValueString()), err.Error())
|
||||
return diags
|
||||
}
|
||||
|
||||
tflog.Info(ctx, "Restore k8ci successfully", map[string]any{"k8ci_id": plan.Id.ValueString()})
|
||||
return diags
|
||||
}
|
||||
|
||||
func K8CIIUpdateVarChecks(plan *models.ResourceK8CIModel, state *models.ResourceK8CIModel) diag.Diagnostics {
|
||||
|
||||
diags := diag.Diagnostics{}
|
||||
|
||||
if !plan.Name.Equal(state.Name) {
|
||||
diags.AddError(
|
||||
"Update resourceK8CI: Invalid input Name",
|
||||
fmt.Sprintf("block Name must not be changed for resource with k8ci_id %s", plan.Id.ValueString()),
|
||||
)
|
||||
return diags
|
||||
}
|
||||
|
||||
if !plan.Version.Equal(state.Version) {
|
||||
diags.AddError(
|
||||
"Update resourceK8CI: Invalid input Version",
|
||||
fmt.Sprintf("block Version must not be changed for resource with k8ci_id %s", plan.Id.ValueString()),
|
||||
)
|
||||
return diags
|
||||
}
|
||||
|
||||
if !plan.MasterDriver.Equal(state.MasterDriver) {
|
||||
diags.AddError(
|
||||
"Update resourceK8CI: Invalid MasterDriver",
|
||||
fmt.Sprintf("block MasterDriver must not be changed for resource with k8ci_id %s", plan.Id.ValueString()),
|
||||
)
|
||||
return diags
|
||||
}
|
||||
|
||||
if !plan.MasterImageId.Equal(state.MasterImageId) {
|
||||
diags.AddError(
|
||||
"Update resourceK8CI: Invalid MasterImageId",
|
||||
fmt.Sprintf("block MasterImageId must not be changed for resource with k8ci_id %s", plan.Id.ValueString()),
|
||||
)
|
||||
return diags
|
||||
}
|
||||
|
||||
if !plan.MaxMasterCount.Equal(state.MaxMasterCount) {
|
||||
diags.AddError(
|
||||
"Update resourceK8CI: Invalid MaxMasterCount",
|
||||
fmt.Sprintf("block MaxMasterCount must not be changed for resource with k8ci_id %s", plan.Id.ValueString()),
|
||||
)
|
||||
return diags
|
||||
}
|
||||
|
||||
if !plan.MaxWorkerCount.Equal(state.MaxWorkerCount) {
|
||||
diags.AddError(
|
||||
"Update resourceK8CI: Invalid MaxWorkerCount",
|
||||
fmt.Sprintf("block MaxWorkerCount must not be changed for resource with k8ci_id %s", plan.Id.ValueString()),
|
||||
)
|
||||
return diags
|
||||
}
|
||||
|
||||
if !plan.NetworkPlugins.Equal(state.NetworkPlugins) {
|
||||
diags.AddError(
|
||||
"Update resourceK8CI: Invalid NetworkPlugins",
|
||||
fmt.Sprintf("block NetworkPlugins must not be changed for resource with k8ci_id %s", plan.Id.ValueString()),
|
||||
)
|
||||
return diags
|
||||
}
|
||||
|
||||
if !plan.WorkerDriver.Equal(state.WorkerDriver) {
|
||||
diags.AddError(
|
||||
"Update resourceK8CI: Invalid WorkerDriver",
|
||||
fmt.Sprintf("block WorkerDriver must not be changed for resource with k8ci_id %s", plan.Id.ValueString()),
|
||||
)
|
||||
return diags
|
||||
}
|
||||
|
||||
if !plan.WorkerImageId.Equal(state.WorkerImageId) {
|
||||
diags.AddError(
|
||||
"Update resourceK8CI: Invalid WorkerImageId",
|
||||
fmt.Sprintf("block WorkerImageId must not be changed for resource with k8ci_id %s", plan.Id.ValueString()),
|
||||
)
|
||||
return diags
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func K8CISharedWithUpdate(ctx context.Context, plan *models.ResourceK8CIModel, state *models.ResourceK8CIModel, c *decort.DecortClient) diag.Diagnostics {
|
||||
diags := diag.Diagnostics{}
|
||||
|
||||
k8ciid, err := strconv.ParseUint(state.Id.ValueString(), 10, 64)
|
||||
if err != nil {
|
||||
diags.AddError("K8CISharedWithUpdate: cannot parsed ID k8ci from state", err.Error())
|
||||
return diags
|
||||
}
|
||||
|
||||
addSet, removeSet := difference(state.SharedWith, plan.SharedWith)
|
||||
|
||||
for _, account := range addSet {
|
||||
accountId := uint64(account.(types.Int64).ValueInt64())
|
||||
tflog.Info(ctx, fmt.Sprintf("K8CISharedWithUpdate: Start add account with ID - %d to sharedWith access list to k8ci with ID - %d", accountId, k8ciid))
|
||||
req := k8ci.AccessAddRequest{
|
||||
K8CIID: k8ciid,
|
||||
AccountId: accountId,
|
||||
}
|
||||
res, err := c.CloudBroker().K8CI().AccessAdd(ctx, req)
|
||||
tflog.Info(ctx, "K8CISharedWithUpdate: response from CloudBroker().K8CI().AccessAdd", map[string]any{"k8ci_id": plan.Id.ValueString(), "response": res})
|
||||
if err != nil {
|
||||
diags.AddError(fmt.Sprintf("K8CISharedWithUpdate: Cannot add account with ID - %d", accountId), err.Error())
|
||||
}
|
||||
}
|
||||
|
||||
if diags.HasError() {
|
||||
tflog.Error(ctx, "K8CISharedWithUpdate: Errors occurred while managing add accounts")
|
||||
return diags
|
||||
}
|
||||
|
||||
for _, account := range removeSet {
|
||||
accountId := uint64(account.(types.Int64).ValueInt64())
|
||||
tflog.Info(ctx, fmt.Sprintf("K8CISharedWithUpdate: Start remove account with ID - %d from sharedWith access list to k8ci with ID - %d", accountId, k8ciid))
|
||||
req := k8ci.AccessRemoveRequest{
|
||||
K8CIID: k8ciid,
|
||||
AccountId: accountId,
|
||||
}
|
||||
res, err := c.CloudBroker().K8CI().AccessRemove(ctx, req)
|
||||
tflog.Info(ctx, "K8CISharedWithUpdate: response from CloudBroker().K8CI().AccessRemove", map[string]any{"k8ci_id": plan.Id.ValueString(), "response": res})
|
||||
if err != nil {
|
||||
diags.AddError(fmt.Sprintf("K8CISharedWithUpdate: Cannot remove account with ID - %d", accountId), err.Error())
|
||||
}
|
||||
}
|
||||
|
||||
if diags.HasError() {
|
||||
tflog.Error(ctx, "K8CISharedWithUpdate: Errors occurred while managing remove accounts")
|
||||
return diags
|
||||
}
|
||||
|
||||
tflog.Info(ctx, "K8CISharedWithUpdate: sharedWith access list is successfully update", map[string]any{"k8ci_id": k8ciid})
|
||||
return nil
|
||||
}
|
||||
|
||||
// difference returns lists added and removed values
|
||||
func difference(oldSet, newSet types.List) (added, removed []any) {
|
||||
oldMap := make(map[interface{}]struct{})
|
||||
newMap := make(map[interface{}]struct{})
|
||||
|
||||
for _, elem := range oldSet.Elements() {
|
||||
oldMap[elem] = struct{}{}
|
||||
}
|
||||
|
||||
for _, elem := range newSet.Elements() {
|
||||
newMap[elem] = struct{}{}
|
||||
}
|
||||
|
||||
for elem := range newMap {
|
||||
if _, found := oldMap[elem]; !found {
|
||||
added = append(added, elem)
|
||||
}
|
||||
}
|
||||
|
||||
for elem := range oldMap {
|
||||
if _, found := newMap[elem]; !found {
|
||||
removed = append(removed, elem)
|
||||
}
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
Reference in New Issue
Block a user