From c9c9205be7f09e9635ebbd5bf58a49d996846242 Mon Sep 17 00:00:00 2001 From: Lourens de Jager <165963988+lourens-octopus@users.noreply.github.com> Date: Tue, 3 Dec 2024 10:50:32 +1300 Subject: [PATCH] feat: Add Git trigger functionality --- docs/resources/git_trigger.md | 45 ++++ go.mod | 2 +- go.sum | 4 +- octopusdeploy_framework/framework_provider.go | 1 + .../resource_git_trigger.go | 240 ++++++++++++++++++ .../resource_git_trigger_test.go | 149 +++++++++++ .../schemas/git_trigger.go | 66 +++++ 7 files changed, 504 insertions(+), 3 deletions(-) create mode 100644 docs/resources/git_trigger.md create mode 100644 octopusdeploy_framework/resource_git_trigger.go create mode 100644 octopusdeploy_framework/resource_git_trigger_test.go create mode 100644 octopusdeploy_framework/schemas/git_trigger.go diff --git a/docs/resources/git_trigger.md b/docs/resources/git_trigger.md new file mode 100644 index 00000000..6376c904 --- /dev/null +++ b/docs/resources/git_trigger.md @@ -0,0 +1,45 @@ +--- +# generated by https://github.com/hashicorp/terraform-plugin-docs +page_title: "octopusdeploy_git_trigger Resource - terraform-provider-octopusdeploy" +subcategory: "" +description: |- + This resource manages Git triggers in Octopus Deploy +--- + +# octopusdeploy_git_trigger (Resource) + +This resource manages Git triggers in Octopus Deploy + + + + +## Schema + +### Required + +- `channel_id` (String) The ID of the channel in which the release will be created if the action type is CreateRelease. +- `name` (String) The name of this resource. +- `project_id` (String) The ID of the project to attach the trigger. +- `sources` (Attributes List) (see [below for nested schema](#nestedatt--sources)) + +### Optional + +- `description` (String) The description of this Git trigger.. +- `is_disabled` (Boolean) Disables the trigger from being run when set. +- `space_id` (String) The space ID associated with this Git trigger. + +### Read-Only + +- `id` (String) The unique ID for this resource. + + +### Nested Schema for `sources` + +Required: + +- `deployment_action_slug` (String) The deployment action slug. +- `exclude_file_paths` (List of String) The file paths to exclude. +- `git_dependency_name` (String) The git dependency name. +- `include_file_paths` (List of String) The file paths to include. + + diff --git a/go.mod b/go.mod index 7e10c51b..b7845b10 100644 --- a/go.mod +++ b/go.mod @@ -3,7 +3,7 @@ module github.com/OctopusDeploy/terraform-provider-octopusdeploy go 1.21 require ( - github.com/OctopusDeploy/go-octopusdeploy/v2 v2.60.0 + github.com/OctopusDeploy/go-octopusdeploy/v2 v2.63.0 github.com/OctopusSolutionsEngineering/OctopusTerraformTestFramework v0.0.0-20240729041805-46db6fb717b4 github.com/google/uuid v1.6.0 github.com/hashicorp/go-cty v1.4.1-0.20200723130312-85980079f637 diff --git a/go.sum b/go.sum index d8eeb086..1a52f3ae 100644 --- a/go.sum +++ b/go.sum @@ -20,8 +20,8 @@ github.com/Microsoft/hcsshim v0.12.4 h1:Ev7YUMHAHoWNm+aDSPzc5W9s6E2jyL1szpVDJeZ/ github.com/Microsoft/hcsshim v0.12.4/go.mod h1:Iyl1WVpZzr+UkzjekHZbV8o5Z9ZkxNGx6CtY2Qg/JVQ= github.com/OctopusDeploy/go-octodiff v1.0.0 h1:U+ORg6azniwwYo+O44giOw6TiD5USk8S4VDhOQ0Ven0= github.com/OctopusDeploy/go-octodiff v1.0.0/go.mod h1:Mze0+EkOWTgTmi8++fyUc6r0aLZT7qD9gX+31t8MmIU= -github.com/OctopusDeploy/go-octopusdeploy/v2 v2.60.0 h1:9j4IQ1UcAuaTytlBzQ7Mmoy/dLtofYfSGNiM22+sLXs= -github.com/OctopusDeploy/go-octopusdeploy/v2 v2.60.0/go.mod h1:ggvOXzMnq+w0pLg6C9zdjz6YBaHfO3B3tqmmB7JQdaw= +github.com/OctopusDeploy/go-octopusdeploy/v2 v2.63.0 h1:TshwN+IqKt21uY9aXzj0ou0Ew92uIi3+ZGTccVd9Z8g= +github.com/OctopusDeploy/go-octopusdeploy/v2 v2.63.0/go.mod h1:ggvOXzMnq+w0pLg6C9zdjz6YBaHfO3B3tqmmB7JQdaw= github.com/OctopusSolutionsEngineering/OctopusTerraformTestFramework v0.0.0-20240729041805-46db6fb717b4 h1:QfbVf0bOIRMp/WHAWsuVDB7KHoWnRsGbvDuOf2ua7k4= github.com/OctopusSolutionsEngineering/OctopusTerraformTestFramework v0.0.0-20240729041805-46db6fb717b4/go.mod h1:Oq9KbiRNDBB5jFmrwnrgLX0urIqR/1ptY18TzkqXm7M= github.com/ProtonMail/go-crypto v1.1.0-alpha.2 h1:bkyFVUP+ROOARdgCiJzNQo2V2kiB97LyUpzH9P6Hrlg= diff --git a/octopusdeploy_framework/framework_provider.go b/octopusdeploy_framework/framework_provider.go index 161b5461..f49536ea 100644 --- a/octopusdeploy_framework/framework_provider.go +++ b/octopusdeploy_framework/framework_provider.go @@ -127,6 +127,7 @@ func (p *octopusDeployFrameworkProvider) Resources(ctx context.Context) []func() NewScriptModuleResource, NewUserResource, NewServiceAccountOIDCIdentity, + NewGitTriggerResource, } } diff --git a/octopusdeploy_framework/resource_git_trigger.go b/octopusdeploy_framework/resource_git_trigger.go new file mode 100644 index 00000000..453b63d5 --- /dev/null +++ b/octopusdeploy_framework/resource_git_trigger.go @@ -0,0 +1,240 @@ +package octopusdeploy_framework + +import ( + "context" + "fmt" + "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/actions" + "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/filters" + "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/projects" + "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/triggers" + "github.com/OctopusDeploy/terraform-provider-octopusdeploy/internal/errors" + "github.com/OctopusDeploy/terraform-provider-octopusdeploy/octopusdeploy_framework/schemas" + "github.com/OctopusDeploy/terraform-provider-octopusdeploy/octopusdeploy_framework/util" + "github.com/hashicorp/terraform-plugin-framework/attr" + "github.com/hashicorp/terraform-plugin-framework/path" + "github.com/hashicorp/terraform-plugin-framework/resource" + "github.com/hashicorp/terraform-plugin-framework/types" + "github.com/hashicorp/terraform-plugin-log/tflog" +) + +type gitTriggerResource struct { + *Config +} + +func NewGitTriggerResource() resource.Resource { + return &gitTriggerResource{} +} + +var _ resource.ResourceWithImportState = &gitTriggerResource{} + +func (r *gitTriggerResource) Metadata(ctx context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) { + resp.TypeName = util.GetTypeName("git_trigger") +} + +func (r *gitTriggerResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) { + resp.Schema = schemas.GitTriggerSchema{}.GetResourceSchema() +} + +func (r *gitTriggerResource) Configure(_ context.Context, req resource.ConfigureRequest, resp *resource.ConfigureResponse) { + r.Config = ResourceConfiguration(req, resp) +} + +func (r *gitTriggerResource) ImportState(ctx context.Context, req resource.ImportStateRequest, resp *resource.ImportStateResponse) { + resource.ImportStatePassthroughID(ctx, path.Root("id"), req, resp) +} + +func (r *gitTriggerResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) { + var data *schemas.GitTriggerResourceModel + resp.Diagnostics.Append(req.Plan.Get(ctx, &data)...) + if resp.Diagnostics.HasError() { + return + } + + gitTriggerSources := convertListToGitTriggerSources(data.Sources) + + action := actions.NewCreateReleaseAction(data.ChannelId.ValueString()) + filter := filters.NewGitTriggerFilter(gitTriggerSources) + + client := r.Config.Client + + project, err := projects.GetByID(client, data.SpaceId.ValueString(), data.ProjectId.ValueString()) + + if err != nil { + resp.Diagnostics.AddError("error finding project", err.Error()) + return + } + + tflog.Info(ctx, fmt.Sprintf("creating Git trigger: %s", data.Name.ValueString())) + + projectTrigger := triggers.NewProjectTrigger(data.Name.ValueString(), data.Description.ValueString(), data.IsDisabled.ValueBool(), project, action, filter) + + createdGitTrigger, err := client.ProjectTriggers.Add(projectTrigger) + + if err != nil { + resp.Diagnostics.AddError("unable to create Git trigger", err.Error()) + return + } + + data.ID = types.StringValue(createdGitTrigger.GetID()) + data.Name = types.StringValue(createdGitTrigger.Name) + data.ProjectId = types.StringValue(createdGitTrigger.ProjectID) + data.SpaceId = types.StringValue(createdGitTrigger.SpaceID) + data.IsDisabled = types.BoolValue(createdGitTrigger.IsDisabled) + data.Sources = convertGitTriggerSourcesToList(createdGitTrigger.Filter.(*filters.GitTriggerFilter).Sources) + + tflog.Info(ctx, fmt.Sprintf("Git trigger created (%s)", data.ID)) + resp.Diagnostics.Append(resp.State.Set(ctx, &data)...) +} + +func (r *gitTriggerResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) { + var data *schemas.GitTriggerResourceModel + resp.Diagnostics.Append(req.State.Get(ctx, &data)...) + if resp.Diagnostics.HasError() { + return + } + + tflog.Info(ctx, fmt.Sprintf("reading Git Trigger (%s)", data.ID)) + + client := r.Config.Client + + gitTrigger, err := client.ProjectTriggers.GetByID(data.ID.ValueString()) + if err != nil { + if err := errors.ProcessApiErrorV2(ctx, resp, data, err, "error retrieving Git Trigger"); err != nil { + resp.Diagnostics.AddError("unable to load Git Trigger", err.Error()) + } + return + } + + data.ID = types.StringValue(gitTrigger.GetID()) + data.Name = types.StringValue(gitTrigger.Name) + data.ProjectId = types.StringValue(gitTrigger.ProjectID) + data.SpaceId = types.StringValue(gitTrigger.SpaceID) + data.IsDisabled = types.BoolValue(gitTrigger.IsDisabled) + data.Sources = convertGitTriggerSourcesToList(gitTrigger.Filter.(*filters.GitTriggerFilter).Sources) + + resp.Diagnostics.Append(resp.State.Set(ctx, &data)...) +} + +func (r *gitTriggerResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) { + var data, state *schemas.GitTriggerResourceModel + resp.Diagnostics.Append(req.Plan.Get(ctx, &data)...) + resp.Diagnostics.Append(req.State.Get(ctx, &state)...) + if resp.Diagnostics.HasError() { + return + } + + tflog.Debug(ctx, fmt.Sprintf("updating Git Trigger '%s'", data.ID.ValueString())) + + client := r.Config.Client + + gitTrigger, err := client.ProjectTriggers.GetByID(data.ID.ValueString()) + if err != nil { + resp.Diagnostics.AddError("unable to load Git Trigger", err.Error()) + return + } + + gitTriggerSources := convertListToGitTriggerSources(data.Sources) + action := actions.NewCreateReleaseAction(data.ChannelId.ValueString()) + filter := filters.NewGitTriggerFilter(gitTriggerSources) + project, err := projects.GetByID(client, data.SpaceId.ValueString(), data.ProjectId.ValueString()) + + if err != nil { + resp.Diagnostics.AddError("error finding project", err.Error()) + return + } + + updatedGitTrigger := triggers.NewProjectTrigger(data.Name.ValueString(), data.Description.ValueString(), data.IsDisabled.ValueBool(), project, action, filter) + updatedGitTrigger.ID = gitTrigger.ID + + updatedGitTrigger, err = client.ProjectTriggers.Update(updatedGitTrigger) + tflog.Info(ctx, fmt.Sprintf("Git Trigger updated (%s)", data.ID)) + + resp.Diagnostics.Append(resp.State.Set(ctx, &data)...) +} + +func (r *gitTriggerResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) { + var data schemas.GitTriggerResourceModel + + resp.Diagnostics.Append(req.State.Get(ctx, &data)...) + if resp.Diagnostics.HasError() { + return + } + + client := r.Config.Client + + if err := client.ProjectTriggers.DeleteByID(data.ID.ValueString()); err != nil { + resp.Diagnostics.AddError("unable to delete Git Trigger", err.Error()) + return + } +} + +func convertListToGitTriggerSources(list types.List) []filters.GitTriggerSource { + var gitTriggerSources []filters.GitTriggerSource + + for _, elem := range list.Elements() { + obj := elem.(types.Object) + attrs := obj.Attributes() + + deploymentActionSlug := attrs["deployment_action_slug"].(types.String).ValueString() + gitDependencyName := attrs["git_dependency_name"].(types.String).ValueString() + includeFilePaths := convertToStringSlice(attrs["include_file_paths"].(types.List)) + excludeFilePaths := convertToStringSlice(attrs["exclude_file_paths"].(types.List)) + + gitTriggerSource := filters.GitTriggerSource{ + DeploymentActionSlug: deploymentActionSlug, + GitDependencyName: gitDependencyName, + IncludeFilePaths: includeFilePaths, + ExcludeFilePaths: excludeFilePaths, + } + + gitTriggerSources = append(gitTriggerSources, gitTriggerSource) + } + + return gitTriggerSources +} + +func convertToStringSlice(list types.List) []string { + var result []string + for _, elem := range list.Elements() { + result = append(result, elem.(types.String).ValueString()) + } + return result +} + +func convertGitTriggerSourcesToList(gitTriggerSources []filters.GitTriggerSource) types.List { + var elements []attr.Value + + for _, source := range gitTriggerSources { + attributes := map[string]attr.Value{ + "deployment_action_slug": types.StringValue(source.DeploymentActionSlug), + "git_dependency_name": types.StringValue(source.GitDependencyName), + "include_file_paths": convertStringSliceToList(source.IncludeFilePaths), + "exclude_file_paths": convertStringSliceToList(source.ExcludeFilePaths), + } + objectValue, _ := types.ObjectValue(sourcesObjectType(), attributes) + elements = append(elements, objectValue) + } + + listValue, _ := types.ListValue(types.ObjectType{AttrTypes: sourcesObjectType()}, elements) + return listValue +} + +func convertStringSliceToList(strings []string) types.List { + var elements []attr.Value + + for _, str := range strings { + elements = append(elements, types.StringValue(str)) + } + + listValue, _ := types.ListValue(types.StringType, elements) + return listValue +} + +func sourcesObjectType() map[string]attr.Type { + return map[string]attr.Type{ + "deployment_action_slug": types.StringType, + "git_dependency_name": types.StringType, + "include_file_paths": types.ListType{ElemType: types.StringType}, + "exclude_file_paths": types.ListType{ElemType: types.StringType}, + } +} diff --git a/octopusdeploy_framework/resource_git_trigger_test.go b/octopusdeploy_framework/resource_git_trigger_test.go new file mode 100644 index 00000000..f7daee5b --- /dev/null +++ b/octopusdeploy_framework/resource_git_trigger_test.go @@ -0,0 +1,149 @@ +package octopusdeploy_framework + +import ( + "fmt" + "github.com/hashicorp/terraform-plugin-testing/helper/acctest" + "github.com/hashicorp/terraform-plugin-testing/helper/resource" + "github.com/hashicorp/terraform-plugin-testing/terraform" + "strconv" + "strings" + "testing" +) + +type gitTriggerSourcesTestData struct { + deploymentActionSlug string + gitDependencyName string + includeFilePaths []string + excludeFilePaths []string +} + +type gitTriggerTestData struct { + name string + description string + projectId string + spaceId string + channelId string + isDisabled bool + sources []gitTriggerSourcesTestData +} + +func TestAccOctopusDeployGitTrigger(t *testing.T) { + localName := acctest.RandStringFromCharSet(20, acctest.CharSetAlpha) + prefix := "octopusdeploy_git_trigger." + localName + createData := gitTriggerTestData{ + name: acctest.RandStringFromCharSet(20, acctest.CharSetAlpha), + description: acctest.RandStringFromCharSet(20, acctest.CharSetAlpha), + projectId: acctest.RandStringFromCharSet(20, acctest.CharSetAlpha), + spaceId: acctest.RandStringFromCharSet(20, acctest.CharSetAlpha), + isDisabled: false, + sources: []gitTriggerSourcesTestData{ + { + deploymentActionSlug: acctest.RandStringFromCharSet(20, acctest.CharSetAlpha), + gitDependencyName: acctest.RandStringFromCharSet(20, acctest.CharSetAlpha), + includeFilePaths: []string{acctest.RandStringFromCharSet(20, acctest.CharSetAlpha)}, + excludeFilePaths: []string{acctest.RandStringFromCharSet(20, acctest.CharSetAlpha)}, + }, + }, + } + updateData := gitTriggerTestData{ + name: createData.name + "-updated", + description: createData.description + "-updated", + projectId: createData.projectId + "-updated", + spaceId: createData.spaceId + "-updated", + isDisabled: true, + sources: []gitTriggerSourcesTestData{ + { + deploymentActionSlug: createData.sources[0].deploymentActionSlug + "-updated", + gitDependencyName: createData.sources[0].gitDependencyName + "-updated", + includeFilePaths: []string{createData.sources[0].includeFilePaths[0] + "-updated"}, + excludeFilePaths: []string{createData.sources[0].excludeFilePaths[0] + "-updated"}, + }, + }, + } + + resource.Test(t, resource.TestCase{ + CheckDestroy: func(s *terraform.State) error { return testGitTriggerCheckDestroy(s) }, + PreCheck: func() { TestAccPreCheck(t) }, + ProtoV6ProviderFactories: ProtoV6ProviderFactories(), + Steps: []resource.TestStep{ + { + Config: testGitTriggerBasic(createData, localName), + Check: testAssertGitTriggerAttributes(createData, prefix), + }, + { + Config: testGitTriggerBasic(updateData, localName), + Check: testAssertGitTriggerAttributes(updateData, prefix), + }, + }, + }) +} + +func testGitTriggerBasic(data gitTriggerTestData, localName string) string { + return fmt.Sprintf(` + resource "octopusdeploy_azure_container_registry" "%s" { + name = "%s" + space_id = "%s" + project_id = "%s" + channel_id = "%s" + is_disabled = "%t" + sources = [%s] + } + `, + localName, + data.name, + data.spaceId, + data.projectId, + data.channelId, + data.isDisabled, + convertGitTriggerSourcesToString(data.sources), + ) +} + +func testAssertGitTriggerAttributes(expected gitTriggerTestData, prefix string) resource.TestCheckFunc { + return resource.ComposeTestCheckFunc( + resource.TestCheckResourceAttr(prefix, "name", expected.name), + resource.TestCheckResourceAttr(prefix, "space_id", expected.spaceId), + resource.TestCheckResourceAttr(prefix, "project_id", expected.projectId), + resource.TestCheckResourceAttr(prefix, "channel_id", expected.channelId), + resource.TestCheckResourceAttr(prefix, "is_disabled", strconv.FormatBool(expected.isDisabled)), + resource.TestCheckResourceAttr(prefix, "sources", convertGitTriggerSourcesToString(expected.sources)), + ) +} + +func testGitTriggerCheckDestroy(s *terraform.State) error { + for _, rs := range s.RootModule().Resources { + if rs.Type != "octopusdeploy_git_trigger" { + continue + } + + projectTrigger, err := octoClient.ProjectTriggers.GetByID(rs.Primary.ID) + if err == nil && projectTrigger != nil { + return fmt.Errorf("azure container registry feed (%s) still exists", rs.Primary.ID) + } + } + + return nil +} + +func convertGitTriggerSourcesToString(sources []gitTriggerSourcesTestData) string { + var result string + for _, source := range sources { + result += fmt.Sprintf(` + { + deployment_action_slug = "%s" + git_dependency_name = "%s" + include_file_paths = [%s] + exclude_file_paths = [%s] + }`, + source.deploymentActionSlug, + source.gitDependencyName, + convertStringSliceToString(source.includeFilePaths), + convertStringSliceToString(source.excludeFilePaths), + ) + } + return result +} + +func convertStringSliceToString(slice []string) string { + return fmt.Sprintf(`"%s"`, strings.Join(slice, `", "`)) +} diff --git a/octopusdeploy_framework/schemas/git_trigger.go b/octopusdeploy_framework/schemas/git_trigger.go new file mode 100644 index 00000000..b4fe25c4 --- /dev/null +++ b/octopusdeploy_framework/schemas/git_trigger.go @@ -0,0 +1,66 @@ +package schemas + +import ( + "github.com/OctopusDeploy/terraform-provider-octopusdeploy/octopusdeploy_framework/util" + datasourceSchema "github.com/hashicorp/terraform-plugin-framework/datasource/schema" + resourceSchema "github.com/hashicorp/terraform-plugin-framework/resource/schema" + "github.com/hashicorp/terraform-plugin-framework/types" +) + +type GitTriggerSchema struct{} + +var _ EntitySchema = GitTriggerSchema{} + +func (d GitTriggerSchema) GetResourceSchema() resourceSchema.Schema { + return resourceSchema.Schema{ + Description: "This resource manages Git triggers in Octopus Deploy", + Attributes: map[string]resourceSchema.Attribute{ + "id": GetIdResourceSchema(), + "name": GetNameResourceSchema(true), + "description": GetDescriptionResourceSchema("Git trigger."), + "space_id": GetSpaceIdResourceSchema("Git trigger"), + "project_id": GetRequiredStringResourceSchema("The ID of the project to attach the trigger."), + "channel_id": GetRequiredStringResourceSchema("The ID of the channel in which the release will be created if the action type is CreateRelease."), + "sources": GetSourcesAttributeSchema(), + "is_disabled": GetOptionalBooleanResourceAttribute("Disables the trigger from being run when set.", false), + }, + } +} + +func GetSourcesAttributeSchema() datasourceSchema.ListNestedAttribute { + return datasourceSchema.ListNestedAttribute{ + Required: true, + NestedObject: datasourceSchema.NestedAttributeObject{ + Attributes: map[string]datasourceSchema.Attribute{ + "deployment_action_slug": util.DataSourceString().Required().Description("The deployment action slug.").Build(), + "git_dependency_name": util.DataSourceString().Required().Description("The git dependency name.").Build(), + "include_file_paths": resourceSchema.ListAttribute{ + Required: true, + ElementType: types.StringType, + Description: "The file paths to include.", + }, + "exclude_file_paths": resourceSchema.ListAttribute{ + Required: true, + ElementType: types.StringType, + Description: "The file paths to exclude.", + }, + }, + }, + } +} + +func (d GitTriggerSchema) GetDatasourceSchema() datasourceSchema.Schema { + return datasourceSchema.Schema{} +} + +type GitTriggerResourceModel struct { + Name types.String `tfsdk:"name"` + Description types.String `tfsdk:"description"` + SpaceId types.String `tfsdk:"space_id"` + ProjectId types.String `tfsdk:"project_id"` + ChannelId types.String `tfsdk:"channel_id"` + Sources types.List `tfsdk:"sources"` + IsDisabled types.Bool `tfsdk:"is_disabled"` + + ResourceModel +}