-
Notifications
You must be signed in to change notification settings - Fork 631
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 #4734 from mrusso19/add-content-scanning-expression
- Loading branch information
Showing
9 changed files
with
408 additions
and
0 deletions.
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,3 @@ | ||
```release-note:new-resource | ||
cloudflare_content_scanning_expression | ||
``` |
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,49 @@ | ||
--- | ||
page_title: "cloudflare_content_scanning_expression Resource - Cloudflare" | ||
subcategory: "" | ||
description: |- | ||
Provides a Cloudflare Content Scanning Expression resource for managing custom scan expression within a specific zone. | ||
--- | ||
|
||
# cloudflare_content_scanning_expression (Resource) | ||
|
||
Provides a Cloudflare Content Scanning Expression resource for managing custom scan expression within a specific zone. | ||
|
||
## Example Usage | ||
|
||
```terraform | ||
# Enable Content Scanning before trying to add custom scan expressions | ||
resource "cloudflare_content_scanning" "example" { | ||
zone_id = "399c6f4950c01a5a141b99ff7fbcbd8b" | ||
enabled = true | ||
} | ||
resource "cloudflare_content_scanning_expression" "first_example" { | ||
zone_id = cloudflare_content_scanning.example.zone_id | ||
payload = "lookup_json_string(http.request.body.raw, \"file\")" | ||
} | ||
resource "cloudflare_content_scanning_expression" "second_example" { | ||
zone_id = cloudflare_content_scanning.example.zone_id | ||
payload = "lookup_json_string(http.request.body.raw, \"document\")" | ||
} | ||
``` | ||
<!-- schema generated by tfplugindocs --> | ||
## Schema | ||
|
||
### Required | ||
|
||
- `payload` (String) Custom scan expression to tell the content scanner where to find the content objects. | ||
- `zone_id` (String) The zone identifier to target for the resource. | ||
|
||
### Read-Only | ||
|
||
- `id` (String) The identifier of this resource. | ||
|
||
## Import | ||
|
||
Import is supported using the following syntax: | ||
|
||
```shell | ||
terraform import cloudflare_content_scanning_expression.example <zone_id>/<resource_id> | ||
``` |
1 change: 1 addition & 0 deletions
1
examples/resources/cloudflare_content_scanning_expression/import.sh
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 @@ | ||
terraform import cloudflare_content_scanning_expression.example <zone_id>/<resource_id> |
15 changes: 15 additions & 0 deletions
15
examples/resources/cloudflare_content_scanning_expression/resource.tf
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,15 @@ | ||
# Enable Content Scanning before trying to add custom scan expressions | ||
resource "cloudflare_content_scanning" "example" { | ||
zone_id = "399c6f4950c01a5a141b99ff7fbcbd8b" | ||
enabled = true | ||
} | ||
|
||
resource "cloudflare_content_scanning_expression" "first_example" { | ||
zone_id = cloudflare_content_scanning.example.zone_id | ||
payload = "lookup_json_string(http.request.body.raw, \"file\")" | ||
} | ||
|
||
resource "cloudflare_content_scanning_expression" "second_example" { | ||
zone_id = cloudflare_content_scanning.example.zone_id | ||
payload = "lookup_json_string(http.request.body.raw, \"document\")" | ||
} |
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
9 changes: 9 additions & 0 deletions
9
internal/framework/service/content_scanning_expression/model.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,9 @@ | ||
package content_scanning_expression | ||
|
||
import "github.com/hashicorp/terraform-plugin-framework/types" | ||
|
||
type ContentScanningExpressionModel struct { | ||
ZoneID types.String `tfsdk:"zone_id"` | ||
ID types.String `tfsdk:"id"` | ||
Payload types.String `tfsdk:"payload"` | ||
} |
196 changes: 196 additions & 0 deletions
196
internal/framework/service/content_scanning_expression/resource.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,196 @@ | ||
package content_scanning_expression | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
"strings" | ||
|
||
"github.com/cloudflare/cloudflare-go" | ||
"github.com/cloudflare/terraform-provider-cloudflare/internal/framework/muxclient" | ||
|
||
"github.com/hashicorp/terraform-plugin-framework/path" | ||
"github.com/hashicorp/terraform-plugin-framework/resource" | ||
"github.com/hashicorp/terraform-plugin-framework/types" | ||
) | ||
|
||
var ( | ||
_ resource.Resource = &ContentScanningExpressionResource{} | ||
_ resource.ResourceWithImportState = &ContentScanningExpressionResource{} | ||
) | ||
|
||
func NewResource() resource.Resource { | ||
return &ContentScanningExpressionResource{} | ||
} | ||
|
||
type ContentScanningExpressionResource struct { | ||
client *muxclient.Client | ||
} | ||
|
||
func (r *ContentScanningExpressionResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) { | ||
resp.TypeName = req.ProviderTypeName + "_content_scanning_expression" | ||
} | ||
|
||
func (r *ContentScanningExpressionResource) Configure(_ context.Context, req resource.ConfigureRequest, resp *resource.ConfigureResponse) { | ||
if req.ProviderData == nil { | ||
return | ||
} | ||
|
||
client, ok := req.ProviderData.(*muxclient.Client) | ||
|
||
if !ok { | ||
resp.Diagnostics.AddError( | ||
"unexpected resource configure type", | ||
fmt.Sprintf("Expected *muxclient.Client, got: %T. Please report this issue to the provider developers.", req.ProviderData), | ||
) | ||
|
||
return | ||
} | ||
|
||
r.client = client | ||
} | ||
|
||
func (r *ContentScanningExpressionResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) { | ||
var data ContentScanningExpressionModel | ||
diags := req.Plan.Get(ctx, &data) | ||
resp.Diagnostics.Append(diags...) | ||
if resp.Diagnostics.HasError() { | ||
return | ||
} | ||
params := cloudflare.ContentScanningAddCustomExpressionsParams{ | ||
Payloads: []cloudflare.ContentScanningCustomPayload{ | ||
{ | ||
Payload: data.Payload.ValueString(), | ||
}, | ||
}, | ||
} | ||
expressions, err := r.client.V1.ContentScanningAddCustomExpressions(ctx, cloudflare.ZoneIdentifier(data.ZoneID.ValueString()), params) | ||
if err != nil { | ||
resp.Diagnostics.AddError("Error creating a custom scan expression for Content Scanning", err.Error()) | ||
return | ||
} | ||
|
||
// The Add API returns a list of all exiting custom scan expression | ||
// loop until we find the newly created one, matching on payload | ||
// payload uniqueness is enforced by the service | ||
for _, exp := range expressions { | ||
if exp.Payload == data.Payload.ValueString() { | ||
data.ID = types.StringValue(exp.ID) | ||
break | ||
} | ||
} | ||
|
||
diags = resp.State.Set(ctx, &data) | ||
resp.Diagnostics.Append(diags...) | ||
} | ||
|
||
func (r *ContentScanningExpressionResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) { | ||
var state ContentScanningExpressionModel | ||
diags := req.State.Get(ctx, &state) | ||
resp.Diagnostics.Append(diags...) | ||
if resp.Diagnostics.HasError() { | ||
return | ||
} | ||
|
||
zoneID := state.ZoneID.ValueString() | ||
var foundExp cloudflare.ContentScanningCustomExpression | ||
expressions, err := r.client.V1.ContentScanningListCustomExpressions(ctx, cloudflare.ZoneIdentifier(zoneID), cloudflare.ContentScanningListCustomExpressionsParams{}) | ||
if err != nil { | ||
resp.Diagnostics.AddError("Error listing customs scan expressions for Content Scanning", err.Error()) | ||
return | ||
} | ||
|
||
// content scanning doens't offer a single get operation so | ||
// loop until we find the matching ID. | ||
for _, exp := range expressions { | ||
if exp.ID == state.ID.ValueString() { | ||
foundExp = exp | ||
break | ||
} | ||
} | ||
|
||
state.ID = types.StringValue(foundExp.ID) | ||
state.Payload = types.StringValue(foundExp.Payload) | ||
|
||
diags = resp.State.Set(ctx, &state) | ||
resp.Diagnostics.Append(diags...) | ||
} | ||
|
||
func (r *ContentScanningExpressionResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) { | ||
var plan ContentScanningExpressionModel | ||
diags := req.Plan.Get(ctx, &plan) | ||
resp.Diagnostics.Append(diags...) | ||
if resp.Diagnostics.HasError() { | ||
return | ||
} | ||
var state ContentScanningExpressionModel | ||
diags = req.State.Get(ctx, &state) | ||
resp.Diagnostics.Append(diags...) | ||
if resp.Diagnostics.HasError() { | ||
return | ||
} | ||
zoneID := cloudflare.ZoneIdentifier(plan.ZoneID.ValueString()) | ||
plan.ID = state.ID | ||
|
||
// API does not offer an update operation so we use delete/create | ||
delParams := cloudflare.ContentScanningDeleteCustomExpressionsParams{ID: plan.ID.ValueString()} | ||
_, err := r.client.V1.ContentScanningDeleteCustomExpression(ctx, zoneID, delParams) | ||
if err != nil { | ||
resp.Diagnostics.AddError("Error in Update while deleting custom scan expression for Content Scanning", err.Error()) | ||
return | ||
} | ||
createParams := cloudflare.ContentScanningAddCustomExpressionsParams{ | ||
Payloads: []cloudflare.ContentScanningCustomPayload{ | ||
{ | ||
Payload: plan.Payload.ValueString(), | ||
}, | ||
}, | ||
} | ||
expressions, err := r.client.V1.ContentScanningAddCustomExpressions(ctx, zoneID, createParams) | ||
if err != nil { | ||
resp.Diagnostics.AddError("Error in Update while creating a custom scan expression for Content Scanning", err.Error()) | ||
return | ||
} | ||
|
||
// The Add API returns a list of all exiting custom scan expression | ||
// loop until we find the newly created one, matching on payload | ||
// payload uniqueness is enforced by the service | ||
for _, exp := range expressions { | ||
if exp.Payload == plan.Payload.ValueString() { | ||
plan.ID = types.StringValue(exp.ID) | ||
break | ||
} | ||
} | ||
|
||
diags = resp.State.Set(ctx, &plan) | ||
resp.Diagnostics.Append(diags...) | ||
} | ||
|
||
func (r *ContentScanningExpressionResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) { | ||
var state ContentScanningExpressionModel | ||
diags := req.State.Get(ctx, &state) | ||
resp.Diagnostics.Append(diags...) | ||
if resp.Diagnostics.HasError() { | ||
return | ||
} | ||
zoneID := cloudflare.ZoneIdentifier(state.ZoneID.ValueString()) | ||
deleteParam := cloudflare.ContentScanningDeleteCustomExpressionsParams{ID: state.ID.ValueString()} | ||
_, err := r.client.V1.ContentScanningDeleteCustomExpression(ctx, zoneID, deleteParam) | ||
if err != nil { | ||
resp.Diagnostics.AddError("Error deleting custom scan expression for Content Scanning", err.Error()) | ||
return | ||
} | ||
} | ||
|
||
func (r *ContentScanningExpressionResource) ImportState(ctx context.Context, req resource.ImportStateRequest, resp *resource.ImportStateResponse) { | ||
idparts := strings.Split(req.ID, "/") | ||
if len(idparts) != 2 { | ||
resp.Diagnostics.AddError("error importing content scanning custom expression", "invalid ID specified. Please specify the ID as \"zone_id/resource_id\"") | ||
return | ||
} | ||
resp.Diagnostics.Append(resp.State.SetAttribute( | ||
ctx, path.Root("zone_id"), idparts[0], | ||
)...) | ||
resp.Diagnostics.Append(resp.State.SetAttribute( | ||
ctx, path.Root("id"), idparts[1], | ||
)...) | ||
} |
Oops, something went wrong.