-
Notifications
You must be signed in to change notification settings - Fork 9.3k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #33328 from chasse-code/f-shield-advanced
Feature: AWS Shield Advanced: DRT AccessRole ARN and Access Log Bucket Associations
- Loading branch information
Showing
9 changed files
with
1,324 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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
379
internal/service/shield/drt_access_log_bucket_association.go
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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"` | ||
} |
Oops, something went wrong.