Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature: AWS Shield Advanced: DRT AccessRole ARN and Access Log Bucket Associations #33328

Merged
merged 16 commits into from
Sep 6, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions .changelog/33328.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
```release-note:new-resource
aws_shield_drt_access_log_bucket_association
```

```release-note:new-resource
aws_shield_drt_access_role_arn_association
```
379 changes: 379 additions & 0 deletions internal/service/shield/drt_access_log_bucket_association.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,379 @@
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: MPL-2.0

package shield

import (
"context"
"errors"
"time"

"github.com/aws/aws-sdk-go/aws"
"github.com/aws/aws-sdk-go/service/shield"
"github.com/hashicorp/terraform-plugin-framework-timeouts/resource/timeouts"
"github.com/hashicorp/terraform-plugin-framework-validators/stringvalidator"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/resource/schema/planmodifier"
"github.com/hashicorp/terraform-plugin-framework/resource/schema/stringplanmodifier"
"github.com/hashicorp/terraform-plugin-framework/schema/validator"
"github.com/hashicorp/terraform-plugin-framework/types"
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/retry"
"github.com/hashicorp/terraform-provider-aws/internal/create"
"github.com/hashicorp/terraform-provider-aws/internal/framework"
"github.com/hashicorp/terraform-provider-aws/internal/framework/flex"
"github.com/hashicorp/terraform-provider-aws/internal/tfresource"
"github.com/hashicorp/terraform-provider-aws/names"
)

// @FrameworkResource(name="DRT Access Log Bucket Association")
func newResourceDRTAccessLogBucketAssociation(_ context.Context) (resource.ResourceWithConfigure, error) {
r := &resourceDRTAccessLogBucketAssociation{}

r.SetDefaultCreateTimeout(30 * time.Minute)
r.SetDefaultUpdateTimeout(30 * time.Minute)
r.SetDefaultDeleteTimeout(30 * time.Minute)

return r, nil
}

const (
ResNameDRTAccessLogBucketAssociation = "DRT Access Log Bucket Association"
)

type resourceDRTAccessLogBucketAssociation struct {
framework.ResourceWithConfigure
framework.WithTimeouts
}

func (r *resourceDRTAccessLogBucketAssociation) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = "aws_shield_drt_access_log_bucket_association"
}

func (r *resourceDRTAccessLogBucketAssociation) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
Attributes: map[string]schema.Attribute{
"id": schema.StringAttribute{ // required by hashicorps terraform plugin testing framework
DeprecationMessage: "id is only for framework compatibility and not used by the provider",
MarkdownDescription: "The ID of the directory.",
Computed: true,
PlanModifiers: []planmodifier.String{
stringplanmodifier.UseStateForUnknown(),
},
},
"log_bucket": schema.StringAttribute{
Required: true,
Validators: []validator.String{
// Validate string value length must be at least 3 characters and max 63.
stringvalidator.LengthBetween(3, 63),
},
},
"role_arn_association_id": schema.StringAttribute{
Required: true,
},
},
Blocks: map[string]schema.Block{
"timeouts": timeouts.Block(ctx, timeouts.Opts{
Create: true,
Delete: true,
Read: true,
}),
},
}
}

func (r *resourceDRTAccessLogBucketAssociation) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
conn := r.Meta().ShieldConn(ctx)

var plan resourceDRTAccessLogBucketAssociationData
resp.Diagnostics.Append(req.Plan.Get(ctx, &plan)...)
if resp.Diagnostics.HasError() {
return
}

in := &shield.AssociateDRTLogBucketInput{
LogBucket: aws.String(plan.LogBucket.ValueString()),
}
out, err := conn.AssociateDRTLogBucketWithContext(ctx, in)
if err != nil {
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionCreating, ResNameDRTAccessLogBucketAssociation, plan.LogBucket.String(), err),
err.Error(),
)
return
}
if out == nil {
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionCreating, ResNameDRTAccessLogBucketAssociation, plan.LogBucket.String(), nil),
errors.New("empty output").Error(),
)
return
}

createTimeout := r.CreateTimeout(ctx, plan.Timeouts)
_, err = waitDRTAccessLogBucketAssociationCreated(ctx, conn, plan.LogBucket.ValueString(), createTimeout)
if err != nil {
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionWaitingForCreation, ResNameDRTAccessLogBucketAssociation, plan.LogBucket.String(), err),
err.Error(),
)
return
}
plan.ID = types.StringValue(plan.LogBucket.ValueString())

resp.Diagnostics.Append(resp.State.Set(ctx, plan)...)
}

func (r *resourceDRTAccessLogBucketAssociation) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
conn := r.Meta().ShieldConn(ctx)

var state resourceDRTAccessLogBucketAssociationData
resp.Diagnostics.Append(req.State.Get(ctx, &state)...)
if resp.Diagnostics.HasError() {
return
}
in := &shield.DescribeDRTAccessInput{}

out, err := conn.DescribeDRTAccessWithContext(ctx, in)
if tfresource.NotFound(err) {
resp.State.RemoveResource(ctx)
return
}

if err != nil {
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionSetting, ResNameDRTAccessLogBucketAssociation, state.LogBucket.String(), err),
err.Error(),
)
return
}
var associatedLogBucket *string
if out != nil {
associatedLogBucket = getAssociatedLogBucket(state.LogBucket.ValueString(), out.LogBucketList)
if len(out.LogBucketList) > 0 && associatedLogBucket == nil {
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionSetting, ResNameDRTAccessLogBucketAssociation, state.LogBucket.String(), nil),
errors.New("Log Bucket not in list").Error(),
)
}
}

if state.ID.IsNull() || state.ID.IsUnknown() {
// Setting ID of state - required by hashicorps terraform plugin testing framework for Import. See issue https://github.com/hashicorp/terraform-plugin-testing/issues/84
state.ID = types.StringValue(state.LogBucket.ValueString())
}
state.LogBucket = flex.StringToFramework(ctx, associatedLogBucket)

resp.Diagnostics.Append(resp.State.Set(ctx, &state)...)
}

func getAssociatedLogBucket(bucket string, bucketList []*string) *string {
for _, bkt := range bucketList {
if aws.StringValue(bkt) == bucket {
return bkt
}
}
return nil
}

func (r *resourceDRTAccessLogBucketAssociation) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
conn := r.Meta().ShieldConn(ctx)

var plan, state resourceDRTAccessLogBucketAssociationData
resp.Diagnostics.Append(req.Plan.Get(ctx, &plan)...)
resp.Diagnostics.Append(req.State.Get(ctx, &state)...)
if resp.Diagnostics.HasError() {
return
}

if !plan.LogBucket.Equal(state.LogBucket) {
in := &shield.AssociateDRTLogBucketInput{
LogBucket: aws.String(plan.LogBucket.ValueString()),
}
out, err := conn.AssociateDRTLogBucketWithContext(ctx, in)
if err != nil {
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionUpdating, ResNameDRTAccessLogBucketAssociation, plan.LogBucket.String(), err),
err.Error(),
)
return
}
if out == nil {
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionUpdating, ResNameDRTAccessLogBucketAssociation, plan.LogBucket.String(), nil),
errors.New("empty output").Error(),
)
return
}
}

updateTimeout := r.UpdateTimeout(ctx, plan.Timeouts)
_, err := waitDRTAccessLogBucketAssociationUpdated(ctx, conn, plan.LogBucket.ValueString(), updateTimeout)
if err != nil {
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionWaitingForUpdate, ResNameDRTAccessLogBucketAssociation, plan.LogBucket.String(), err),
err.Error(),
)
return
}

resp.Diagnostics.Append(resp.State.Set(ctx, &plan)...)
}

func (r *resourceDRTAccessLogBucketAssociation) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
conn := r.Meta().ShieldConn(ctx)

var state resourceDRTAccessLogBucketAssociationData
resp.Diagnostics.Append(req.State.Get(ctx, &state)...)
if resp.Diagnostics.HasError() {
return
}
if state.LogBucket.ValueString() == "" {
return
}

in := &shield.DisassociateDRTLogBucketInput{
LogBucket: aws.String(state.LogBucket.ValueString()),
}

_, err := conn.DisassociateDRTLogBucketWithContext(ctx, in)
if err != nil {
var nfe *shield.ResourceNotFoundException
if errors.As(err, &nfe) {
return
}
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionDeleting, ResNameDRTAccessLogBucketAssociation, state.LogBucket.String(), err),
err.Error(),
)
return
}
deleteTimeout := r.DeleteTimeout(ctx, state.Timeouts)
_, err = waitDRTAccessLogBucketAssociationDeleted(ctx, conn, state.LogBucket.ValueString(), deleteTimeout)
if err != nil {
resp.Diagnostics.AddError(
create.ProblemStandardMessage(names.Shield, create.ErrActionWaitingForDeletion, ResNameDRTAccessLogBucketAssociation, state.LogBucket.String(), err),
err.Error(),
)
return
}
}

const (
statusChangePending = "Pending"
statusDeleting = "Deleting"
statusNormal = "Normal"
statusUpdated = "Updated"
)

func waitDRTAccessLogBucketAssociationCreated(ctx context.Context, conn *shield.Shield, bucket string, timeout time.Duration) (*shield.DescribeDRTAccessOutput, error) {
stateConf := &retry.StateChangeConf{
Pending: []string{},
Target: []string{statusNormal},
Refresh: statusDRTAccessLogBucketAssociation(ctx, conn, bucket),
Timeout: timeout,
NotFoundChecks: 2,
ContinuousTargetOccurence: 2,
}

outputRaw, err := stateConf.WaitForStateContext(ctx)
if out, ok := outputRaw.(*shield.DescribeDRTAccessOutput); ok {
return out, err
}

return nil, err
}

func waitDRTAccessLogBucketAssociationUpdated(ctx context.Context, conn *shield.Shield, bucket string, timeout time.Duration) (*shield.DescribeDRTAccessOutput, error) {
stateConf := &retry.StateChangeConf{
Pending: []string{statusChangePending},
Target: []string{statusUpdated},
Refresh: statusDRTAccessLogBucketAssociation(ctx, conn, bucket),
Timeout: timeout,
NotFoundChecks: 2,
ContinuousTargetOccurence: 2,
}

outputRaw, err := stateConf.WaitForStateContext(ctx)
if out, ok := outputRaw.(*shield.DescribeDRTAccessOutput); ok {
return out, err
}

return nil, err
}

func waitDRTAccessLogBucketAssociationDeleted(ctx context.Context, conn *shield.Shield, bucket string, timeout time.Duration) (*shield.DescribeDRTAccessOutput, error) {
stateConf := &retry.StateChangeConf{
Pending: []string{statusDeleting, statusNormal},
Target: []string{},
Refresh: statusDRTAccessLogBucketAssociation(ctx, conn, bucket),
Timeout: timeout,
}

outputRaw, err := stateConf.WaitForStateContext(ctx)
if out, ok := outputRaw.(*shield.DescribeDRTAccessOutput); ok {
return out, err
}

return nil, err
}

func statusDRTAccessLogBucketAssociation(ctx context.Context, conn *shield.Shield, bucket string) retry.StateRefreshFunc {
return func() (interface{}, string, error) {
out, err := describeDRTAccessLogBucketAssociation(ctx, conn, bucket)
if tfresource.NotFound(err) {
return nil, "", nil
}

if err != nil {
return nil, "", err
}
if out == nil || out.LogBucketList == nil || len(out.LogBucketList) == 0 {
return nil, "", nil
}

if out != nil && len(out.LogBucketList) > 0 {
for _, bkt := range out.LogBucketList {
if aws.StringValue(bkt) == bucket {
return out, statusNormal, nil
}
}
return nil, "", nil
}

return out, statusNormal, nil
}
}

func describeDRTAccessLogBucketAssociation(ctx context.Context, conn *shield.Shield, bucketName string) (*shield.DescribeDRTAccessOutput, error) {
in := &shield.DescribeDRTAccessInput{}

out, err := conn.DescribeDRTAccessWithContext(ctx, in)
if err != nil {
var nfe *shield.ResourceNotFoundException
if errors.As(err, &nfe) {
return nil, &retry.NotFoundError{
LastError: err,
LastRequest: in,
}
}
}

if out == nil || out.LogBucketList == nil || len(out.LogBucketList) == 0 {
return nil, tfresource.NewEmptyResultError(in)
}

for _, bucket := range out.LogBucketList {
if aws.StringValue(bucket) == bucketName {
return out, nil
}
}
return nil, err
}

type resourceDRTAccessLogBucketAssociationData struct {
ID types.String `tfsdk:"id"`
RoleArnAssociationID types.String `tfsdk:"role_arn_association_id"`
LogBucket types.String `tfsdk:"log_bucket"`
Timeouts timeouts.Value `tfsdk:"timeouts"`
}
Loading