From 5f1bbe628e7e48da66e537e02a0a35532cffe40a Mon Sep 17 00:00:00 2001
From: Lourens de Jager <165963988+lourens-octopus@users.noreply.github.com>
Date: Tue, 12 Nov 2024 10:00:10 +1300
Subject: [PATCH] Add Git Triggers
---
docs/resources/git_trigger.md | 44 ++++++
.../octopusdeploy_git_trigger/import.sh | 1 +
.../octopusdeploy_git_trigger/resource.tf | 12 ++
go.mod | 2 +-
go.sum | 1 +
octopusdeploy/provider.go | 1 +
octopusdeploy/resource_git_trigger.go | 130 ++++++++++++++++++
octopusdeploy/schema_git_trigger.go | 42 ++++++
.../schema_git_trigger_source_package.go | 66 +++++++++
9 files changed, 298 insertions(+), 1 deletion(-)
create mode 100644 docs/resources/git_trigger.md
create mode 100644 examples/resources/octopusdeploy_git_trigger/import.sh
create mode 100644 examples/resources/octopusdeploy_git_trigger/resource.tf
create mode 100644 octopusdeploy/resource_git_trigger.go
create mode 100644 octopusdeploy/schema_git_trigger.go
create mode 100644 octopusdeploy/schema_git_trigger_source_package.go
diff --git a/docs/resources/git_trigger.md b/docs/resources/git_trigger.md
new file mode 100644
index 00000000..4d371d4f
--- /dev/null
+++ b/docs/resources/git_trigger.md
@@ -0,0 +1,44 @@
+---
+# generated by https://github.com/hashicorp/terraform-plugin-docs
+page_title: "octopusdeploy_git_trigger Resource - terraform-provider-octopusdeploy"
+subcategory: ""
+description: |-
+ This resource manages Git repository triggers in Octopus Deploy.
+---
+
+# octopusdeploy_git_trigger (Resource)
+
+This resource manages Git repository 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.
+
+### Optional
+
+- `is_disabled` (Boolean) Disables the trigger from being run when set.
+- `sources` (Block List) List of Git trigger sources. Contains details of the deployment action slug, the git dependency and what file paths to monitor. (see [below for nested schema](#nestedblock--sources))
+- `space_id` (String) The space ID associated with the project to attach the trigger.
+
+### Read-Only
+
+- `id` (String) The ID of this resource.
+
+
+### Nested Schema for `sources`
+
+Required:
+
+- `deployment_action_slug` (String)
+- `exclude_file_paths` (List of String)
+- `git_dependency_name` (String)
+- `include_file_paths` (List of String)
+
+
diff --git a/examples/resources/octopusdeploy_git_trigger/import.sh b/examples/resources/octopusdeploy_git_trigger/import.sh
new file mode 100644
index 00000000..38f0466b
--- /dev/null
+++ b/examples/resources/octopusdeploy_git_trigger/import.sh
@@ -0,0 +1 @@
+terraform import [options] octopusdeploy_git_trigger.
\ No newline at end of file
diff --git a/examples/resources/octopusdeploy_git_trigger/resource.tf b/examples/resources/octopusdeploy_git_trigger/resource.tf
new file mode 100644
index 00000000..bed9184d
--- /dev/null
+++ b/examples/resources/octopusdeploy_git_trigger/resource.tf
@@ -0,0 +1,12 @@
+resource "octopusdeploy_git_trigger" "my_trigger" {
+ name = "My Git trigger"
+ space_id = "Spaces-1"
+ project_id = "Projects-1"
+ channel_id = "Channels-1"
+ sources {
+ deployment_action_slug = "deploy-action-slug"
+ git_dependency_name = ""
+ include_file_paths = ["include/me", "include/me/too"]
+ exclude_file_paths = ["exclude/me", "exclude/me/too"]
+ }
+}
diff --git a/go.mod b/go.mod
index 8dcc68c2..b7605719 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.55.0
+ github.com/OctopusDeploy/go-octopusdeploy/v2 v2.56.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 7a39471c..49d3724d 100644
--- a/go.sum
+++ b/go.sum
@@ -22,6 +22,7 @@ github.com/OctopusDeploy/go-octodiff v1.0.0 h1:U+ORg6azniwwYo+O44giOw6TiD5USk8S4
github.com/OctopusDeploy/go-octodiff v1.0.0/go.mod h1:Mze0+EkOWTgTmi8++fyUc6r0aLZT7qD9gX+31t8MmIU=
github.com/OctopusDeploy/go-octopusdeploy/v2 v2.55.0 h1:kX6qRRy8AgbqTiYdenqVNe69pGhntwJGEgJx9rtn9/8=
github.com/OctopusDeploy/go-octopusdeploy/v2 v2.55.0/go.mod h1:ggvOXzMnq+w0pLg6C9zdjz6YBaHfO3B3tqmmB7JQdaw=
+github.com/OctopusDeploy/go-octopusdeploy/v2 v2.56.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/provider.go b/octopusdeploy/provider.go
index 58a85a55..362bfcb7 100644
--- a/octopusdeploy/provider.go
+++ b/octopusdeploy/provider.go
@@ -46,6 +46,7 @@ func Provider() *schema.Provider {
"octopusdeploy_deployment_process": resourceDeploymentProcess(),
"octopusdeploy_dynamic_worker_pool": resourceDynamicWorkerPool(),
"octopusdeploy_gcp_account": resourceGoogleCloudPlatformAccount(),
+ "octopusdeploy_git_trigger": resourceGitTrigger(),
"octopusdeploy_kubernetes_agent_deployment_target": resourceKubernetesAgentDeploymentTarget(),
"octopusdeploy_kubernetes_agent_worker": resourceKubernetesAgentWorker(),
"octopusdeploy_kubernetes_cluster_deployment_target": resourceKubernetesClusterDeploymentTarget(),
diff --git a/octopusdeploy/resource_git_trigger.go b/octopusdeploy/resource_git_trigger.go
new file mode 100644
index 00000000..6aa51ce7
--- /dev/null
+++ b/octopusdeploy/resource_git_trigger.go
@@ -0,0 +1,130 @@
+package octopusdeploy
+
+import (
+ "context"
+ "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/projects"
+ "log"
+
+ "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/actions"
+ "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/client"
+ "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/filters"
+ "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/triggers"
+ "github.com/hashicorp/terraform-plugin-sdk/v2/diag"
+ "github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema"
+)
+
+func resourceGitTrigger() *schema.Resource {
+ return &schema.Resource{
+ CreateContext: resourceGitTriggerCreate,
+ DeleteContext: resourceGitTriggerDelete,
+ Description: "This resource manages Git repository triggers in Octopus Deploy.",
+ Importer: getImporter(),
+ ReadContext: resourceGitTriggerRead,
+ Schema: getGitTriggerSchema(),
+ UpdateContext: resourceGitTriggerUpdate,
+ }
+}
+
+func buildGitTriggerResource(d *schema.ResourceData, client *client.Client) (*triggers.ProjectTrigger, error) {
+ name := d.Get("name").(string)
+ spaceId := d.Get("space_id").(string)
+ projectId := d.Get("project_id").(string)
+ channelId := d.Get("channel_id").(string)
+
+ isDisabled := false
+ if v, ok := d.GetOk("is_disabled"); ok {
+ isDisabled = v.(bool)
+ }
+
+ flattenedGitTriggerSources := d.Get("sources")
+ gitTriggerSources := expandGitTriggerSources(flattenedGitTriggerSources)
+
+ action := actions.NewCreateReleaseAction(channelId)
+ filter := filters.NewGitTriggerFilter(gitTriggerSources)
+
+ project, err := projects.GetByID(client, spaceId, projectId)
+ if err != nil {
+ return nil, err
+ }
+
+ createReleaseTrigger := triggers.NewProjectTrigger(name, "", isDisabled, project, action, filter)
+
+ return createReleaseTrigger, nil
+}
+
+func resourceGitTriggerCreate(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics {
+ client := m.(*client.Client)
+
+ projectTrigger, err := buildGitTriggerResource(d, client)
+ if err != nil {
+ return diag.FromErr(err)
+ }
+
+ resource, err := client.ProjectTriggers.Add(projectTrigger)
+ if err != nil {
+ return diag.FromErr(err)
+ }
+
+ if isEmpty(resource.GetID()) {
+ log.Println("ID is nil")
+ } else {
+ d.SetId(resource.GetID())
+ }
+
+ return nil
+}
+
+func resourceGitTriggerRead(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics {
+ id := d.Id()
+
+ client := m.(*client.Client)
+ projectTrigger, err := client.ProjectTriggers.GetByID(id)
+ if err != nil {
+ return diag.FromErr(err)
+ }
+ if projectTrigger == nil {
+ d.SetId("")
+ return nil
+ }
+
+ action := projectTrigger.Action.(*actions.CreateReleaseAction)
+ filter := projectTrigger.Filter.(*filters.GitTriggerFilter)
+
+ d.Set("name", projectTrigger.Name)
+ d.Set("space_id", projectTrigger.SpaceID)
+ d.Set("project_id", projectTrigger.ProjectID)
+ d.Set("is_disabled", projectTrigger.IsDisabled)
+ d.Set("channel_id", action.ChannelID)
+ d.Set("sources", flattenGitTriggerSources(filter.Sources))
+
+ return nil
+}
+
+func resourceGitTriggerUpdate(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics {
+ client := m.(*client.Client)
+ projectTrigger, err := buildGitTriggerResource(d, client)
+ if err != nil {
+ return diag.FromErr(err)
+ }
+ projectTrigger.ID = d.Id() // set ID so Octopus API knows which project trigger to update
+
+ resource, err := client.ProjectTriggers.Update(projectTrigger)
+ if err != nil {
+ return diag.FromErr(err)
+ }
+
+ d.SetId(resource.GetID())
+
+ return nil
+}
+
+func resourceGitTriggerDelete(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics {
+ client := m.(*client.Client)
+ err := client.ProjectTriggers.DeleteByID(d.Id())
+ if err != nil {
+ return diag.FromErr(err)
+ }
+
+ d.SetId("")
+ return nil
+}
diff --git a/octopusdeploy/schema_git_trigger.go b/octopusdeploy/schema_git_trigger.go
new file mode 100644
index 00000000..2b83dbd7
--- /dev/null
+++ b/octopusdeploy/schema_git_trigger.go
@@ -0,0 +1,42 @@
+package octopusdeploy
+
+import (
+ "github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema"
+ "github.com/hashicorp/terraform-plugin-sdk/v2/helper/validation"
+)
+
+func getGitTriggerSchema() map[string]*schema.Schema {
+ return map[string]*schema.Schema{
+ "name": getNameSchema(true),
+ "space_id": {
+ Optional: true,
+ Description: "The space ID associated with the project to attach the trigger.",
+ Type: schema.TypeString,
+ ValidateDiagFunc: validation.ToDiagFunc(validation.StringIsNotWhiteSpace),
+ },
+ "project_id": {
+ Description: "The ID of the project to attach the trigger.",
+ Required: true,
+ Type: schema.TypeString,
+ ValidateDiagFunc: validation.ToDiagFunc(validation.StringIsNotWhiteSpace),
+ },
+ "channel_id": {
+ Description: "The ID of the channel in which the release will be created if the action type is CreateRelease.",
+ Required: true,
+ Type: schema.TypeString,
+ ValidateDiagFunc: validation.ToDiagFunc(validation.StringIsNotWhiteSpace),
+ },
+ "sources": {
+ Description: "List of Git trigger sources. Contains details of the deployment action slug, the git dependency and what file paths to monitor.",
+ Optional: true,
+ Type: schema.TypeList,
+ Elem: &schema.Resource{Schema: getGitTriggerSourceSchema()},
+ },
+ "is_disabled": {
+ Description: "Disables the trigger from being run when set.",
+ Optional: true,
+ Default: false,
+ Type: schema.TypeBool,
+ },
+ }
+}
diff --git a/octopusdeploy/schema_git_trigger_source_package.go b/octopusdeploy/schema_git_trigger_source_package.go
new file mode 100644
index 00000000..60a9fe5a
--- /dev/null
+++ b/octopusdeploy/schema_git_trigger_source_package.go
@@ -0,0 +1,66 @@
+package octopusdeploy
+
+import (
+ "github.com/OctopusDeploy/go-octopusdeploy/v2/pkg/filters"
+ "github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema"
+)
+
+func expandGitTriggerSources(values interface{}) []filters.GitTriggerSource {
+ if values == nil {
+ return nil
+ }
+
+ var gitTriggerSources []filters.GitTriggerSource
+ for _, v := range values.([]interface{}) {
+ flattenedMap := v.(map[string]interface{})
+ gitTriggerSources = append(gitTriggerSources, filters.GitTriggerSource{
+ DeploymentActionSlug: flattenedMap["deployment_action_slug"].(string),
+ GitDependencyName: flattenedMap["git_dependency_name"].(string),
+ IncludeFilePaths: flattenedMap["include_file_paths"].([]string),
+ ExcludeFilePaths: flattenedMap["exclude_file_paths"].([]string),
+ })
+ }
+
+ return gitTriggerSources
+}
+
+func flattenGitTriggerSources(gitTriggerSources []filters.GitTriggerSource) []interface{} {
+ if len(gitTriggerSources) == 0 {
+ return nil
+ }
+
+ flattenedGitTriggerSources := []interface{}{}
+ for _, v := range gitTriggerSources {
+ flattenedGitTriggerSources = append(flattenedGitTriggerSources, map[string]interface{}{
+ "deployment_action_slug": v.DeploymentActionSlug,
+ "git_dependency_name": v.GitDependencyName,
+ "include_file_paths": v.IncludeFilePaths,
+ "exclude_file_paths": v.ExcludeFilePaths,
+ })
+ }
+
+ return flattenedGitTriggerSources
+}
+
+func getGitTriggerSourceSchema() map[string]*schema.Schema {
+ return map[string]*schema.Schema{
+ "deployment_action_slug": {
+ Required: true,
+ Type: schema.TypeString,
+ },
+ "git_dependency_name": {
+ Required: true,
+ Type: schema.TypeString,
+ },
+ "include_file_paths": {
+ Required: true,
+ Type: schema.TypeList,
+ Elem: &schema.Schema{Type: schema.TypeString},
+ },
+ "exclude_file_paths": {
+ Required: true,
+ Type: schema.TypeList,
+ Elem: &schema.Schema{Type: schema.TypeString},
+ },
+ }
+}