My AWS Cloudwatch bill is huge. How do I work out which log stream is causing it?
Asked Answered
M

7

54

I got a $1,200 invoice from Amazon for Cloudwatch services last month (specifically for 2 TB of log data ingestion in "AmazonCloudWatch PutLogEvents"), when I was expecting a few tens of dollars. I've logged into the Cloudwatch section of the AWS Console, and can see that one of my log groups used about 2TB of data, but there are thousands of different log streams in that log group, how can I tell which one used that amount of data?

Meader answered 10/4, 2017 at 15:50 Comment(0)
B
86

On the CloudWatch console, use the IncomingBytes metrics to find the amount of data ingested by each log group for a particular time period in uncompressed bytes using Metrics page. Follow the below steps -

  1. Go to CloudWatch metrics page and click on AWS namespace 'Logs' --> 'Log Group Metrics'.
  2. Select the IncomingBytes metrics of the required log groups and click on 'Graphed metrics' tab to see the graph.
  3. Change the start time and end time such that their difference is 30 days and change the period to 30 days. This way, we will get only one data point. Also changed the graph to Number and statistics to Sum.

This way, you will see the amount of data ingested by each log groups and get an idea about which log group is ingesting how much.

enter image description here

You can also achieve the same result using AWS CLI. An example scenario where you just want to know the total amount of data ingested by log groups for say 30 days, you can use get-metric-statistics CLI command-

sample CLI command -

aws cloudwatch get-metric-statistics --metric-name IncomingBytes --start-time 2018-05-01T00:00:00Z --end-time 2018-05-30T23:59:59Z --period 2592000 --namespace AWS/Logs --statistics Sum --region us-east-1

sample output -

{
    "Datapoints": [
        {
            "Timestamp": "2018-05-01T00:00:00Z", 
            "Sum": 1686361672.0, 
            "Unit": "Bytes"
        }
    ], 
    "Label": "IncomingBytes"
}

To find the same for a particular log group, you can change the command to accommodate dimensions like -

aws cloudwatch get-metric-statistics --metric-name IncomingBytes --start-time 2018-05-01T00:00:00Z --end-time 2018-05-30T23:59:59Z --period 2592000 --namespace AWS/Logs --statistics Sum --region us-east-1 --dimensions Name=LogGroupName,Value=test1

One by one, you can run this command on all log groups and check which log group is responsible for most of the bill for data ingested and take corrective measures.

NOTE: Change the parameters specific to your environment and requirement.

The solution provided by OP gives data for the amount of logs stored which is different from logs ingested.

What is the difference?

Data ingested per month is not same as Data storage bytes. After the data is ingested to CloudWatch, it is archived by CloudWatch which includes 26 bytes of metadata per log event and is compressed using gzip level 6 compression. So the Storage bytes refers to the storage space used by Cloudwatch to store the logs after they're ingested.

Reference : https://docs.aws.amazon.com/cli/latest/reference/cloudwatch/get-metric-statistics.html

Bearden answered 24/7, 2018 at 11:49 Comment(3)
If you have a LOT of log groups (we have 2.5k) then this script below will work a lot better https://mcmap.net/q/335037/-my-aws-cloudwatch-bill-is-huge-how-do-i-work-out-which-log-stream-is-causing-itSwahili
Interesting answer, but how does this differentiate between Log Streams within the Log Group, like OP was asking for?Jargon
If you don't see the write numbers as shown in the answer's screenshots (and see mini-graphs instead), just go to the "Options" tab and under "Value" select the option with says "Time range value shows the value from the entire time range."Monochromat
T
18

We had a lambda logging GB of data of due to accidental check-in. Here's a boto3 based python script based on the info from the answers above that scans all log groups and prints out any group with logs greater than 1GB in the past 7 days. This helped me more than trying to use AWS dashboard which was slow to update.

#!/usr/bin/env python3

# Outputs all loggroups with > 1GB of incomingBytes in the past 7 days

import boto3
from datetime import datetime as dt
from datetime import timedelta


logs_client = boto3.client('logs')
cloudwatch_client = boto3.client('cloudwatch')

end_date = dt.today().isoformat(timespec='seconds')
start_date = (dt.today() - timedelta(days=7)).isoformat(timespec='seconds')
print("looking from %s to %s" % (start_date, end_date))

paginator = logs_client.get_paginator('describe_log_groups')
pages = paginator.paginate()
for page in pages:
     for json_data in page['logGroups']:
        log_group_name = json_data.get("logGroupName") 

        cw_response = cloudwatch_client.get_metric_statistics(
           Namespace='AWS/Logs',    
           MetricName='IncomingBytes',
           Dimensions=[
            {
                'Name': 'LogGroupName',
                'Value': log_group_name
            },
            ],
            StartTime= start_date,
            EndTime=end_date,
            Period=3600 * 24 * 7,
            Statistics=[
                'Sum'
            ],
            Unit='Bytes'
        )
        if len(cw_response.get("Datapoints")):
            stats_data = cw_response.get("Datapoints")[0]
            stats_sum = stats_data.get("Sum")   
            sum_GB = stats_sum /  (1000 * 1000 * 1000)
            if sum_GB > 1.0:
                print("%s = %.2f GB" % (log_group_name , sum_GB))
        
Toneytong answered 7/9, 2018 at 13:0 Comment(3)
Awesome, works just like as said. I managed to quickly identify our log group that produced unnecessary amounts of log (5.7TB / week).Tennietenniel
I had to add boto3.setup_default_session(profile_name="default") near the top to get this to work with my credentials -- but this is so far working much better than the AWS CLI call that AWS support recommendsSwahili
This script was immensely helpful. I added some logging etc but overall this did exactly what i needed. We had 2.5k+ log groups, so it took a while to run!Swahili
C
13

You can also click the gear on the gear on the cloudwatch logs dashboard and choose the stored bytes column.

I also clicked anything that said 'never expire' and changed the logs to expire.

Use cloudwatch logs gear and select "Stored Bytes" column

Copenhagen answered 24/3, 2020 at 20:22 Comment(0)
W
12

Although the author of the question and other folks have answered the question in a good way, I will try to have a generic solution that could be applied without knowing the exact log-group-name which is causing too much of logs.

To do this, we can not use describe-log-streams function because this would need --log-group-name and as I said earlier I do not know the value of my log-group-name.

We can use describe-log-groups function because this function does not require any parameter.

Note that I am assuming that you have the required flag (--region) configured in ~/.aws/config file and your EC2 instance have the required permission to execute this command.

aws logs describe-log-groups

This command would list all the log groups in your aws account. The sample output of this would be

 {
    "logGroups": [
        {
            "metricFilterCount": 0,
            "storedBytes": 62299573,
            "arn": "arn:aws:logs:ap-southeast-1:855368385138:log-group:RDSOSMetrics:*",
            "retentionInDays": 30,
            "creationTime": 1566472016743,
            "logGroupName": "/aws/lambda/us-east-1.test"
        }
    ]
}

If you are interested in a specific prefix pattern only for the log group, you can use --log-group-name-prefix like this

aws logs describe-log-groups --log-group-name-prefix /aws/lambda

The output JSON of this command would also be similar to the above output.

If you have too many log groups in your account, analyzing the output of this becomes difficult and we need some command-line utility to give a brief insight into the result. We will use the 'jq' command-line utility to get the desired thing. The intention is to get which log group has produced the most amount of log and hence caused more money.

From the output JSON, the fields which we need for our analysis would be "logGroupName" and "storedBytes". So taking these 2 fields in the 'jq' command.

aws logs describe-log-groups --log-group-name-prefix /aws/ 
| jq -M -r '.logGroups[] | "{\"logGroupName\":\"\(.logGroupName)\",
\"storedBytes\":\(.storedBytes)}"'

Using '\' in the command to do the escape because we want the output to be in the JSON format only to use the sort_by function of jq. The sample output of this would be something like below:

{"logGroupName":"/aws/lambda/test1","storedBytes":3045647212}
{"logGroupName":"/aws/lambda/projectTest","storedBytes":200165401}
{"logGroupName":"/aws/lambda/projectTest2","storedBytes":200}

Note that the output result would not be sorted on storedBytes, so we want to sort them in order to get which log group is the most problematic one.

we will use sort_by function of jq to accomplish this. The sample command would be like this

aws logs describe-log-groups --log-group-name-prefix /aws/ 
| jq -M -r '.logGroups[] | "{\"logGroupName\":\"\(.logGroupName)\",
\"storedBytes\":\(.storedBytes)}"' 
|  jq -s -c 'sort_by(.storedBytes) | .[]'

This would produce the below result for the above sample output

{"logGroupName":"/aws/lambda/projectTest2","storedBytes":200}
{"logGroupName":"/aws/lambda/projectTest","storedBytes":200165401}
{"logGroupName":"/aws/lambda/test1","storedBytes":3045647212}

The elements from the bottom of this list are the ones that have the most log associated with it. You may set the Expire Events After property to a finite period say 1 month to these log group.

If you want to know what is the sum of all the log byte then you can use the 'map' and 'add' function of jq like below.

 aws logs describe-log-groups --log-group-name-prefix /aws/ 
        | jq -M -r '.logGroups[] | "{\"logGroupName\":\"\(.logGroupName)\",
\"storedBytes\":\(.storedBytes)}"'
        |  jq -s -c 'sort_by(.storedBytes) | .[]' 
        | jq -s 'map(.storedBytes) | add '

The output of this command for the above sample output would be

3245812813

The answer has become lengthy but I hope it helps in figuring out the most problematic log group in cloudwatch.

Wilmawilmar answered 28/3, 2020 at 17:36 Comment(0)
M
8

*** UPDATE 20210907 - as @davur points out in one of the comments below, AWS deprecated storedBytes for individual LogStreams, so the method described in this answer no longer fulfills the requirement, although it might be interesting in other ways ***

Okay, I'm answering my own question here, but here we go (with all other answers welcome):

You can use a combination of AWS CLI tools, the csvfix CSV package and a spreadsheet to work this out.

  • Log into the AWS Cloudwatch Console and grab the name of the log group which has generated all the data. In my case it's called "test01-ecs".

  • Unfortunately in the Cloudwatch Console you can't sort the streams by "Stored Bytes" (which would tell you which ones are biggest). If there are too many streams in the log group to look through in the Console then you need to dump them somehow. For this you can use the AWS CLI tool:

    $ aws logs describe-log-streams --log-group-name test01-ecs

  • The command above will give you JSON output (assuming your AWS CLI tool is set to JSON output - set it to output = json in ~/.aws/config if not) and it will look something like this:

    { "logStreams": [ { "creationTime": 1479218045690, "arn": "arn:aws:logs:eu-west-1:902720333704:log-group:test01-ecs:log-stream:test-spec/test-spec/0307d251-7764-459e-a68c-da47c3d9ecd9", "logStreamName": "test-spec/test-spec/0308d251-7764-4d9f-b68d-da47c3e9ebd8", "storedBytes": 7032 } ] }

  • Pipe this output to a JSON file - in my case the file was 31 MB in size:

    $ aws logs describe-log-streams --log-group-name test01-ecs >> ./cloudwatch-output.json

  • Use the in2csv package (part of csvfix) to convert the JSON file to a CSV file which can easily be imported into a spreadsheet, making sure you define the logStreams key to be used to import on:

    $ in2csv cloudwatch-output.json --key logStreams >> ./cloudwatch-output.csv

  • Import the resulting CSV file into a spreadsheet (I use LibreOffice myself as it seems great at dealing with CSV) making sure the storedBytes field is imported as an integer.

  • Sort the storedBytes column in the spreadsheet to work out which log stream or streams are generating the most data.

In my case this worked - it turned out one of my log streams (with logs from a broken TCP pipe in a redis instance) was 4,000 times the size of all the other streams combined!

Meader answered 10/4, 2017 at 16:24 Comment(2)
aws logs describe-log-streams --log-group-name LOG_GROUP_NAME --query 'logStreams[?storedBytes > `0`].[logStreamName, storedBytes] | sort_by(@, &[1]) | reverse(@)' For an easier solutionPleistocene
Sadly storedBytes on LogStreams has been deprecated. "Important: On June 17, 2019, this parameter was deprecated for log streams, and is always reported as zero. This change applies only to log streams. The storedBytes parameter for log groups is not affected."Jacquard
G
4

An alternative to using the now deprecated storedBytes for log streams is to use Cloudwatch > Logs Insights and then run a query to count events by the log steam:

stats count(*) by @logStream

The log stream with the larger number of events will then probably be what is causing the high bill usage.

Gonad answered 3/8, 2022 at 11:28 Comment(0)
S
1

Just today I was solving very same problem. I have same retention date to all log groups so only thing I had to do was to sort them by size. I made following notebook (but feel free to put it in one single python file)

import boto3 
client = boto3.client('logs')
from pprint import pprint
log_group_list = []

ret = client.describe_log_groups()
if 'logGroups' in ret:
    for l in ret['logGroups']:
        log_group_list.append(l)

while 'nextToken' in ret:
    ret = client.describe_log_groups(nextToken = ret['nextToken'])
    for l in ret['logGroups']:
        log_group_list.append(l)
import copy

def sort_key(o):
    if 'storedBytes' in o:
        return o['storedBytes']
    else:  
        return 0

sorted_log_group = copy.deepcopy(log_group_list)

sorted_log_group.sort(reverse=True, key=sort_key)
import pandas as pd

df = pd.DataFrame(sorted_log_group)
def bytes_to_human_readable(bytes, units=[' bytes','KB','MB','GB','TB', 'PB', 'EB']):
    """ Returns a human readable string representation of bytes"""
    return "%3.1f %s" % (bytes, units[0]) if bytes < 1024 else bytes_to_human_readable(bytes / 1000, units[1:])


df['storedBytesH'] = df['storedBytes'].apply(bytes_to_human_readable)

df[['logGroupName', 'storedBytesH']].head(30)
Stevana answered 8/6, 2023 at 16:44 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.