From bf862202b7767f4cb2decc48964ec8dd714b1b32 Mon Sep 17 00:00:00 2001 From: Ryan Oaks Date: Tue, 23 Mar 2021 10:20:20 -0400 Subject: [PATCH] HCPE-1000 - Update contribution docs to include guidance on acceptance tests (#79) * Update contribution docs to include guidance on acceptance tests * Add line to highlight that the acceptance test documentation is meant to augment the general testing docs * Clarify that the Terraform testing documentation should be read before our acceptance testing docs --- contributing/README.md | 2 +- contributing/checklist-resource.md | 1 + contributing/writing-tests.md | 201 +++++++++++++++++++++++++++++ 3 files changed, 203 insertions(+), 1 deletion(-) create mode 100644 contributing/writing-tests.md diff --git a/contributing/README.md b/contributing/README.md index 9f6dd1b92..e19f6c1e9 100644 --- a/contributing/README.md +++ b/contributing/README.md @@ -30,7 +30,7 @@ terraform { In order to run the full suite of Acceptance tests, run `make testacc`. -*Note:* Acceptance tests create real resources, and often cost money to run. +*Note:* Acceptance tests create real resources, and often cost money to run. Please read [Writing Acceptance Tests](writing-tests.md) in the contribution guidelines for more information on usage. ```sh $ make testacc diff --git a/contributing/checklist-resource.md b/contributing/checklist-resource.md index cdc951974..bacb3b5b2 100644 --- a/contributing/checklist-resource.md +++ b/contributing/checklist-resource.md @@ -3,6 +3,7 @@ Implementing a new resource is a good way to learn more about how Terraform interacts with upstream APIs. There are plenty of examples to draw from in the existing resources, but you still get to implement something completely new. - [ ] __Minimal LOC__: It can be inefficient for both the reviewer and author to go through long feedback cycles on a big PR with many resources. We therefore encourage you to only submit **1 resource at a time**. +- [ ] __Acceptance Tests__: New resources should include acceptance tests covering their behavior. See [Writing Acceptance Tests](writing-tests.md) below for a detailed guide on how to approach these. - [ ] __Documentation__: Each resource gets a page in the [Terraform Registry documentation](https://registry.terraform.io/providers/hashicorp/hcp/latest/docs). For a new resource, you'll want to add an example and field descriptions. A guide is required if the new feature requires multiple dependent resources to use. - [ ] __Well-formed Code__: Do your best to follow existing conventions you see in the codebase, and ensure your code is formatted with `go fmt`. The PR reviewers can help out on this front, and may provide comments with suggestions on how to improve the code. diff --git a/contributing/writing-tests.md b/contributing/writing-tests.md new file mode 100644 index 000000000..3c88d768d --- /dev/null +++ b/contributing/writing-tests.md @@ -0,0 +1,201 @@ +# Writing Acceptance Tests + +Terraform includes an acceptance test harness that does most of the repetitive +work involved in testing a resource. Please read the [Extending Terraform documentation](https://www.terraform.io/docs/extend/testing/index.html) +first for a general understanding of how this test harness works. The guide +below is meant to augment that documentation with information specific to the +HCP provider. + +## Acceptance Tests Often Cost Money to Run + +Because acceptance tests create real resources, they often cost money to run. +Because the resources only exist for a short period of time, the total amount +of money required is usually a relatively small. Nevertheless, we don't want +financial limitations to be a barrier to contribution, so if you are unable to +pay to run acceptance tests for your contribution, mention this in your +pull request. We will happily accept "best effort" implementations of +acceptance tests and run them for you on our side. This might mean that your PR +takes a bit longer to merge, but it most definitely is not a blocker for +contributions. + +## Running an Acceptance Test + +Acceptance tests can be run using the `testacc` target in the +`GNUmakefile`. The individual tests to run can be controlled using a regular +expression. Prior to running the tests, provider configuration details such as +access keys must be made available as environment variables: + +```sh +export HCP_CLIENT_ID=... +export HCP_CLIENT_SECRET=... +``` + +Tests can then be run by specifying a regular expression defining the tests to run: + +```sh +$ make testacc TESTARGS='-run=TestAccConsulCluster' +TF_ACC=1 go test ./... -v -run=TestAccConsulCluster -timeout 120m +ok github.com/hashicorp/terraform-provider-hcp/internal/consul (cached) [no tests to run] +=== RUN TestAccConsulCluster +--- PASS: TestAccConsulCluster (538.48s) +PASS +ok github.com/hashicorp/terraform-provider-hcp/internal/provider 539.112s +``` + +Entire resource test suites can be targeted by using the naming convention to +write the regular expression. + +For advanced developers, the acceptance testing framework accepts some additional environment variables that can be used to control Terraform CLI binary selection, logging, and other behaviors. See the [Extending Terraform documentation](https://www.terraform.io/docs/extend/testing/acceptance-tests/index.html#environment-variables) for more information. + +## Writing an Acceptance Test + +Terraform has a framework for writing acceptance tests which minimises the +amount of boilerplate code necessary to use common testing patterns. The entry +point to the framework is the `resource.Test()` function. + +Tests are divided into `TestStep`s. Each `TestStep` proceeds by applying some +Terraform configuration using the provider under test, and then verifying that +results are as expected by making assertions using the provider API. It is +common for a single test function to exercise both the creation of and updates +to a single resource. Most tests follow a similar structure. + +1. Pre-flight checks are made to ensure that sufficient provider configuration + is available to be able to proceed - in the case of HCP, `HCP_CLIENT_ID` and `HCP_CLIENT_SECRET` must be set prior + to running acceptance tests. This is common to all tests exercising a single + provider. + +Each `TestStep` is defined in the call to `resource.Test()`. Most assertion +functions are defined out of band with the tests. This keeps the tests +readable, and allows reuse of assertion functions across different tests of the +same type of resource. The definition of a complete test looks like this: + +```go +func TestAccConsulCluster(t *testing.T) { + resourceName := "hcp_consul_cluster.test" + + resource.Test(t, resource.TestCase{ + PreCheck: func() { testAccPreCheck(t) }, + ProviderFactories: providerFactories, + CheckDestroy: testAccCheckConsulClusterDestroy, + Steps: []resource.TestStep{ + { + Config: testConfig(testAccConsulClusterConfig), + Check: resource.ComposeTestCheckFunc( + testAccCheckConsulClusterExists(resourceName), + resource.TestCheckResourceAttr(resourceName, "cluster_id", "test-consul-cluster"), + ), + }, + }, + }) +} +``` + +When executing the test, the following steps are taken for each `TestStep`: + +1. The Terraform configuration required for the test is applied. This is + responsible for configuring the resource under test, and any dependencies it + may have. For example, to test the `hcp_consul_cluster` resource, an + `hcp_hvn` is required. This results in configuration which + looks like this: + + ```hcl + resource "hcp_hvn" "test" { + hvn_id = "test-hvn" + cloud_provider = "aws" + region = "us-west-2" + } + + resource "hcp_consul_cluster" "test" { + cluster_id = "test-consul-cluster" + hvn_id = hcp_hvn.test.hvn_id + tier = "development" + } + ``` + +1. Assertions are run using the provider API. These use the provider API + directly rather than asserting against the resource state. For example, to + verify that the `hcp_consul_cluster` described above was created + successfully, a test function like this is used: + + ```go + func testAccCheckConsulClusterExists(name string) resource.TestCheckFunc { + return func(s *terraform.State) error { + rs, ok := s.RootModule().Resources[name] + if !ok { + return fmt.Errorf("not found: %s", name) + } + + id := rs.Primary.ID + if id == "" { + return fmt.Errorf("no ID is set") + } + + client := testAccProvider.Meta().(*clients.Client) + + link, err := buildLinkFromURL(id, ConsulClusterResourceType, client.Config.OrganizationID) + if err != nil { + return fmt.Errorf("unable to build link for %q: %v", id, err) + } + + clusterID := link.ID + loc := link.Location + + if _, err := clients.GetConsulClusterByID(context.Background(), client, loc, clusterID); err != nil { + return fmt.Errorf("unable to read Consul cluster %q: %v", id, err) + } + + return nil + } + } + ``` + + Notice that the only information used from the Terraform state is the ID of + the resource - though in this case it is necessary to split the ID into + constituent parts in order to use the provider API. For computed properties, + we instead assert that the value saved in the Terraform state was the + expected value if possible. The testing framework provides helper functions + for several common types of check - for example: + + ```go + resource.TestCheckResourceAttr(resourceName, "cluster_id", "test-consul-cluster"), + ``` + +1. The resources created by the test are destroyed. This step happens + automatically, and is the equivalent of calling `terraform destroy`. + +1. Assertions are made against the provider API to verify that the resources + have indeed been removed. If these checks fail, the test fails and reports + "dangling resources". The code to ensure that the `hcp_consul_cluster` shown + above is removed looks like this: + + ```go + func testAccCheckConsulClusterDestroy(s *terraform.State) error { + client := testAccProvider.Meta().(*clients.Client) + + for _, rs := range s.RootModule().Resources { + switch rs.Type { + case "hcp_consul_cluster": + id := rs.Primary.ID + + link, err := buildLinkFromURL(id, ConsulClusterResourceType, client.Config.OrganizationID) + if err != nil { + return fmt.Errorf("unable to build link for %q: %v", id, err) + } + + clusterID := link.ID + loc := link.Location + + _, err = clients.GetConsulClusterByID(context.Background(), client, loc, clusterID) + if err == nil || !clients.IsResponseCodeNotFound(err) { + return fmt.Errorf("didn't get a 404 when reading destroyed Consul cluster %q: %v", id, err) + } + + default: + continue + } + } + return nil + } + ``` + + These functions usually test only for the resource directly under test.