Skip to content

Latest commit

 

History

History
 
 

cloudwatchlogs-emf

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 
 
 

FireLens Example: Sending Custom EMF Metrics to CloudWatch

CloudWatch supports the Embedded Metrics Format which allows you to create json events sent with PutLogEvents that are processed and become CloudWatch Metrics. The PutLogEvents call must include an HTTP header noting that all the log events are EMF; Fluent Bit can be configured to send this header. This tutorial will show you how to collect and send custom EMF metrics with Fluent Bit. Fluent Bit can be used a replacement for the CloudWatch Agent to upload EMF.

You can instrument your application to produce EMF metrics by manually constructing the JSON or by using one of the AWS EMF libraries (search GitHub for the most up to date list of available libraries):

Once your application is instrumented to produce EMF metrics, you then need to ingest them into Fluent Bit using an input plugin. There are two commonly options for ingesting EMF, which are outlined in the tutorials below.

Once the EMF records are ingested into Fluent Bit, you simply need to send them to a cloudwatch output that has log_format json/emf set its options. This tells Fluent Bit to set the EMF HTTP header when it calls PutLogEvents. Please note that since all records will be interpreted as EMF, the logs sent by the output with the log_format json/emf option configured should probably be all EMF events. This is not a hard requirement, however, the best practice is to split EMF events and other events to separate output instances matching different tags and sending to different log streams.

Both the older golang CloudWatch plugin and the newer high performance plugin support this option.

Tutorial 1: Send EMF to Fluent Bit over local TCP connection

This is the recommended option if you are using one of the AWS EMF libraries.

See the emf-over-tcp folder for example Task Definitions and Fluent Bit configuration.

First, we create a Fluent Bit TCP input and cloudwatch output for the EMF:

# TCP input used for EMF payloads
[INPUT]
    Name        tcp
    Listen      0.0.0.0
    Port        25888
    Chunk_Size  32
    Buffer_Size 64
    Format      none
    Tag         emf-${HOSTNAME}
    # This tag is used by the output plugin to determine the LogStream
    # including the HOSTNAME is a way to increase the number of LogStreams.
    # The maximum throughput on a
    # single LogStream is 5 MB/s (max 1 MB at max 5 TPS).
    # In AWSVPC mode, the HOSTNAME is the ENI private IP
    # in bridge mode, the HOSTNAME is the Docker container ID

# Output for EMF over TCP -> CloudWatch
[OUTPUT]
    Name                cloudwatch
    Match               emf-*
    region              us-west-2
    log_key             log
    log_group_name      aws-emf-ecs-firelens-example-metrics
    log_stream_prefix   from-fluent-bit-
    auto_create_group   true
    log_format          json/emf

Fluent Bit will accept EMF over TCP port 25888. Build this config file into a custom Fluent Bit image as shown in the base config-file-type example.

Now we configure the EMF libraries in your app to send EMF to TCP port 25888. The setup steps here are similar to using the CW Agent to collect EMF in ECS. Go to the section for your chosen network mode.

If you are using Bridge Network Mode

If your network mode is bridge, you can add the following env var to your app container definition to tell the library where to emit EMF. This env var uses a Docker hostname inserted by the link to the firelens container:

"name": "app",
"links": [ "fluentbit" ], // the link is based on the container name
"environment": [{
    "name": "AWS_EMF_AGENT_ENDPOINT",
    "value": "tcp://fluentbit:25888"
}]

The value tcp://fluentbit:25888 comes from:

  1. fluentbit hostname is set from the docker link.
  2. Port 25888 is the port the Fluent Bit TCP port is configured to listen on.

And then setup a port mapping on the FireLens container:

"name": "fluentbit",
"portMappings": [{
    "protocol": "tcp",
    "containerPort": 25888
}],

See the full task definition in the emf-over-tcp directory. The setup steps here are again similar to using the CW Agent to collect EMF in ECS.

Note on FLUENT_HOST hostname

The above linked steps and the bridge network mode example task definition in this tutorial use a Docker link to create a hostname in the app container for your Fluent Bit/FireLens container. It should be noted that FireLens will set a hostname in app containers for Fluent Bit called FLUENT_HOST. The value of FLUENT_HOST is just the IP address of the Fluent Bit container. However, the AWS EMF libraries require an environment variable to reach the collection agent:

"environment": [
	{
        "name": "AWS_EMF_AGENT_ENDPOINT",
        "value": "tcp://fluentbit:25888"
    }
],

It is not possible to reference the value of the FLUENT_HOST environment variable in the value of this environment variable. However, if you are constructing EMF directly in your application code and sending it over TCP instead of using the AWS EMF libraries, using FLUENT_HOST to get the IP of Fluent Bit may be a desirable option.

If you are using AWSVPC or Host Network Mode

If you are using host or AWSVPC network mode, the FireLens container can be reached on localhost. Note that for host mode, the host ENI is used, and so Fluent Bit must be the only process on the host using this port.

In this case, you simply set an env var on the app container for the EMF libraries:

"environment": [{
                "name": "AWS_EMF_AGENT_ENDPOINT",
                "value": "tcp://127.0.0.1:25888"
              }],

Tutorial 2: Send EMF to Fluent Bit via stdout

If your app's EMF library is configured to send EMF json events to stdout, then you can simply use the FireLens log driver options to send it:

	"logConfiguration": {
		"logDriver":"awsfirelens",
		"options": {
			"Name": "cloudwatch_logs",
			"region": "us-west-2",
			"log_key": "log",
			"log_group_name": "emf-log-group/application",
			"auto_create_group": "true",
			"log_stream_prefix": "emf-",
			"log_format": "json/emf",
			"retry_limit": "2"
		}
	},

However, note that if you choose this option, all events emitted to stdout will be sent with the json/emf header to a single log stream.

Other Options for Ingesting EMF

In theory, you could use almost any Fluent Bit input plugin to ingest EMF. In practice, the only other commonly used option besides those mentioned above is the tail input- your application could write its EMF json events to a file.