Skip to main content

Update Cloudflare DNS A and AAAA records from your current external IP addresses

Project description

cloudflare-dns-updater

CI Release PyPI Python ≥ 3.12 License: MIT

Update Cloudflare DNS A and AAAA records when your external IP address changes.

Install from PyPI: pipx install cloudflare-dns-updater.

Install with pipx (recommended)

pipx installs the CLI in an isolated environment and puts cloudflare-dns-updater on your PATH (usually ~/.local/bin).

# install pipx once (Debian/Ubuntu example)
sudo apt install pipx
pipx ensurepath
# open a new shell, or: source ~/.bashrc

pipx install cloudflare-dns-updater
cloudflare-dns-updater --help

Upgrade or reinstall later:

pipx upgrade cloudflare-dns-updater
# or pin a version:
pipx install cloudflare-dns-updater==0.1.0 --force

Other installers:

uv tool install cloudflare-dns-updater
pip install --user cloudflare-dns-updater   # not isolated; prefer pipx

Quick start

mkdir -p ~/.config/cloudflare-dns-updater
curl -fsSL https://raw.githubusercontent.com/the-hcma/cloudflare-dns-updater/main/config.example.json \
  -o ~/.config/cloudflare-dns-updater/config.json
# edit config.json — set cloudflare_api_token, zone, and dns_entries
cloudflare-dns-updater -v -d    # dry-run: discover IPs, no Cloudflare writes
cloudflare-dns-updater          # update DNS when IPv4 or IPv6 changed

Create a Cloudflare API token at https://dash.cloudflare.com/profile/api-tokens with permission to edit DNS records for your zone.

Before scheduling: run a dry-run and confirm exit code 0 or 1. If config is missing, the tool may write a starter file but still exits with code 2 until you replace the placeholder token and zone settings.

Run on a schedule (cron)

Cron runs with a minimal environment: set HOME (and usually PATH) so the tool finds ~/.config/cloudflare-dns-updater/config.json and your pipx-installed binary. Create and edit config before enabling the job — do not rely on the auto-created starter file in production.

Exit codes

Code Meaning
0 No update needed (addresses unchanged).
1 Success — DNS records were updated (or dry-run completed).
2 Hard failure — missing/invalid config, discovery error, Cloudflare API error, etc.

Exit 1 is a successful run that changed DNS; exit 2 is what you should alert on.

Logging and alerts

Cron sends mail when a job writes to stdout or stderr, not merely when the exit code is non-zero. Redirecting all output to a log file (>>/tmp/...log 2>&1) therefore suppresses cron mail, even when the tool fails.

Recommended — log to a file, mail only on hard failure (exit 2):

PATH=/usr/bin:/bin
HOME=/home/you
*/5 * * * * /home/you/.local/bin/cloudflare-dns-updater >>/tmp/cloudflare-dns-updater.log 2>&1; r=$?; if [ "$r" -eq 2 ]; then echo "cloudflare-dns-updater failed (exit $r); see /tmp/cloudflare-dns-updater.log" >&2; fi

The final echo runs only when exit code is 2, writing to stderr so cron can mail you without spamming on every successful DNS update (exit 1).

Alternative — no log file, mail on any non-zero exit (includes exit 1 when records were updated):

HOME=/home/you
*/5 * * * * /usr/bin/chronic /home/you/.local/bin/cloudflare-dns-updater

Do not wrap chronic with >>log 2>&1 — that captures chronic's failure output into the log and cron will not mail you.

Use -f if you want to recheck Cloudflare even when local state files show no change. For non-standard home paths or multiple configs, pass -c /full/path/to/config.json explicitly.

Configuration

Settings live in config.json. Search order:

  1. -c / --config path
  2. CLOUDFLARE_DNS_UPDATER_CONFIG environment variable
  3. ./config.json in the current working directory
  4. ~/.config/cloudflare-dns-updater/config.json

Copy from config.example.json:

{
  "cloudflare_api_token": "your-cloudflare-api-token",
  "zone": "example.com",
  "dns_entries": ["example.com", "home.example.com"],
  "record_ttl": 120,
  "ipv6_enabled": true
}
Field Required Description
cloudflare_api_token Yes Cloudflare API token.
zone Yes Cloudflare zone name.
dns_entries Yes Hostnames to update (A and optional AAAA).
record_ttl No TTL in seconds (default 120).
ipv6_enabled No Set false to skip AAAA updates (default true).
nest_router_url No Nest / Google Wifi base URL. Omitted = http://<LAN>.1 from your default route. Set null to skip Nest.

CLOUDFLARE_API_TOKEN in the environment overrides only the token field in the file. In cron, set HOME (and optionally CLOUDFLARE_API_TOKEN) in the crontab — cron does not load your shell profile.

IP discovery

IPv4

  1. Google Nest / WifiGET {nest_router_url}/api/v1/statuswan.localIpAddress
  2. Fallbackhttps://checkip.amazonaws.com

IPv6

When ipv6_enabled is true, the tool reads globally addressable IPv6 addresses from local network interfaces via netifaces.

Usage

cloudflare-dns-updater          # update when IPv4 or IPv6 changed
cloudflare-dns-updater -f       # recheck Cloudflare even if local state is unchanged
cloudflare-dns-updater -d       # dry-run (no Cloudflare calls, no state file writes)
cloudflare-dns-updater -v       # verbose logging and discovery details
cloudflare-dns-updater -c /path/to/config.json

Run cloudflare-dns-updater -h for full option descriptions.

Exit codes: 0 no update needed, 1 records updated (success), 2 configuration or execution failure. See Run on a schedule (cron) for alerting patterns.

State is stored under ~/.local/state/cloudflare-dns-updater/ (override with XDG_STATE_HOME).

Run from a git checkout

./bin/cloudflare-dns-updater -v -d

The wrapper runs uv sync --group dev when .venv is missing or uv.lock has changed, then invokes the CLI. You can still use uv run cloudflare-dns-updater directly after a manual sync.

Development

uv sync --group dev
uv run ruff check .
uv run ruff format .
uv run mypy src/dns_updater
uv run pytest
uv run pytest -m integration   # live network checks

Releases

Versioning and PyPI publish are documented in RELEASING.md. End users install with pipx install cloudflare-dns-updater.

Repository setup

This repo follows the-hcma conventions. Validate with:

scripts/check-repo-practices --repo the-hcma/cloudflare-dns-updater --suggest

See GRAPHITE.md for stacked PRs and the merge-it merge queue.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

cloudflare_dns_updater-0.3.0.tar.gz (72.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

cloudflare_dns_updater-0.3.0-py3-none-any.whl (20.5 kB view details)

Uploaded Python 3

File details

Details for the file cloudflare_dns_updater-0.3.0.tar.gz.

File metadata

  • Download URL: cloudflare_dns_updater-0.3.0.tar.gz
  • Upload date:
  • Size: 72.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for cloudflare_dns_updater-0.3.0.tar.gz
Algorithm Hash digest
SHA256 49e47f4f856dff2e310840c7b0c18117e3b70e085a56b49a6ea86ec35e5a4c69
MD5 8d73fc0a8e26bf0c3e6d70bb60deaec7
BLAKE2b-256 f839ce92cd95d1787e914d4c64cae4366bcc9d58794dd0bb3f42a439bf09dad2

See more details on using hashes here.

Provenance

The following attestation bundles were made for cloudflare_dns_updater-0.3.0.tar.gz:

Publisher: release-please.yml on the-hcma/cloudflare-dns-updater

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file cloudflare_dns_updater-0.3.0-py3-none-any.whl.

File metadata

File hashes

Hashes for cloudflare_dns_updater-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 4164708a63f49561a2623501c992db1afb1128ea1d5e9abe5db704e05a7997ed
MD5 d78205693b7ff4057e6dbd5da609d5d7
BLAKE2b-256 a8a3e4915a97d33a1e5cc90ab7fe24de208eac0c8fb6122ea14b669753af0e1d

See more details on using hashes here.

Provenance

The following attestation bundles were made for cloudflare_dns_updater-0.3.0-py3-none-any.whl:

Publisher: release-please.yml on the-hcma/cloudflare-dns-updater

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page