Skip to content

Commit

Permalink
DWX-18781 Data warehouse catalog object management
Browse files Browse the repository at this point in the history
Adding creation and deletion options for the data warehouse catalog
object type.
  • Loading branch information
tevesz committed Oct 24, 2024
1 parent b763a5c commit f53bf93
Show file tree
Hide file tree
Showing 7 changed files with 722 additions and 0 deletions.
2 changes: 2 additions & 0 deletions provider/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import (
"github.com/cloudera/terraform-provider-cdp/resources/de"
"github.com/cloudera/terraform-provider-cdp/resources/dw"
dwaws "github.com/cloudera/terraform-provider-cdp/resources/dw/cluster/aws"
dwdatabasecatalog "github.com/cloudera/terraform-provider-cdp/resources/dw/databasecatalog"
"github.com/cloudera/terraform-provider-cdp/resources/environments"
"github.com/cloudera/terraform-provider-cdp/resources/iam"
"github.com/cloudera/terraform-provider-cdp/resources/ml"
Expand Down Expand Up @@ -250,6 +251,7 @@ func (p *CdpProvider) Resources(_ context.Context) []func() resource.Resource {
de.NewServiceResource,
dw.NewHiveResource,
dwaws.NewDwClusterResource,
dwdatabasecatalog.NewDwDatabaseCatalogResource,
}
}

Expand Down
2 changes: 2 additions & 0 deletions provider/provider_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import (
"github.com/cloudera/terraform-provider-cdp/resources/de"
"github.com/cloudera/terraform-provider-cdp/resources/dw"
dwaws "github.com/cloudera/terraform-provider-cdp/resources/dw/cluster/aws"
dwdatabasecatalog "github.com/cloudera/terraform-provider-cdp/resources/dw/databasecatalog"
"github.com/cloudera/terraform-provider-cdp/resources/environments"
"github.com/cloudera/terraform-provider-cdp/resources/iam"
"github.com/cloudera/terraform-provider-cdp/resources/ml"
Expand Down Expand Up @@ -636,6 +637,7 @@ func TestCdpProvider_Resources(t *testing.T) {
de.NewServiceResource,
dw.NewHiveResource,
dwaws.NewDwClusterResource,
dwdatabasecatalog.NewDwDatabaseCatalogResource,
}

provider := CdpProvider{testVersion}
Expand Down
42 changes: 42 additions & 0 deletions resources/dw/databasecatalog/model_catalog.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
// Copyright 2024 Cloudera. All Rights Reserved.
//
// This file is licensed under the Apache License Version 2.0 (the "License").
// You may not use this file except in compliance with the License.
// You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0.
//
// This file is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
// OF ANY KIND, either express or implied. Refer to the License for the specific
// permissions and limitations governing your use of the file.

package databasecatalog

import (
"time"

"github.com/hashicorp/terraform-plugin-framework/types"

"github.com/cloudera/terraform-provider-cdp/utils"
)

type resourceModel struct {
ID types.String `tfsdk:"id"`
Name types.String `tfsdk:"name"`
ClusterID types.String `tfsdk:"cluster_id"`
LastUpdated types.String `tfsdk:"last_updated"`
Status types.String `tfsdk:"status"`
PollingOptions *utils.PollingOptions `tfsdk:"polling_options"`
}

func (p *resourceModel) getPollingTimeout() time.Duration {
if p.PollingOptions != nil {
return time.Duration(p.PollingOptions.PollingTimeout.ValueInt64()) * time.Minute
}
return 40 * time.Minute
}

func (p *resourceModel) getCallFailureThreshold() int {
if p.PollingOptions != nil {
return int(p.PollingOptions.CallFailureThreshold.ValueInt64())
}
return 3
}
51 changes: 51 additions & 0 deletions resources/dw/databasecatalog/model_catalog_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
// Copyright 2024 Cloudera. All Rights Reserved.
//
// This file is licensed under the Apache License Version 2.0 (the "License").
// You may not use this file except in compliance with the License.
// You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0.
//
// This file is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
// OF ANY KIND, either express or implied. Refer to the License for the specific
// permissions and limitations governing your use of the file.

package databasecatalog

import (
"context"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/tfsdk"
"github.com/stretchr/testify/suite"
"testing"
"time"
)

type DwDatabaseCatalogModelTestSuite struct {
suite.Suite
rm *resourceModel
}

func TestDwModelDatabaseCatalogTestSuite(t *testing.T) {
suite.Run(t, new(DwDatabaseCatalogModelTestSuite))
}

func (s *DwDatabaseCatalogModelTestSuite) SetupSuite() {
req := resource.CreateRequest{
Plan: tfsdk.Plan{
Raw: createRawCatalogResource(),
Schema: testDatabaseCatalogSchema,
},
}
rm := &resourceModel{}
req.Plan.Get(context.Background(), &rm)
s.rm = rm
}

func (s *DwDatabaseCatalogModelTestSuite) TestGetPollingTimeout() {
timeout := s.rm.getPollingTimeout()
s.Equal(90*time.Minute, timeout)
}

func (s *DwDatabaseCatalogModelTestSuite) TestGetCallFailureThreshold() {
out := s.rm.getCallFailureThreshold()
s.Equal(3, out)
}
170 changes: 170 additions & 0 deletions resources/dw/databasecatalog/resource_catalog.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,170 @@
// Copyright 2024 Cloudera. All Rights Reserved.
//
// This file is licensed under the Apache License Version 2.0 (the "License").
// You may not use this file except in compliance with the License.
// You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0.
//
// This file is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
// OF ANY KIND, either express or implied. Refer to the License for the specific
// permissions and limitations governing your use of the file.

package databasecatalog

import (
"context"
"fmt"
"time"

"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/types"
"github.com/hashicorp/terraform-plugin-log/tflog"
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/retry"

"github.com/cloudera/terraform-provider-cdp/cdp-sdk-go/cdp"
"github.com/cloudera/terraform-provider-cdp/cdp-sdk-go/gen/dw/client/operations"
"github.com/cloudera/terraform-provider-cdp/cdp-sdk-go/gen/dw/models"
"github.com/cloudera/terraform-provider-cdp/utils"
)

type dwDatabaseCatalogResource struct {
client *cdp.Client
}

var (
_ resource.Resource = (*dwDatabaseCatalogResource)(nil)
_ resource.ResourceWithConfigure = (*dwDatabaseCatalogResource)(nil)
)

func NewDwDatabaseCatalogResource() resource.Resource {
return &dwDatabaseCatalogResource{}
}

func (r *dwDatabaseCatalogResource) Configure(_ context.Context, req resource.ConfigureRequest, resp *resource.ConfigureResponse) {
r.client = utils.GetCdpClientForResource(req, resp)
}

func (r *dwDatabaseCatalogResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_dw_database_catalog"
}

func (r *dwDatabaseCatalogResource) Schema(_ context.Context, _ resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = dwDefaultDatabaseCatalogSchema
}

func (r *dwDatabaseCatalogResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
// Retrieve values from plan
var plan resourceModel
diags := req.Plan.Get(ctx, &plan)
resp.Diagnostics.Append(diags...)
if resp.Diagnostics.HasError() {
return
}

clusterID := plan.ClusterID.ValueStringPointer()

if opts := plan.PollingOptions; !(opts != nil && opts.Async.ValueBool()) {
callFailedCount := 0
stateConf := &retry.StateChangeConf{
Pending: []string{"Accepted", "Creating", "Created", "Loading", "Starting"},
Target: []string{"Running"},
Delay: 30 * time.Second,
Timeout: plan.getPollingTimeout(),
PollInterval: 30 * time.Second,
Refresh: r.stateRefresh(ctx, clusterID, &callFailedCount, plan.getCallFailureThreshold()),
}
if _, err := stateConf.WaitForStateContext(ctx); err != nil {
resp.Diagnostics.AddError(
"Error waiting for Data Warehouse database catalog",
fmt.Sprintf("Could not create database catalog, unexpected error: %v", err),
)
return
}
}

catalog, err := r.getDatabaseCatalog(ctx, clusterID)
if err != nil {
resp.Diagnostics.AddError(
"Error finding Data Warehouse catalog", fmt.Sprintf("unexpected error: %v", err),
)
return
}
plan.ID = types.StringValue(catalog.ID)
plan.Name = types.StringValue(catalog.Name)
plan.Status = types.StringValue(catalog.Status)
plan.LastUpdated = types.StringValue(time.Now().Format(time.RFC850))

// Set state to fully populated data
diags = resp.State.Set(ctx, plan)
resp.Diagnostics.Append(diags...)
}

func (r *dwDatabaseCatalogResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
tflog.Warn(ctx, "Read operation is not implemented yet.")
}

func (r *dwDatabaseCatalogResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
tflog.Warn(ctx, "Update operation is not implemented yet.")
}

func (r *dwDatabaseCatalogResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
var state resourceModel
resp.Diagnostics.Append(req.State.Get(ctx, &state)...)
if resp.Diagnostics.HasError() {
return
}
// Catalog is deleted as part of environment deletion, no need to delete it explicitly
state.Status = types.StringValue("Deleted")
resp.State.Set(ctx, state)
}

func (r *dwDatabaseCatalogResource) stateRefresh(ctx context.Context, clusterID *string, callFailedCount *int, callFailureThreshold int) func() (any, string, error) {
return func() (any, string, error) {
tflog.Debug(ctx, "About to get DBCs")
catalogParams := operations.NewListDbcsParamsWithContext(ctx).WithInput(&models.ListDbcsRequest{ClusterID: clusterID})
// List existing catalogs
response, err := r.client.Dw.Operations.ListDbcs(catalogParams)
if err != nil {
tflog.Error(ctx,
fmt.Sprintf("could not list database catalogs, unexpected error: %s", err.Error()),
)
return nil, "", err
}
resp := response.GetPayload()
if len(resp.Dbcs) == 0 {
*callFailedCount++
if *callFailedCount <= callFailureThreshold {
tflog.Warn(ctx, fmt.Sprintf("could not find Data Warehouse database catalog "+
"due to [%s] but threshold limit is not reached yet (%d out of %d).", err.Error(), callFailedCount, callFailureThreshold))
return nil, "", nil
}
tflog.Error(ctx, fmt.Sprintf("error describing Data Warehouse database catalog due to [%s] "+
"failure threshold limit exceeded.", err.Error()))
return nil, "", err
}
if len(resp.Dbcs) > 1 {
err = fmt.Errorf("found more than one Data Warehouse database catalog for cluster %s", *clusterID)
tflog.Error(ctx, fmt.Sprintf("error describing Data Warehouse database catalog due to [%s] ", err.Error()))
return nil, "", err
}
*callFailedCount = 0

tflog.Debug(ctx, fmt.Sprintf("Found database catalog %s with status %s", resp.Dbcs[0].ID, resp.Dbcs[0].Status))
return resp.Dbcs[0], resp.Dbcs[0].Status, nil
}
}

func (r *dwDatabaseCatalogResource) getDatabaseCatalog(ctx context.Context, clusterID *string) (*models.DbcSummary, error) {
catalogParams := operations.NewListDbcsParamsWithContext(ctx).WithInput(&models.ListDbcsRequest{ClusterID: clusterID})
// List existing catalogs
response, err := r.client.Dw.Operations.ListDbcs(catalogParams)
if err != nil {
err = fmt.Errorf("could not list database catalogs, unexpected error: %s", err.Error())
return nil, err
}
resp := response.GetPayload()
if len(resp.Dbcs) != 1 {
err = fmt.Errorf("exactly one Data Warehouse database catalog should be deployed for cluster %s", *clusterID)
return nil, err
}
return resp.Dbcs[0], nil
}
Loading

0 comments on commit f53bf93

Please sign in to comment.