diff --git a/mso/resource_mso_template.go b/mso/resource_mso_template.go index 7cc95995..a50f01ec 100644 --- a/mso/resource_mso_template.go +++ b/mso/resource_mso_template.go @@ -180,7 +180,7 @@ func (ndoTemplate *ndoTemplate) validateConfig() []error { } duplicates := duplicatesInList(ndoTemplate.sites) if len(duplicates) > 0 { - duplicatesErrors := []error{fmt.Errorf("duplication found in the sites list")} + duplicatesErrors := []error{fmt.Errorf("Duplication found in the sites list")} for _, site := range duplicates { duplicatesErrors = append(duplicatesErrors, fmt.Errorf(fmt.Sprintf("Site %s is duplicated", site))) } diff --git a/mso/resource_mso_template_test.go b/mso/resource_mso_template_test.go new file mode 100644 index 00000000..3604bffd --- /dev/null +++ b/mso/resource_mso_template_test.go @@ -0,0 +1,436 @@ +package mso + +import ( + "fmt" + "regexp" + "strconv" + "testing" + + "github.com/hashicorp/terraform-plugin-sdk/helper/resource" + "github.com/hashicorp/terraform-plugin-sdk/terraform" +) + +func TestAccMsoTemplateTenant(t *testing.T) { + + resource.Test(t, resource.TestCase{ + PreCheck: func() { testAccPreCheck(t) }, + Providers: testAccProviders, + Steps: []resource.TestStep{ + { + Config: testAccCheckMsoTemplateTenantDuplicateSitesConfig(), + ExpectError: regexp.MustCompile(`Duplication found in the sites list`), + }, + { + Config: testAccCheckMsoTemplateTenanNoTenantConfig(), + ExpectError: regexp.MustCompile(`Tenant is required for template of type tenant.`), + }, + { + Config: testAccCheckMsoTemplateTenantConfig(), + Check: resource.ComposeTestCheckFunc( + testAccCheckMSOTemplateState( + "mso_template.template_tenant", + &TemplateTest{ + TemplateName: "test_template_tenant", + TemplateType: "tenant", + Tenant: "ansible_test", + Sites: []string{}, + }, + ), + ), + }, + { + Config: testAccCheckMsoTemplateTenanSiteAnsibleTestConfig(), + Check: resource.ComposeTestCheckFunc( + testAccCheckMSOTemplateState( + "mso_template.template_tenant", + &TemplateTest{ + TemplateName: "test_template_tenant", + TemplateType: "tenant", + Tenant: "ansible_test", + Sites: []string{"ansible_test"}, + }, + ), + ), + }, + { + Config: testAccCheckMsoTemplateTenanTwoSitesConfig(), + Check: resource.ComposeTestCheckFunc( + testAccCheckMSOTemplateState( + "mso_template.template_tenant", + &TemplateTest{ + TemplateName: "test_template_tenant", + TemplateType: "tenant", + Tenant: "ansible_test", + Sites: []string{"ansible_test", "ansible_test_2"}, + }, + ), + ), + }, + { + Config: testAccCheckMsoTemplateTenanTwoSitesReversedConfig(), + Check: resource.ComposeTestCheckFunc( + testAccCheckMSOTemplateState( + "mso_template.template_tenant", + &TemplateTest{ + TemplateName: "test_template_tenant", + TemplateType: "tenant", + Tenant: "ansible_test", + Sites: []string{"ansible_test", "ansible_test_2"}, + }, + ), + ), + }, + { + Config: testAccCheckMsoTemplateTenanSiteAnsibleTest2Config(), + Check: resource.ComposeTestCheckFunc( + testAccCheckMSOTemplateState( + "mso_template.template_tenant", + &TemplateTest{ + TemplateName: "test_template_tenant", + TemplateType: "tenant", + Tenant: "ansible_test", + Sites: []string{"ansible_test_2"}, + }, + ), + ), + }, + { + Config: testAccCheckMsoTemplateTenantNoSitesConfig(), + Check: resource.ComposeTestCheckFunc( + testAccCheckMSOTemplateState( + "mso_template.template_tenant", + &TemplateTest{ + TemplateName: "test_template_tenant", + TemplateType: "tenant", + Tenant: "ansible_test", + Sites: []string{}, + }, + ), + ), + }, + { + Config: testAccCheckMsoTemplateTenantNameChangeConfig(), + Check: resource.ComposeTestCheckFunc( + testAccCheckMSOTemplateState( + "mso_template.template_tenant", + &TemplateTest{ + TemplateName: "test_template_tenant_changed", + TemplateType: "tenant", + Tenant: "ansible_test", + Sites: []string{}, + }, + ), + ), + }, + }, + }) +} + +type TemplateTest struct { + TemplateName string `json:",omitempty"` + TemplateType string `json:",omitempty"` + Tenant string `json:",omitempty"` + Sites []string `json:",omitempty"` +} + +func testAccCheckMSOTemplateState(resourceName string, stateTemplate *TemplateTest) resource.TestCheckFunc { + return func(s *terraform.State) error { + + rootModule, err := s.RootModule().Resources[resourceName] + if !err { + return fmt.Errorf("%v", err) + } + + if rootModule.Primary.ID == "" { + return fmt.Errorf("No ID is set for the template") + } + + if rootModule.Primary.Attributes["tenant_id"] == "" && stateTemplate.Tenant != "" { + return fmt.Errorf("No tenant ID is set for the template") + } else if stateTemplate.Tenant != "" { + tenantState, err := s.RootModule().Resources[fmt.Sprintf("data.mso_tenant.%s", stateTemplate.Tenant)] + if !err { + return fmt.Errorf("Tenant %s not found in state", stateTemplate.Tenant) + } + if tenantState.Primary.Attributes["display_name"] != stateTemplate.Tenant { + return fmt.Errorf("Tenant display name does not match, expected: %s, got: %s", stateTemplate.Tenant, tenantState.Primary.Attributes["display_name"]) + } + } + + if rootModule.Primary.Attributes["template_name"] != stateTemplate.TemplateName { + return fmt.Errorf("Template name does not match, expected: %s, got: %s", stateTemplate.TemplateName, rootModule.Primary.Attributes["template_name"]) + } + + if rootModule.Primary.Attributes["template_type"] != stateTemplate.TemplateType { + return fmt.Errorf("Template type does not match, expected: %s, got: %s", stateTemplate.TemplateType, rootModule.Primary.Attributes["template_type"]) + } + + if sites, ok := rootModule.Primary.Attributes["sites.#"]; ok { + if siteAmount, e := strconv.Atoi(sites); e != nil { + return fmt.Errorf("Could not convert sites amount to integer") + } else if siteAmount != len(stateTemplate.Sites) { + return fmt.Errorf("Amount of sites do not match, expected: %d, got: %d", len(stateTemplate.Sites), len(rootModule.Primary.Attributes["sites.#"])) + } + + for _, site := range stateTemplate.Sites { + siteState, err := s.RootModule().Resources[fmt.Sprintf("data.mso_site.%s", site)] + if !err { + return fmt.Errorf("Site %s not found in state", site) + } + if siteState.Primary.Attributes["name"] != site { + return fmt.Errorf("Site display name does not match, expected: %s, got: %s", site, siteState.Primary.Attributes["display_name"]) + } + } + } else { + if len(stateTemplate.Sites) != 0 { + return fmt.Errorf("Amount of sites do not match, expected: %d, got: 0", len(stateTemplate.Sites)) + } + } + + return nil + } +} + +func testSiteConfigAnsibleTest() string { + return fmt.Sprintf(` + data "mso_site" "ansible_test" { + name = "ansible_test" + } + `) +} + +func testSiteConfigAnsibleTest2() string { + return fmt.Sprintf(` + data "mso_site" "ansible_test_2" { + name = "ansible_test_2" + } + `) +} + +func testTenantConfigAnsibleTest() string { + return fmt.Sprintf(` + data "mso_tenant" "ansible_test" { + name = "ansible_test" + } + `) +} + +func testAccCheckMsoTemplateTenantConfig() string { + return fmt.Sprintf(`%s + resource "mso_template" "template_tenant" { + template_name = "test_template_tenant" + template_type = "tenant" + tenant_id = data.mso_tenant.ansible_test.id + } + `, testTenantConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateTenantNameChangeConfig() string { + return fmt.Sprintf(`%s + resource "mso_template" "template_tenant" { + template_name = "test_template_tenant_changed" + template_type = "tenant" + tenant_id = data.mso_tenant.ansible_test.id + } + `, testTenantConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateTenantNoSitesConfig() string { + return fmt.Sprintf(`%s + resource "mso_template" "template_tenant" { + template_name = "test_template_tenant" + template_type = "tenant" + tenant_id = data.mso_tenant.ansible_test.id + sites = [] + } + `, testTenantConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateTenanSiteAnsibleTestConfig() string { + return fmt.Sprintf(`%s%s + resource "mso_template" "template_tenant" { + template_name = "test_template_tenant" + template_type = "tenant" + tenant_id = data.mso_tenant.ansible_test.id + sites = [data.mso_site.ansible_test.id] + } + `, testSiteConfigAnsibleTest(), testTenantConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateTenanSiteAnsibleTest2Config() string { + return fmt.Sprintf(`%s%s + resource "mso_template" "template_tenant" { + template_name = "test_template_tenant" + template_type = "tenant" + tenant_id = data.mso_tenant.ansible_test.id + sites = [data.mso_site.ansible_test_2.id] + } + `, testSiteConfigAnsibleTest2(), testTenantConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateTenanTwoSitesConfig() string { + return fmt.Sprintf(`%s%s%s + resource "mso_template" "template_tenant" { + template_name = "test_template_tenant" + template_type = "tenant" + tenant_id = data.mso_tenant.ansible_test.id + sites = [data.mso_site.ansible_test.id, data.mso_site.ansible_test_2.id] + } + `, testSiteConfigAnsibleTest(), testSiteConfigAnsibleTest2(), testTenantConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateTenanTwoSitesReversedConfig() string { + return fmt.Sprintf(`%s%s%s + resource "mso_template" "template_tenant" { + template_name = "test_template_tenant" + template_type = "tenant" + tenant_id = data.mso_tenant.ansible_test.id + sites = [data.mso_site.ansible_test_2.id, data.mso_site.ansible_test.id] + } + `, testSiteConfigAnsibleTest(), testSiteConfigAnsibleTest2(), testTenantConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateTenantDuplicateSitesConfig() string { + return fmt.Sprintf(`%s%s + resource "mso_template" "template_tenant" { + template_name = "test_template_tenant" + template_type = "tenant" + tenant_id = data.mso_tenant.ansible_test.id + sites = [data.mso_site.ansible_test.id, data.mso_site.ansible_test.id] + } + `, testSiteConfigAnsibleTest(), testTenantConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateTenanNoTenantConfig() string { + return fmt.Sprintf(`%s%s + resource "mso_template" "template_tenant" { + template_name = "test_template_tenant" + template_type = "tenant" + sites = [data.mso_site.ansible_test.id, data.mso_site.ansible_test_2.id] + } + `, testSiteConfigAnsibleTest(), testSiteConfigAnsibleTest2()) +} + +func TestAccMsoTemplateL3out(t *testing.T) { + + resource.Test(t, resource.TestCase{ + PreCheck: func() { testAccPreCheck(t) }, + Providers: testAccProviders, + Steps: []resource.TestStep{ + { + Config: testAccCheckMsoTemplateL3outErrorNoTenantConfig(), + ExpectError: regexp.MustCompile(`Tenant is required for template of type l3out.`), + }, + { + Config: testAccCheckMsoTemplateL3outErrorNoSitesConfig(), + ExpectError: regexp.MustCompile(`Site is required for template of type l3out.`), + }, + { + Config: testAccCheckMsoTemplateL3outErrorTwositesConfig(), + ExpectError: regexp.MustCompile(`Only one site is allowed for template of type l3out.`), + }, + { + Config: testAccCheckMsoTemplateL3outConfig(), + Check: resource.ComposeTestCheckFunc( + testAccCheckMSOTemplateState( + "mso_template.template_l3out", + &TemplateTest{ + TemplateName: "test_template_l3out", + TemplateType: "l3out", + Tenant: "ansible_test", + Sites: []string{"ansible_test"}, + }, + ), + ), + }, + }, + }) +} + +func testAccCheckMsoTemplateL3outConfig() string { + return fmt.Sprintf(`%s%s + resource "mso_template" "template_l3out" { + template_name = "test_template_l3out" + template_type = "l3out" + tenant_id = data.mso_tenant.ansible_test.id + sites = [data.mso_site.ansible_test.id] + } + `, testTenantConfigAnsibleTest(), testSiteConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateL3outErrorNoTenantConfig() string { + return fmt.Sprintf(`%s + resource "mso_template" "template_l3out" { + template_name = "test_template_l3out" + template_type = "l3out" + sites = [data.mso_site.ansible_test.id] + } + `, testSiteConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateL3outErrorNoSitesConfig() string { + return fmt.Sprintf(`%s%s + resource "mso_template" "template_l3out" { + template_name = "test_template_l3out" + template_type = "l3out" + tenant_id = data.mso_tenant.ansible_test.id + } + `, testTenantConfigAnsibleTest(), testSiteConfigAnsibleTest()) +} + +func testAccCheckMsoTemplateL3outErrorTwositesConfig() string { + return fmt.Sprintf(`%s%s%s + resource "mso_template" "template_l3out" { + template_name = "test_template_l3out" + template_type = "l3out" + tenant_id = data.mso_tenant.ansible_test.id + sites = [data.mso_site.ansible_test.id, data.mso_site.ansible_test_2.id] + } + `, testTenantConfigAnsibleTest(), testSiteConfigAnsibleTest(), testSiteConfigAnsibleTest2()) +} + +func TestAccMsoTemplateFabricPolicy(t *testing.T) { + + resource.Test(t, resource.TestCase{ + PreCheck: func() { testAccPreCheck(t) }, + Providers: testAccProviders, + Steps: []resource.TestStep{ + { + Config: testAccCheckMsoTemplateFabricPolicyErrorTenantConfig(), + ExpectError: regexp.MustCompile(`Tenant cannot be attached to template of type fabric_policy.`), + }, + { + Config: testAccCheckMsoTemplateFabricPolicyConfig(), + Check: resource.ComposeTestCheckFunc( + testAccCheckMSOTemplateState( + "mso_template.template_fabric_policy", + &TemplateTest{ + TemplateName: "test_template_fabric_policy", + TemplateType: "fabric_policy", + Sites: []string{}, + }, + ), + ), + }, + }, + }) +} + +func testAccCheckMsoTemplateFabricPolicyConfig() string { + return fmt.Sprintf(` + resource "mso_template" "template_fabric_policy" { + template_name = "test_template_fabric_policy" + template_type = "fabric_policy" + } + `) +} + +func testAccCheckMsoTemplateFabricPolicyErrorTenantConfig() string { + return fmt.Sprintf(`%s + resource "mso_template" "template_fabric_policy" { + template_name = "test_template_fabric_policy" + template_type = "fabric_policy" + tenant_id = data.mso_tenant.ansible_test.id + } + `, testTenantConfigAnsibleTest()) +}