Search code examples
dockerterraformsystemdcoreos

Terraform: Passing JSON file as environment variable value with a systemd unit file inside docker container


I am trying to pass json in an environmental variable of a systemd unit file with terraform. I am using an external provider named CT to generate ignition from the YAML configuration.

CT Config:

data "ct_config" "ignition" {
  # Reference: https://github.com/poseidon/terraform-provider-ct/
  content      = data.template_file.bastion_user_data.rendered
  strict       = true
  pretty_print = false
}

Error:

Error: error converting to Ignition: error at line 61, column 17
invalid unit content: unexpected newline encountered while parsing option name

  on ../../modules/example/launch-template.tf line 91, in data "ct_config" "ignition":
  91: data "ct_config" "ignition" {

Unit File Content:

- name: cw-agent.service
  enabled: true
  contents: |
    [Unit]
    Description=Cloudwatch Agent Service
    Documentation=https://docs.aws.amazon.com/AmazonCloudWatch/latest/monitoring/CloudWatch-Agent-Configuration-File-Details.html
    Requires=docker.socket
    After=docker.socket
    [Service]
    TimeoutStartSec=0
    Restart=always
    RestartSec=10s
    Environment=CONFIG=${cw-agent-config}
    ExecStartPre=-/usr/bin/docker kill cloudwatch-agent
    ExecStartPre=-/usr/bin/docker rm cloudwatch-agent
    ExecStartPre=/usr/bin/docker pull amazon/cloudwatch-agent
    ExecStart=/usr/bin/docker run --name cloudwatch-agent \
                              --net host \
                              --env CW_CONFIG_CONTENT=$CONFIG \
                              amazon/cloudwatch-agent
    ExecStop=/usr/bin/docker stop cloudwatch-agent
    [Install]
    WantedBy=multi-user.target

Rendering:

data "template_file" "cw_agent_config" {
  template = file("${path.module}/../cw-agent-config.json")
}

cw-agent-config = indent(10, data.template_file.cw_agent_config.rendered)

File Content:

{
    "agent": {
        "metrics_collection_interval": 60,
"run_as_user": "cwagent"
    },
    "metrics": {
        "append_dimensions": {
            "AutoScalingGroupName": "$${aws:AutoScalingGroupName}",
            "ImageId": "$${aws:ImageId}",
            "InstanceId": "$${aws:InstanceId}",
            "InstanceType": "$${aws:InstanceType}"
        },
        "metrics_collected": {
            "disk": {
                "drop_device" : true,
                "measurement": [
                    "used_percent"
                ],
                "metrics_collection_interval": 120,
                "resources": [
                    "/"
                ]
            },
            "mem": {
                "measurement": [
                    "mem_used_percent"
                ],
                "metrics_collection_interval": 120
            }
        }
    }
}

I need this json file to be available as a value of environment variable named CW_CONFIG_CONTENT inside a docker container.


Solution

  • This was solved by using the Terraform jsonencode function.

        cw-agent-config = jsonencode(data.template_file.cw_agent_config.rendered)