From 4f90ecf76cd52f9695564dcd6d3db7a6c1337b11 Mon Sep 17 00:00:00 2001 From: Chris Taylor Date: Mon, 30 Oct 2023 18:37:24 -0400 Subject: [PATCH] More gracevully reject colonCompoundIDs that aren't compound IDs Avoids an array out of bounds (panic) resulting from colon compound resource IDs that have only one component (no colon). --- ...tion_actions_action_service_association.go | 12 +++++- ...actions_action_service_association_test.go | 12 +++++- ...omation_actions_action_team_association.go | 12 +++++- ...on_actions_action_team_association_test.go | 12 +++++- ...omation_actions_runner_team_association.go | 12 +++++- ...on_actions_runner_team_association_test.go | 12 +++++- ...gerduty_event_orchestration_integration.go | 5 ++- .../resource_pagerduty_team_membership.go | 11 ++++- pagerduty/util.go | 9 +++- pagerduty/util_test.go | 41 +++++++++++++++++++ 10 files changed, 121 insertions(+), 17 deletions(-) create mode 100644 pagerduty/util_test.go diff --git a/pagerduty/resource_pagerduty_automation_actions_action_service_association.go b/pagerduty/resource_pagerduty_automation_actions_action_service_association.go index ace0fd8ac..52d09ee78 100644 --- a/pagerduty/resource_pagerduty_automation_actions_action_service_association.go +++ b/pagerduty/resource_pagerduty_automation_actions_action_service_association.go @@ -71,7 +71,11 @@ func fetchPagerDutyAutomationActionsActionServiceAssociation(d *schema.ResourceD return err } - actionID, serviceID := resourcePagerDutyParseColonCompoundID(d.Id()) + actionID, serviceID, err := resourcePagerDutyParseColonCompoundID(d.Id()) + if err != nil { + return err + } + return resource.Retry(30*time.Second, func() *resource.RetryError { resp, _, err := client.AutomationActionsAction.GetAssociationToService(actionID, serviceID) if err != nil { @@ -111,7 +115,11 @@ func resourcePagerDutyAutomationActionsActionServiceAssociationDelete(d *schema. return err } - actionID, serviceID := resourcePagerDutyParseColonCompoundID(d.Id()) + actionID, serviceID, err := resourcePagerDutyParseColonCompoundID(d.Id()) + if err != nil { + return err + } + log.Printf("[INFO] Deleting PagerDuty AutomationActionsActionServiceAssociation %s", d.Id()) retryErr := resource.Retry(2*time.Minute, func() *resource.RetryError { diff --git a/pagerduty/resource_pagerduty_automation_actions_action_service_association_test.go b/pagerduty/resource_pagerduty_automation_actions_action_service_association_test.go index 37e1d1e3d..255e02016 100644 --- a/pagerduty/resource_pagerduty_automation_actions_action_service_association_test.go +++ b/pagerduty/resource_pagerduty_automation_actions_action_service_association_test.go @@ -50,7 +50,11 @@ func testAccCheckPagerDutyAutomationActionsActionServiceAssociationDestroy(s *te if r.Type != "pagerduty_automation_actions_action_service_association" { continue } - actionID, serviceID := resourcePagerDutyParseColonCompoundID(r.Primary.ID) + actionID, serviceID, err := resourcePagerDutyParseColonCompoundID(r.Primary.ID) + if err != nil { + return err + } + if _, _, err := client.AutomationActionsAction.GetAssociationToService(actionID, serviceID); err == nil { return fmt.Errorf("Automation Actions Runner still exists") } @@ -69,7 +73,11 @@ func testAccCheckPagerDutyAutomationActionsActionServiceAssociationExists(n stri } client, _ := testAccProvider.Meta().(*Config).Client() - actionID, serviceID := resourcePagerDutyParseColonCompoundID(rs.Primary.ID) + actionID, serviceID, err := resourcePagerDutyParseColonCompoundID(rs.Primary.ID) + if err != nil { + return err + } + found, _, err := client.AutomationActionsAction.GetAssociationToService(actionID, serviceID) if err != nil { return err diff --git a/pagerduty/resource_pagerduty_automation_actions_action_team_association.go b/pagerduty/resource_pagerduty_automation_actions_action_team_association.go index e363a5970..96a620eef 100644 --- a/pagerduty/resource_pagerduty_automation_actions_action_team_association.go +++ b/pagerduty/resource_pagerduty_automation_actions_action_team_association.go @@ -75,7 +75,11 @@ func fetchPagerDutyAutomationActionsActionTeamAssociation(d *schema.ResourceData return err } - actionID, teamID := resourcePagerDutyParseColonCompoundID(d.Id()) + actionID, teamID, err := resourcePagerDutyParseColonCompoundID(d.Id()) + if err != nil { + return err + } + return resource.Retry(30*time.Second, func() *resource.RetryError { resp, _, err := client.AutomationActionsAction.GetAssociationToTeam(actionID, teamID) if err != nil { @@ -115,7 +119,11 @@ func resourcePagerDutyAutomationActionsActionTeamAssociationDelete(d *schema.Res return err } - actionID, teamID := resourcePagerDutyParseColonCompoundID(d.Id()) + actionID, teamID, err := resourcePagerDutyParseColonCompoundID(d.Id()) + if err != nil { + return err + } + log.Printf("[INFO] Deleting PagerDuty AutomationActionsActionTeamAssociation %s", d.Id()) retryErr := resource.Retry(2*time.Minute, func() *resource.RetryError { diff --git a/pagerduty/resource_pagerduty_automation_actions_action_team_association_test.go b/pagerduty/resource_pagerduty_automation_actions_action_team_association_test.go index f71e601fd..178c351e8 100644 --- a/pagerduty/resource_pagerduty_automation_actions_action_team_association_test.go +++ b/pagerduty/resource_pagerduty_automation_actions_action_team_association_test.go @@ -47,7 +47,11 @@ func testAccCheckPagerDutyAutomationActionsActionTeamAssociationDestroy(s *terra if r.Type != "pagerduty_automation_actions_action_team_association" { continue } - actionID, teamID := resourcePagerDutyParseColonCompoundID(r.Primary.ID) + actionID, teamID, err := resourcePagerDutyParseColonCompoundID(r.Primary.ID) + if err != nil { + return err + } + if _, _, err := client.AutomationActionsAction.GetAssociationToTeam(actionID, teamID); err == nil { return fmt.Errorf("Automation Actions Runner still exists") } @@ -66,7 +70,11 @@ func testAccCheckPagerDutyAutomationActionsActionTeamAssociationExists(n string) } client, _ := testAccProvider.Meta().(*Config).Client() - actionID, teamID := resourcePagerDutyParseColonCompoundID(rs.Primary.ID) + actionID, teamID, err := resourcePagerDutyParseColonCompoundID(rs.Primary.ID) + if err != nil { + return err + } + found, _, err := client.AutomationActionsAction.GetAssociationToTeam(actionID, teamID) if err != nil { return err diff --git a/pagerduty/resource_pagerduty_automation_actions_runner_team_association.go b/pagerduty/resource_pagerduty_automation_actions_runner_team_association.go index 636c3cb05..9ff8557b7 100644 --- a/pagerduty/resource_pagerduty_automation_actions_runner_team_association.go +++ b/pagerduty/resource_pagerduty_automation_actions_runner_team_association.go @@ -71,7 +71,11 @@ func fetchPagerDutyAutomationActionsRunnerTeamAssociation(d *schema.ResourceData return err } - runnerID, teamID := resourcePagerDutyParseColonCompoundID(d.Id()) + runnerID, teamID, err := resourcePagerDutyParseColonCompoundID(d.Id()) + if err != nil { + return err + } + return resource.Retry(30*time.Second, func() *resource.RetryError { resp, _, err := client.AutomationActionsRunner.GetAssociationToTeam(runnerID, teamID) if err != nil { @@ -111,7 +115,11 @@ func resourcePagerDutyAutomationActionsRunnerTeamAssociationDelete(d *schema.Res return err } - runnerID, teamID := resourcePagerDutyParseColonCompoundID(d.Id()) + runnerID, teamID, err := resourcePagerDutyParseColonCompoundID(d.Id()) + if err != nil { + return err + } + log.Printf("[INFO] Deleting PagerDuty AutomationActionsRunnerTeamAssociation %s", d.Id()) retryErr := resource.Retry(2*time.Minute, func() *resource.RetryError { diff --git a/pagerduty/resource_pagerduty_automation_actions_runner_team_association_test.go b/pagerduty/resource_pagerduty_automation_actions_runner_team_association_test.go index 42041c93f..e2498c951 100644 --- a/pagerduty/resource_pagerduty_automation_actions_runner_team_association_test.go +++ b/pagerduty/resource_pagerduty_automation_actions_runner_team_association_test.go @@ -47,7 +47,11 @@ func testAccCheckPagerDutyAutomationActionsRunnerTeamAssociationDestroy(s *terra if r.Type != "pagerduty_automation_actions_runner_team_association" { continue } - runnerID, teamID := resourcePagerDutyParseColonCompoundID(r.Primary.ID) + runnerID, teamID, err := resourcePagerDutyParseColonCompoundID(r.Primary.ID) + if err != nil { + return err + } + if _, _, err := client.AutomationActionsRunner.GetAssociationToTeam(runnerID, teamID); err == nil { return fmt.Errorf("Automation Actions Runner Team association still exists") } @@ -66,7 +70,11 @@ func testAccCheckPagerDutyAutomationActionsRunnerTeamAssociationExists(n string) } client, _ := testAccProvider.Meta().(*Config).Client() - runnerID, teamID := resourcePagerDutyParseColonCompoundID(rs.Primary.ID) + runnerID, teamID, err := resourcePagerDutyParseColonCompoundID(rs.Primary.ID) + if err != nil { + return err + } + found, _, err := client.AutomationActionsRunner.GetAssociationToTeam(runnerID, teamID) if err != nil { return err diff --git a/pagerduty/resource_pagerduty_event_orchestration_integration.go b/pagerduty/resource_pagerduty_event_orchestration_integration.go index aa321b167..b8d0b14f3 100644 --- a/pagerduty/resource_pagerduty_event_orchestration_integration.go +++ b/pagerduty/resource_pagerduty_event_orchestration_integration.go @@ -281,7 +281,10 @@ func resourcePagerDutyEventOrchestrationIntegrationDelete(ctx context.Context, d } func resourcePagerDutyEventOrchestrationIntegrationImport(ctx context.Context, d *schema.ResourceData, meta interface{}) ([]*schema.ResourceData, error) { - oid, id := resourcePagerDutyParseColonCompoundID(d.Id()) + oid, id, err := resourcePagerDutyParseColonCompoundID(d.Id()) + if err != nil { + return []*schema.ResourceData{}, err + } if oid == "" || id == "" { return []*schema.ResourceData{}, fmt.Errorf("Error importing pagerduty_event_orchestration_integration. Expected import ID format: :") diff --git a/pagerduty/resource_pagerduty_team_membership.go b/pagerduty/resource_pagerduty_team_membership.go index febf070e0..9aef090dd 100644 --- a/pagerduty/resource_pagerduty_team_membership.go +++ b/pagerduty/resource_pagerduty_team_membership.go @@ -82,7 +82,11 @@ func fetchPagerDutyTeamMembership(d *schema.ResourceData, meta interface{}, errC return err } - userID, teamID := resourcePagerDutyParseColonCompoundID(d.Id()) + userID, teamID, err := resourcePagerDutyParseColonCompoundID(d.Id()) + if err != nil { + return err + } + log.Printf("[DEBUG] Reading user: %s from team: %s", userID, teamID) return resource.Retry(2*time.Minute, func() *resource.RetryError { resp, _, err := client.Teams.GetMembers(teamID, &pagerduty.GetMembersOptions{}) @@ -191,7 +195,10 @@ func resourcePagerDutyTeamMembershipDelete(d *schema.ResourceData, meta interfac return err } - userID, teamID := resourcePagerDutyParseColonCompoundID(d.Id()) + userID, teamID, err := resourcePagerDutyParseColonCompoundID(d.Id()) + if err != nil { + return err + } log.Printf("[DEBUG] Removing user: %s from team: %s", userID, teamID) diff --git a/pagerduty/util.go b/pagerduty/util.go index f0c307b89..4668ec01f 100644 --- a/pagerduty/util.go +++ b/pagerduty/util.go @@ -262,7 +262,12 @@ func unique(s []string) []string { return result } -func resourcePagerDutyParseColonCompoundID(id string) (string, string) { +func resourcePagerDutyParseColonCompoundID(id string) (string, string, error) { parts := strings.Split(id, ":") - return parts[0], parts[1] + + if len(parts) < 2 { + return "", "", fmt.Errorf("%s: expected colon compound ID to have at least two components", id) + } + + return parts[0], parts[1], nil } diff --git a/pagerduty/util_test.go b/pagerduty/util_test.go new file mode 100644 index 000000000..4d9cfa866 --- /dev/null +++ b/pagerduty/util_test.go @@ -0,0 +1,41 @@ +package pagerduty + +import ( + "strings" + "testing" +) + +func TestResourcePagerDutyParseColonCompoundID(t *testing.T) { + resourceIDComponents := []string{"PABC12A", "PABC12B"} + validCompoundResourceID := strings.Join(resourceIDComponents, ":") + + first, second, err := resourcePagerDutyParseColonCompoundID(validCompoundResourceID) + + if err != nil { + t.Fatalf("%s: expected success while parsing invalid compound resource id: got %s", validCompoundResourceID, err) + } + + for i, component := range []string{first, second} { + expectedResourceIDComponent := resourceIDComponents[i] + + if expectedResourceIDComponent != component { + t.Errorf( + "%s: expected component %d of a valid compound resource ID to be %s: got %s", + validCompoundResourceID, + i+1, + expectedResourceIDComponent, + component, + ) + } + } +} + +func TestResourcePagerDutyParseColonCompoundIDFailsForInvalidCompoundIDs(t *testing.T) { + invalidCompoundResourceID := "PABC12APABC12B" + + _, _, err := resourcePagerDutyParseColonCompoundID(invalidCompoundResourceID) + + if err == nil { + t.Fatalf("%s: expected errors while parsing invalid compound resource id: got success", invalidCompoundResourceID) + } +}