You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
terraform-provider-dynamix/internal/service/cloudbroker/k8ci/utilities/utility_resource_k8ci.go

332 lines
11 KiB

2 months ago
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"
"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudbroker/k8ci"
3 weeks ago
"repository.basistech.ru/BASIS/terraform-provider-dynamix/internal/client"
2 months ago
"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
}
3 weeks ago
func K8CIResourceEnableDisable(ctx context.Context, plan *models.ResourceK8CIModel, c *client.Client) diag.Diagnostics {
2 months ago
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
}
3 weeks ago
func K8CIResourceCheckPresence(ctx context.Context, plan *models.ResourceK8CIModel, c *client.Client) (*k8ci.RecordK8CI, diag.Diagnostics) {
2 months ago
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
}
3 weeks ago
func K8CIReadStatus(ctx context.Context, plan *models.ResourceK8CIModel, c *client.Client) diag.Diagnostics {
2 months ago
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
}
3 weeks ago
func K8CIRestore(ctx context.Context, plan *models.ResourceK8CIModel, c *client.Client) diag.Diagnostics {
2 months ago
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
}
3 weeks ago
func K8CISharedWithUpdate(ctx context.Context, plan *models.ResourceK8CIModel, state *models.ResourceK8CIModel, c *client.Client) diag.Diagnostics {
2 months ago
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
}