forked from gruntwork-io/terratest
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathterraform_packer_example_test.go
181 lines (143 loc) · 7.08 KB
/
terraform_packer_example_test.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
package test
import (
"crypto/tls"
"fmt"
"testing"
"time"
"github.com/gruntwork-io/terratest/modules/aws"
http_helper "github.com/gruntwork-io/terratest/modules/http-helper"
"github.com/gruntwork-io/terratest/modules/logger"
"github.com/gruntwork-io/terratest/modules/packer"
"github.com/gruntwork-io/terratest/modules/random"
"github.com/gruntwork-io/terratest/modules/terraform"
test_structure "github.com/gruntwork-io/terratest/modules/test-structure"
)
// This is a complicated, end-to-end integration test. It builds the AMI from examples/packer-docker-example,
// deploys it using the Terraform code on examples/terraform-packer-example, and checks that the web server in the AMI
// response to requests. The test is broken into "stages" so you can skip stages by setting environment variables (e.g.,
// skip stage "build_ami" by setting the environment variable "SKIP_build_ami=true"), which speeds up iteration when
// running this test over and over again locally.
func TestTerraformPackerExample(t *testing.T) {
t.Parallel()
// The folder where we have our Terraform code
workingDir := "../examples/terraform-packer-example"
// At the end of the test, delete the AMI
defer test_structure.RunTestStage(t, "cleanup_ami", func() {
awsRegion := test_structure.LoadString(t, workingDir, "awsRegion")
deleteAMI(t, awsRegion, workingDir)
})
// At the end of the test, undeploy the web app using Terraform
defer test_structure.RunTestStage(t, "cleanup_terraform", func() {
undeployUsingTerraform(t, workingDir)
})
// At the end of the test, fetch the most recent syslog entries from each Instance. This can be useful for
// debugging issues without having to manually SSH to the server.
defer test_structure.RunTestStage(t, "logs", func() {
awsRegion := test_structure.LoadString(t, workingDir, "awsRegion")
fetchSyslogForInstance(t, awsRegion, workingDir)
})
// Build the AMI for the web app
test_structure.RunTestStage(t, "build_ami", func() {
// Pick a random AWS region to test in. This helps ensure your code works in all regions.
awsRegion := aws.GetRandomStableRegion(t, nil, nil)
test_structure.SaveString(t, workingDir, "awsRegion", awsRegion)
buildAMI(t, awsRegion, workingDir)
})
// Deploy the web app using Terraform
test_structure.RunTestStage(t, "deploy_terraform", func() {
awsRegion := test_structure.LoadString(t, workingDir, "awsRegion")
deployUsingTerraform(t, awsRegion, workingDir)
})
// Validate that the web app deployed and is responding to HTTP requests
test_structure.RunTestStage(t, "validate", func() {
validateInstanceRunningWebServer(t, workingDir)
})
}
// Build the AMI in packer-docker-example
func buildAMI(t *testing.T, awsRegion string, workingDir string) {
packerOptions := &packer.Options{
// The path to where the Packer template is located
Template: "../examples/packer-docker-example/build.json",
// Only build the AMI
Only: "ubuntu-ami",
// Variables to pass to our Packer build using -var options
Vars: map[string]string{
"aws_region": awsRegion,
},
// Configure retries for intermittent errors
RetryableErrors: DefaultRetryablePackerErrors,
TimeBetweenRetries: DefaultTimeBetweenPackerRetries,
MaxRetries: DefaultMaxPackerRetries,
}
// Save the Packer Options so future test stages can use them
test_structure.SavePackerOptions(t, workingDir, packerOptions)
// Build the AMI
amiID := packer.BuildAmi(t, packerOptions)
// Save the AMI ID so future test stages can use them
test_structure.SaveAmiId(t, workingDir, amiID)
}
// Delete the AMI
func deleteAMI(t *testing.T, awsRegion string, workingDir string) {
// Load the AMI ID and Packer Options saved by the earlier build_ami stage
amiID := test_structure.LoadAmiId(t, workingDir)
aws.DeleteAmi(t, awsRegion, amiID)
}
// Deploy the terraform-packer-example using Terraform
func deployUsingTerraform(t *testing.T, awsRegion string, workingDir string) {
// A unique ID we can use to namespace resources so we don't clash with anything already in the AWS account or
// tests running in parallel
uniqueID := random.UniqueId()
// Give this EC2 Instance and other resources in the Terraform code a name with a unique ID so it doesn't clash
// with anything else in the AWS account.
instanceName := fmt.Sprintf("terratest-http-example-%s", uniqueID)
// Specify the text the EC2 Instance will return when we make HTTP requests to it.
instanceText := fmt.Sprintf("Hello, %s!", uniqueID)
// Load the AMI ID saved by the earlier build_ami stage
amiID := test_structure.LoadAmiId(t, workingDir)
terraformOptions := &terraform.Options{
// The path to where our Terraform code is located
TerraformDir: workingDir,
// Variables to pass to our Terraform code using -var options
Vars: map[string]interface{}{
"aws_region": awsRegion,
"instance_name": instanceName,
"instance_text": instanceText,
"ami_id": amiID,
},
}
// Save the Terraform Options struct, instance name, and instance text so future test stages can use it
test_structure.SaveTerraformOptions(t, workingDir, terraformOptions)
// This will run `terraform init` and `terraform apply` and fail the test if there are any errors
terraform.InitAndApply(t, terraformOptions)
}
// Undeploy the terraform-packer-example using Terraform
func undeployUsingTerraform(t *testing.T, workingDir string) {
// Load the Terraform Options saved by the earlier deploy_terraform stage
terraformOptions := test_structure.LoadTerraformOptions(t, workingDir)
terraform.Destroy(t, terraformOptions)
}
// Fetch the most recent syslogs for the instance. This is a handy way to see what happened on the Instance as part of
// your test log output, without having to re-run the test and manually SSH to the Instance.
func fetchSyslogForInstance(t *testing.T, awsRegion string, workingDir string) {
// Load the Terraform Options saved by the earlier deploy_terraform stage
terraformOptions := test_structure.LoadTerraformOptions(t, workingDir)
instanceID := terraform.OutputRequired(t, terraformOptions, "instance_id")
logs := aws.GetSyslogForInstance(t, instanceID, awsRegion)
logger.Logf(t, "Most recent syslog for Instance %s:\n\n%s\n", instanceID, logs)
}
// Validate the web server has been deployed and is working
func validateInstanceRunningWebServer(t *testing.T, workingDir string) {
// Load the Terraform Options saved by the earlier deploy_terraform stage
terraformOptions := test_structure.LoadTerraformOptions(t, workingDir)
// Run `terraform output` to get the value of an output variable
instanceURL := terraform.Output(t, terraformOptions, "instance_url")
// Setup a TLS configuration to submit with the helper, a blank struct is acceptable
tlsConfig := tls.Config{}
// Figure out what text the instance should return for each request
instanceText, _ := terraformOptions.Vars["instance_text"].(string)
// It can take a minute or so for the Instance to boot up, so retry a few times
maxRetries := 30
timeBetweenRetries := 5 * time.Second
// Verify that we get back a 200 OK with the expected instanceText
http_helper.HttpGetWithRetry(t, instanceURL, &tlsConfig, 200, instanceText, maxRetries, timeBetweenRetries)
}