Post

Terraform Static Analysis with tfsec

Security based static analysis for Terraform using tfsec

Terraform Static Analysis with tfsec

We use Terraform for all our deployment automation needs. Thanks to it’s fantastic extensibility, if there isn’t a provider available to do what we need, it’s very easy to create one.

Terraform, for those who haven’t used it before, lets you declaratively specify the resources that you want to deploy and then maintains a state of what has and hasn’t been deployed. For deployed resources, it tracks the characteristics or attributes, and updates accordingly with updates.

As an example, say we wanted to create a new S3 bucket in our AWS account we might define something like;

1
2
3
4
5
6
7
8
9
resource "aws_s3_bucket" "my-bucket-for-secret-things" {
  bucket = "my-bucket-for-secret-things"
  acl    = "public-read"

  tags = {
    Name        = "Sensitive Storage Bucket"
    Environment = "Production"
  }
}

Now, you might see straight away that there is a potential issue here - the bucket has an ACL allowing public-read. Checking the AWS Documentation on canned responses, shows us that public read has the following permissions;

1
Owner gets FULL_CONTROL. The AllUsers group (see Who Is a Grantee?) gets READ access.

We can see this easily with a single resource in our file, but imagine this was just one resource among thousands in a project; this could very easily go unnoticed and end up in production with a wide open bucket.

Enter TFSEC

tfsec is a static analysis tool for Terraform created by liamg. It will tell you if there are security issues in your Terraform. Installation is simple;

Mac

tfsec is available for install with homebrew

1
brew install tfsec

Windows

tfsec is available for install with Chocolatey

1
choco install tfsec

Linux/everything else

Alternatively, you can install using go get

1
go get -u github.com/tfsec/tfsec/cmd/tfsec

We can use tfsec to test our Terraform by simply running the command against the folder with our terraform in it.

1
2
3
.
|_ tf/
    |_ main.tf

Assuming that the above resource for the S3 bucket is in main.tf, we can run

1
tfsec tf

This will give me a list of problems categorised into severities.

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
3 potential problems detected:

Problem 1

  [AWS001][WARNING] Resource 'aws_s3_bucket.my-bucket-for-secret-things' has an ACL which allows public read access.
  /tmp/tfsec/main.tf:5

       2 | 
       3 | resource "aws_s3_bucket" "my-bucket-for-secret-things" {
       4 |   bucket = "my-bucket-for-secret-things"
       5 |   acl    = "public-read"
       6 | 
       7 |   tags = {
       8 |     Name        = "Sensitive Storage Bucket"

  See https://github.com/tfsec/tfsec/wiki/AWS001 for more information.

Problem 2

  [AWS002][ERROR] Resource 'aws_s3_bucket.my-bucket-for-secret-things' does not have logging enabled.
  /tmp/tfsec/main.tf:3-11

       1 | 
       2 | 
       3 | resource "aws_s3_bucket" "my-bucket-for-secret-things" {
       4 |   bucket = "my-bucket-for-secret-things"
       5 |   acl    = "public-read"
       6 | 
       7 |   tags = {
       8 |     Name        = "Sensitive Storage Bucket"
       9 |     Environment = "Production"
      10 |   }
      11 | }
      12 | 

  See https://github.com/tfsec/tfsec/wiki/AWS002 for more information.

Problem 3

  [AWS017][ERROR] Resource 'aws_s3_bucket.my-bucket-for-secret-things' defines an un-encrypted S3 bucket (missing server_side_encryption_configuration block).
  /tmp/tfsec/main.tf:3-11

       1 | 
       2 | 
       3 | resource "aws_s3_bucket" "my-bucket-for-secret-things" {
       4 |   bucket = "my-bucket-for-secret-things"
       5 |   acl    = "public-read"
       6 | 
       7 |   tags = {
       8 |     Name        = "Sensitive Storage Bucket"
       9 |     Environment = "Production"
      10 |   }
      11 | }
      12 | 

  See https://github.com/tfsec/tfsec/wiki/AWS017 for more information.

We can see that it has identified that we have a public acl specified, this is a warning to draw our attention to the fact and allow us to make a judgement call.

The other two errors tell us that we don’t have “at rest” encryption and we could do with access logging. All in all very useful since I’d forgotten them.

But there is more

One of the benefits of tfsec is the ability for it to process default variable values and report on them as well. For example if we had a variable for the acl.

1
2
3
4
variable "bucket-acl" {
    description = "The ACL for S3 buckets"
    default     = "public-read"
}

then we update the main.tf file to use this variable rather than the hard coded value;

1
2
3
4
5
6
7
8
9
resource "aws_s3_bucket" "my-bucket-for-secret-things" {
  bucket = "my-bucket-for-secret-things"
  acl    = var.bucket-acl

  tags = {
    Name        = "Sensitive Storage Bucket"
    Environment = "Production"
  }
}

Our folder structure now looks more like this;

1
2
3
4
.
|_ tf/
    |_ main.tf
    |_ variables.tf

Now when we run tfsec against the tf folder it gives us a slightly different error;

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Problem 1

  [AWS001][] Resource 'aws_s3_bucket.my-bucket-for-secret-things' has an ACL which allows public read access.
  /tmp/tfsec/main.tf:5

       2 | 
       3 | resource "aws_s3_bucket" "my-bucket-for-secret-things" {
       4 |   bucket = "my-bucket-for-secret-things"
       5 |   acl    = var.bucket-acl    [string] "public-read"
       6 | 
       7 |   tags = {
       8 |     Name        = "Sensitive Storage Bucket"

  See https://github.com/tfsec/tfsec/wiki/AWS001 for more information.

The default value as been evaluated from the variables.tf file and seen that without explicitly overriding the value would have the same net effect as hard coding it to public-read

False Positive

Assuming that you did intend to ignore this error for this resource, that is fine, it can be done by adding an ignore declaration to the resource;

1
2
3
4
5
6
7
8
9
10
11
resource "aws_s3_bucket" "my-bucket-for-secret-things" {
  bucket = "my-bucket-for-secret-things"
  #tfsec:ignore:AWS001
  acl    = var.bucket-acl

  tags = {
    Name        = "Sensitive Storage Bucket"
    Environment = "Production"
  }
}

More information

Check out the github project for more information

A note on Terraform

If it sounds like something you want to learn more about then check out Terraform: From Beginner to Master by Kevin Holditch

This post is licensed under CC BY 4.0 by the author.