Skip to content

Latest commit

 

History

History
248 lines (195 loc) · 7.54 KB

README.md

File metadata and controls

248 lines (195 loc) · 7.54 KB

Azure Billing Exporter

Build Docker Pulls

Expose Azure Billing data to prometheus format. Show daily, weekly, monthly cost by subscription. Also allow add custom billing query.

Quick start. Docker images

docker run\
        --env ApiSettings__SubscriptionId='YOUR_SUBSCRIPTION_ID'\
        --env ApiSettings__TenantId='YOUR_TENANT_ID'\
        --env ApiSettings__ClientId='YOUR_CLIENT_ID'\
        --env ApiSettings__ClientSecret='CLIENT_SECRET_SP'\
        -v /PATH_TO_REPO/custom_queries/:/app/custom_queries/\
        -v /PATH_TO_REPO/custom_collectors.yml:/app/custom_collectors.yml\
        -p 9301:80 azure_billing_exporter:latest

How to run locally

  1. Create ServicePrincipal

    This SP should have access as Billing reader role see Manage billing access

  2. Set configuration

    1. Environment Variables

          EXPORT ApiSettings__SubscriptionId="YOUR_SUBSCRIPTION_ID"
          EXPORT ApiSettings__TenantId="YOUR_TENANT_ID"
          EXPORT ApiSettings__ClientId="YOUR_CLIENT_ID"
          EXPORT ApiSettings__ClientSecret="CLIENT_SECRET_SP"
    2. Configuration file appsettings.json

      Using for local developing

      "ApiSettings": {
          "SubscriptionId": "YOUR_SUBSCRIPTION_ID",
          "TenantId": "YOUR_TENANT_ID",
          "ClientId": "YOUR_CLIENT_ID",
          "ClientSecret": "CLIENT_SECRET_SP"
      },
  3. Tracing logs

    For trace all billing query and response set log level to trace info appsettings.Development.json

    "Serilog": {
    "MinimumLevel": {
        "Default": "Trace"
    }
  4. Install dotnet SDK

    Download and install .NET Core 3.1 SDK or above https://dotnet.microsoft.com/download/dotnet-core/3.1

  5. Run dotnet

    dotnet run --project AzureBillingExporter/AzureBillingExporter.csproj
  6. Open metrics

    curl http://localhost:5000/metrics

Architecture

According Microsoft documentation application may create only 30 API calls per minute. After that threshold application will get Too Many Requests response from API.

Error code 429 - Call count has exceeded rate limits

To enable a consistent experience for all Cost Management subscribers, Cost Management APIs are rate limited. When you reach the limit, you receive the HTTP status code 429: Too many requests. The current throughput limits for our APIs are as follows:

30 calls per minute - It's done per scope, per user, or application.

200 calls per minute - It's done per tenant, per user, or application.

To avoid such errors, this exporter has background job to get data from API. Received cost data placed in memory cache. Prometheus scriber calls on /metrics get data from cache and get quick response.

In case of Too Many Requests errors, background job waits 1 minute before next calls.

Configuration

Setting Type Description
LogsAtJsonFormat bool Write logs in plain text or JSON format
CollectPeriodInMinutes int Period in minutes to make API call to the Azure, to get metrics
CachePeriodInMinutes int Period in minutes to cache API call results
CustomCollectorsFilePath string Path to YAML file with custom collectors (see Custom Metrics)

Metrics

Metrics Name Description
azure_billing_daily_today Today all costs !!! Cost data could delay in 12-48 hours !!!
azure_billing_daily_yesterday Yesterday all costs !!! Cost data could delay in 12-48 hours !!!
azure_billing_daily_before_yesterday Day before yesterday all costs
azure_billing_monthly Costs by current month

Custom Metrics

Set custom metrics configs into custom_collectors.yml

# A Prometheus metric with (optional) additional labels, value and labels populated from one query.
metrics:
  - metric_name: azure_billing_by_resource_group
    type: gauge
    help: 'Costs by resource group by current month'
    key_labels:
      # Populated from the `market` column of each row.
      - ResourceGroupName
    static_labels:
      # Arbitrary key/value pair
      company: dodo
    value: PreTaxCost
    replace_date_labels_to_enum: true  # replace `05/01/2020 00:00:00` to `last_month`, `UsageDate="20200624"` to `yesterday`. Default false
    query_file: './custom_queries/azure_billing_by_resource_group.json'

You can set custom path to collectors.yaml file

Into appsettings.Development.json (or env CustomCollectorsFilePath) set:

  "CustomCollectorsFilePath" : "./local/custom_collectors.yml",

Query to billing api

{
  "type": "ActualCost",
  "dataSet": {
    "granularity": "None",
    "aggregation": {
      "totalCost": {
        "name": "PreTaxCost",
        "function": "Sum"
      }
    },
    "grouping": [
      {
        "type": "Dimension",
        "name": "ResourceGroupName"
      }
    ],
    "sorting": [
      {
        "direction": "descending",
        "name": "PreTaxCost"
      }
    ]
  },
  "timeframe": "Custom",
  "timePeriod": {
    "from": "{{ CurrentMonthStart }}",
    "to": "{{ TodayEnd }}"
  }
}

Datetime constants into query files

You can use special constant into query file. For this use {{ }} template notation Liquid Template Language . DateTime Constants (using server datetime). If today is '2020-06-23T08:12:45':

Constant Description Example
CurrentMonthStart This month start date time. '2020-06-01T00:00:00.0000000'
PrevMonthStart Previous month start date. '2020-05-01T00:00:00.0000000'
BeforePrevMonthStart Before previous month. '2020-04-01T00:00:00.0000000'
TodayEnd End of current date. '2020-06-22T23:59:59.0000000'
YesterdayStart Yesterday start date. '2020-06-22T00:00:00.0000000'
WeekAgo This day of week but week ago, start of the day. '2020-06-16T00:00:00.0000000'
YearAgo This month first day year ago. '2019-06-01T00:00:00.0000000'

All this constants you can use into billing query json files:

  "timePeriod": {
    "from": "{{ PrevMonthStart }}",
    "to": "{{ TodayEnd }}"
  }

Try Azure Billing Query on sandbox

Go to docs: https://docs.microsoft.com/en-us/rest/api/cost-management/query/usage

Click Try It

Content-type: application/json Scope: subscriptions/YOUR_SUBSCRIPTION_ID Api version: 2019-10-01

Body:

{
  "type": "ActualCost",
  "dataSet": {
    "granularity": "Daily",
    "aggregation": {
      "totalCost": {
        "name": "PreTaxCost",
        "function": "Sum"
      }
    },
    "sorting": [
      {
        "direction": "ascending",
        "name": "UsageDate"
      }
    ]
  },
  "timeframe": "Custom",
  "timePeriod": {
    "from": "2020-06-24T00:00:00+00:00",
    "to": "2020-06-24T23:59:59+00:00"
  }
}

Notice

Request duration measuring for exporter:

▶ curl -o /dev/null -s -w 'Total: %{time_total}s\n' http://localhost:5000/metrics
Total: 0.009669s