Learn how to use multiple conditions effectively within the `aws_iam_policy_document` data block in Terraform to craft precise and secure IAM policies.
When working with Terraform's aws_iam_policy_document data source, defining multiple conditions within a single statement for an IAM policy requires a structured approach. AWS IAM policies use a JSON-based format where conditions are specified within a "Condition" block. Each "Condition" block, even with multiple key-value pairs, evaluates as a single condition. To illustrate, let's consider an example where we want to permit an action based on the request's IP range and the use of HTTPS.  We'll then discuss the limitations of this approach and explore alternative methods for handling more complex scenarios involving "OR" logic or conditions across multiple keys.
To define multiple conditions within a single statement in an IAM policy document using Terraform's aws_iam_policy_document data source, you need to understand how AWS structures these conditions.
AWS IAM uses a JSON-based policy language. Within a policy statement, the "Condition" block allows you to specify criteria for when the policy should take effect. To achieve the effect of multiple conditions, you would nest conditions within a single "Condition" block using operators like "StringEquals" or "Bool". Each key within "StringEquals" or "Bool" represents a single condition.
Let's illustrate with an example. Suppose you want to allow an action only if the request comes from a specific IP range ("1.2.3.4/32") and the request uses HTTPS:
data "aws_iam_policy_document" "example" {
  statement {
    actions = [ "s3:GetObject" ]
    resources = [ "arn:aws:s3:::examplebucket/*" ]
    condition {
      test = "StringEquals"
      values = ["https"]
      variable = "aws:SecureTransport"
    }
    condition {
      test = "IpAddress"
      values = ["1.2.3.4/32"]
      variable = "aws:SourceIp"
    }
  }
}In this example, we have two separate "condition" blocks within the "statement" block. This signifies that both conditions must be met for the statement to be true and the action to be allowed. The first condition checks if the request is using HTTPS, and the second condition checks if the source IP address of the request falls within the specified range.
It's important to note that while you can have multiple "condition" blocks within a "statement", each "condition" block itself evaluates as a single condition.
Directly expressing "OR" logic between different condition keys within a single "condition" block is not supported by the aws_iam_policy_document data source in Terraform.
If you need to implement complex logic with "OR" conditions or conditions spanning multiple keys, you might need to explore alternative approaches:
Multiple Statements: Create separate statements within your policy document, each handling a different combination of conditions. This can lead to more verbose policies but provides flexibility.
Custom Policy Documents: Instead of using aws_iam_policy_document, define the entire policy document as a JSON string within your Terraform code. This gives you full control over the policy structure but requires careful manual management.
Remember to consult the AWS IAM documentation for detailed information on policy syntax, available condition keys, and best practices.
The code defines two examples of AWS IAM policies using Terraform. The first example demonstrates achieving "OR" logic by using multiple statements within the policy document, allowing access if either HTTPS is used or the request comes from a specific IP address. The second example showcases defining a custom policy document as a JSON string for more complex logic, checking for both HTTPS and whether the source IP address falls within specified ranges.
This example demonstrates how to achieve "OR" logic by using multiple statements within the policy document.
data "aws_iam_policy_document" "example" {
  statement {
    actions = [ "s3:GetObject" ]
    resources = [ "arn:aws:s3:::examplebucket/*" ]
    condition {
      test = "StringEquals"
      values = ["https"]
      variable = "aws:SecureTransport"
    }
  }
  statement {
    actions = [ "s3:GetObject" ]
    resources = [ "arn:aws:s3:::examplebucket/*" ]
    condition {
      test = "IpAddress"
      values = ["1.2.3.4/32"]
      variable = "aws:SourceIp"
    }
  }
}
resource "aws_iam_policy" "example" {
  name   = "example"
  policy = data.aws_iam_policy_document.example.json
}In this example, we have two separate statements. The first statement allows access only if the request uses HTTPS. The second statement allows access only if the request comes from the specified IP address. Since these are separate statements, the effect is an "OR" - the action will be allowed if either statement is true.
This example demonstrates how to define a custom policy document as a JSON string to achieve more complex logic.
resource "aws_iam_policy" "example" {
  name = "example"
  policy = <<EOF
{
  "Version": "2012-10-17",
  "Statement": [
    {
      "Effect": "Allow",
      "Action": [
        "s3:GetObject"
      ],
      "Resource": [
        "arn:aws:s3:::examplebucket/*"
      ],
      "Condition": {
        "Bool": {
          "aws:SecureTransport": "true"
        },
        "IpAddress": {
          "aws:SourceIp": [
            "1.2.3.4/32",
            "192.168.1.0/24"
          ]
        }
      }
    }
  ]
}
EOF
}In this example, we define the entire policy document as a JSON string. This allows us to express more complex conditions. Here, we are checking for both HTTPS and whether the source IP address falls within one of two specified ranges.
Remember to replace the example values with your actual requirements and to consult the AWS IAM documentation for a comprehensive understanding of policy syntax and available options.
aws_iam_policy_document:
aws:SourceIp, aws:SecureTransport, etc.)terraform validate and AWS's IAM policy simulator to catch errors early in your development process.This article explains how to define multiple conditions within a single statement in an IAM policy document using Terraform's aws_iam_policy_document data source.
Key Points:
Example:
The provided example demonstrates allowing an action only if the request comes from a specific IP range and uses HTTPS. This is achieved by using two "condition" blocks within the "statement" block.
Important Notes:
In conclusion, managing multiple conditions within AWS IAM policies using Terraform's aws_iam_policy_document data source requires a clear understanding of how conditions are structured and evaluated. While this data source is useful for generating basic policy structures, it has limitations when it comes to expressing complex logic like "OR" conditions between different keys within a single "condition" block. For such scenarios, consider using multiple statements within your policy document or defining the entire policy as a custom JSON string. Remember to prioritize clarity and security by adhering to IAM best practices, leveraging the official AWS IAM documentation, and thoroughly testing your policies. By understanding these concepts and choosing the right approach for your specific needs, you can effectively manage complex access control requirements in your AWS infrastructure.
 Syntax for conditions in aws_iam_policy resources in terraform ... | May 25, 2021 ... ... correct use of the IAM policy data source. ... What is the correct syntax for multiple conditions in a Terraform
 Syntax for conditions in aws_iam_policy resources in terraform ... | May 25, 2021 ... ... correct use of the IAM policy data source. ... What is the correct syntax for multiple conditions in a Terraform aws_iam_policy_document data ... Aws_iam_policy_document - Condition with multiple tests - AWS ... | Does anyone know if its possible to use aws_iam_policy_document data source to create a statement condition with multiple tests so you can create OR logic?  Looks like this is supported by the AWS documentation: https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_multi-value-conditions.html
 Aws_iam_policy_document - Condition with multiple tests - AWS ... | Does anyone know if its possible to use aws_iam_policy_document data source to create a statement condition with multiple tests so you can create OR logic?  Looks like this is supported by the AWS documentation: https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_multi-value-conditions.html
 Terraform tips & tricks: loops, if-statements, and gotchas | by ... | Update, November 17, 2016: We took this blog post series, expanded it, and turned it into a book called Terraform: Up & Running!
 Terraform tips & tricks: loops, if-statements, and gotchas | by ... | Update, November 17, 2016: We took this blog post series, expanded it, and turned it into a book called Terraform: Up & Running! Terraform AWS IAM Error — “MalformedPolicyDocument: Policy ... | I’m still a bit lost when it comes to understanding the nitty-gritty of IAM and I was a little stumped when I encountered this error…
 Terraform AWS IAM Error — “MalformedPolicyDocument: Policy ... | I’m still a bit lost when it comes to understanding the nitty-gritty of IAM and I was a little stumped when I encountered this error…