Bash script that uploads the Cloudflare Analytics API data to influxdb on an hourly basis
- awk
- bash
- coreutils (cat)
- coreutils (date)
- curl
- gzip
- influxdb v2+
- jq
- systemd
- Optional:
-
Configure
cloudflare_exporter.confandcloudflare_zone_list.json(see the configuration section below). -
Run it.
docker compose up --detach
-
Build the docker image.
docker build . --tag cloudflare-exporter -
Configure
cloudflare_exporter.conf,cloudflare_zone_list.jsonandcloudflare_kv_namespaces.conf(see the configuration section below). -
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
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-
Copy
cloudflare_exporter.shto$HOME/.local/bin/and make it executable. -
Copy
cloudflare_exporter.conf,cloudflare_zone_list.jsonandcloudflare_kv_namespaces_list.confto$HOME/.config/, configure them (see the configuration section below) and make them read only. -
Copy the systemd unit and timer to
$HOME/.config/systemd/user/:cp cloudflare-exporter.* $HOME/.config/systemd/user/
-
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.serviceThe 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_HOSTshould be the FQDN of the influxdb server.ORGshould be the name of the influxdb organization that contains the cloudflare data bucket defined below.BUCKETshould be the name of the influxdb bucket that will hold the cloudflare data.INFLUXDB_API_TOKENshould be the influxdb API token value.- This token should have write access to the
BUCKETdefined above.
- This token should have write access to the
CLOUDFLARE_API_TOKENshould be the cloudflare API token value.- This token should be assigned the
All zones - Analytics:Readpermission. - Additionally, the
Account Analytics:Readpermission is necessary for workers metrics.
- This token should be assigned the
CLOUDFLARE_ACCOUNT_TAGshould be the tag associated with the cloudflare account.- Required for cloudflare accounts on a paid plan:
CLOUDFLARE_ACCOUNT_EMAILshould be the email associated with the paid cloudflare account.
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" }
]The KV namespaces list file is optional and should contain a KV namespace id per line:
999999aba99dd9999ef99ab78965ab1c
111111aba11dd1111ef11ab11111ab1c
Run the script manually with bash set to trace:
bash -x $HOME/.local/bin/cloudflare_exporter.shCheck the systemd service logs and timer info with:
journalctl --user --unit cloudflare-exporter.service
systemctl --user list-timers- 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
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 1727442000In cloudflare-dashboard.json there is an example of the kind of dashboard that can be built with cloudflare-exporter data:
Import it by doing the following:
- Create a dashboard
- Click the dashboard's settings button on the top right.
- Go to JSON Model and then paste there the content of the
cloudflare-dashboard.jsonfile.
For convenience, you can uninstall this exporter with the following command or follow the process described in the next paragraph.
make uninstallRun the following command to deactivate the timer:
systemctl --user disable --now cloudflare-exporter.timerDelete 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.serviceThis project takes inspiration from the following:
