Skip to content

Bash script that uploads the Cloudflare Analytics API data to InfluxDB on an hourly basis

License

Notifications You must be signed in to change notification settings

rare-magma/cloudflare-exporter

Repository files navigation

cloudflare-exporter

Bash script that uploads the Cloudflare Analytics API data to influxdb on an hourly basis

Dependencies

Relevant documentation

Installation

With Docker

docker-compose

  1. Configure cloudflare_exporter.conf and cloudflare_zone_list.json (see the configuration section below).

  2. Run it.

    docker compose up --detach

docker build & run

  1. Build the docker image.

    docker build . --tag cloudflare-exporter
  2. Configure cloudflare_exporter.conf, cloudflare_zone_list.json and cloudflare_kv_namespaces.conf (see the configuration section below).

  3. Run it.

     docker run --rm --init --tty --interactive --read-only --cap-drop ALL --security-opt no-new-privileges:true --cpus 2 -m 64m --pids-limit 16 --volume ./cloudflare_exporter.conf:/app/cloudflare_exporter.conf:ro --volume ./cloudflare_zone_list.json:/app/cloudflare_zone_list.json:ro --volume ./cloudflare_kv_namespaces_list.conf:/app/cloudflare_kv_namespaces_list.conf:ro ghcr.io/rare-magma/cloudflare-exporter:latest

With the Makefile

For convenience, you can install this exporter with the following command or follow the manual process described in the next paragraph.

make install
$EDITOR $HOME/.config/cloudflare_exporter.conf
$EDITOR $HOME/.config/cloudflare_zone_list.json
$EDITOR $HOME/.config/cloudflare_kv_namespaces_list.conf

Manually

  1. Copy cloudflare_exporter.sh to $HOME/.local/bin/ and make it executable.

  2. Copy cloudflare_exporter.conf, cloudflare_zone_list.json and cloudflare_kv_namespaces_list.conf to $HOME/.config/, configure them (see the configuration section below) and make them read only.

  3. Copy the systemd unit and timer to $HOME/.config/systemd/user/:

    cp cloudflare-exporter.* $HOME/.config/systemd/user/
  4. and run the following command to activate the timer:

    systemctl --user enable --now cloudflare-exporter.timer

It's possible to trigger the execution by running manually:

systemctl --user start cloudflare-exporter.service

Config file

The config file has a few options:

INFLUXDB_HOST='influxdb.example.com'
INFLUXDB_API_TOKEN='ZXhhbXBsZXRva2VuZXhhcXdzZGFzZGptcW9kcXdvZGptcXdvZHF3b2RqbXF3ZHFhc2RhCg=='
ORG='home'
BUCKET='cloudflare'
CLOUDFLARE_API_TOKEN='ZXhhbXBsZXRva2VuZXhhcXdzZGFzZGptcW9kcXdvZGptcXdvZHF3b2RqbXF3ZHFhc2RhCg=='
CLOUDFLARE_ACCOUNT_EMAIL='email@example.com'
CLOUDFLARE_ACCOUNT_TAG='aa0a0aa000a0000aa00a00aa0e000a0a'
  • INFLUXDB_HOST should be the FQDN of the influxdb server.
  • ORG should be the name of the influxdb organization that contains the cloudflare data bucket defined below.
  • BUCKET should be the name of the influxdb bucket that will hold the cloudflare data.
  • INFLUXDB_API_TOKEN should be the influxdb API token value.
    • This token should have write access to the BUCKET defined above.
  • CLOUDFLARE_API_TOKEN should be the cloudflare API token value.
    • This token should be assigned the All zones - Analytics:Read permission.
    • Additionally, the Account Analytics:Read permission is necessary for workers metrics.
  • CLOUDFLARE_ACCOUNT_TAG should be the tag associated with the cloudflare account.
  • Required for cloudflare accounts on a paid plan:
    • CLOUDFLARE_ACCOUNT_EMAIL should be the email associated with the paid cloudflare account.

Zone list file

The zone list file should contain a list of zone ids and domain names in json format:

[
  { "id": "999999aba99dd9999ef99ab78965ab1c", "domain": "example.com" },
  { "id": "111111aba11dd1111ef11ab11111ab1c", "domain": "example2.com" }
]

KV namespaces list file

The KV namespaces list file is optional and should contain a KV namespace id per line:

999999aba99dd9999ef99ab78965ab1c
111111aba11dd1111ef11ab11111ab1c

Troubleshooting

Run the script manually with bash set to trace:

bash -x $HOME/.local/bin/cloudflare_exporter.sh

Check the systemd service logs and timer info with:

journalctl --user --unit cloudflare-exporter.service
systemctl --user list-timers

Exported metrics

  • cloudflare_stats_browser: Page views broken down by browser
  • cloudflare_stats_content_type: Request statistics broken down by content type
  • cloudflare_stats_countries: Request statistics broken down by country
  • cloudflare_stats_ip: Request statistics broken down by robot type
  • cloudflare_stats_responses: Request statistics broken down by response status code
  • cloudflare_stats: General request statistics
  • cloudflare_stats_workers: Workers statistics grouped by hour
  • cloudflare_stats_pf: Pages Functions statistics grouped by hour
  • cloudflare_stats_kv_ops: KV operation statistics grouped by hour
  • cloudflare_stats_kv_storage: KV storage statistics

Exported metrics example

cloudflare_stats_browser,zone="example.com",browserFamily="ChromeMobileWebview" pageViews=2 1703894400
cloudflare_stats_content_type,zone="example.com",edgeResponse="txt" bytes=11089,requests=8 1703894400
cloudflare_stats_countries,zone="example.com",country="CA" bytes=312170,requests=14,threats=0 1703894400
cloudflare_stats_ip,zone="example.com",ipType="searchEngine" requests=21 1703894400
cloudflare_stats_responses,zone="example.com",status=403 requests=1 1703894400
cloudflare_stats,zone="example.com" bytes=2032039,cachedBytes=40607,cachedRequests=17,encryptedBytes=2020727,encryptedRequests=251,pageViews=178,requests=266,threats=0,uniqueVisitors=2 1703894400
cloudflare_stats_workers,account=aa0a0aa000a0000aa00a00aa0e000a0a,worker=worker-name status="scriptThrewException",cpuTimeP50=1246,cpuTimeP99=1246,durationP50=0.001246,durationP99=0.001246,responseBodySizeP50=0,responseBodySizeP99=0,wallTimeP50=1605,wallTimeP99=1605,clientDisconnects=0,cpuTimeUs=1246,duration=0.001246,errors=1,requests=1,responseBodySize=0,subrequests=0,wallTime=1605 1727340566
cloudflare_stats_pf,account=aa0a0aa000a0000aa00a00aa0e000a0a,scriptName=pages-worker--1111111-production status="success",usageModel="standard",cpuTimeP50=3492,cpuTimeP99=3700,durationP50=0.004010875,durationP99=0.016313376,clientDisconnects=0,duration=0.024276250000000003,errors=0,requests=3,responseBodySize=4614,subrequests=0,wallTime=194210 1727431200
cloudflare_stats_kv_ops,account=aa0a0aa000a0000aa00a00aa0e000a0a,namespace=999999aba99dd9999ef99ab78965ab1c actionType="read",result="hot_read",responseStatusCode=200,latencyMsP50=116,latencyMsP99=116,objectBytes=1737,requests=1 1727445600
cloudflare_stats_kv_storage,account=aa0a0aa000a0000aa00a00aa0e000a0a,namespace=999999aba99dd9999ef99ab78965ab1c byteCount=5369,keyCount=1 1727442000

Example grafana dashboard

In cloudflare-dashboard.json there is an example of the kind of dashboard that can be built with cloudflare-exporter data:

Grafana dashboard screenshot

Import it by doing the following:

  1. Create a dashboard
  2. Click the dashboard's settings button on the top right.
  3. Go to JSON Model and then paste there the content of the cloudflare-dashboard.json file.

Uninstallation

With the Makefile

For convenience, you can uninstall this exporter with the following command or follow the process described in the next paragraph.

make uninstall

Manually

Run the following command to deactivate the timer:

systemctl --user disable --now cloudflare-exporter.timer

Delete the following files:

~/.local/bin/cloudflare_exporter.sh
~/.config/cloudflare_exporter.conf
~/.config/cloudflare_zone_list.json
~/.config/cloudflare_kv_namespaces_list.conf
~/.config/systemd/user/cloudflare-exporter.timer
~/.config/systemd/user/cloudflare-exporter.service

Credits

This project takes inspiration from the following:

About

Bash script that uploads the Cloudflare Analytics API data to InfluxDB on an hourly basis

Topics

Resources

License

Stars

Watchers

Forks

Packages