debuntu-toolkit/cfddnsv6.sh

64 lines
3.4 KiB
Bash

#!/bin/bash
# AUTHOR: Josh North
# EMAIL: josh.north@point808.com
# BRIEF: This script updates CloudFlare IPv6 DNS records with your current IP
# address, as pulled from the interface directly. Best use case is having an
# ISP who arbitrarily changes your prefix. For safety, we do not create
# records, we only update records that match the hostname in the zone, so
# you must log in to CloudFlare and create your records first!
# USAGE: Put this file in /usr/local/bin/cfddnsv6.sh or wherever you prefer is ok.
# Make sure to chmod +x and make it executable. Change the settings below to
# Match your login and environment needs. Run it from command line to test.
# Right now we run this from cron--- at reboot and every so often after.
# This example runs it at reboot and every half hour, adjust to taste
# @reboot /usr/local/bin/cfddnsv6.sh >/dev/null 2>&1
# */30 * * * * /usr/local/bin/cfddnsv6.sh >/dev/null 2>&1
# SETTINGS:
CF_EMAIL="you@zone.com" # Email address you use to log in to CloudFlare.
CF_KEY="YourApiKeyHere" # API key for CloudFlare account (get from Account Settings page).
CF_ZONE="zone.com" # MUST match zone you want to update host in!
CF_HOST="webserver" # Hostname of record. Create in CloudFlare first!
CF_IF="eth0" # What interface should we get the IP address for?
CF_LOG="/var/log/cfddnsv6.log" # Where to log actions taken by this script.
# -------------------- NOTHING TO CHANGE BELOW THIS LINE --------------------
# CloudFlare API URI
CF_URL="https://api.cloudflare.com/client/v4"
CF_AUTH=(-H "X-Auth-Email: $CF_EMAIL" -H "X-Auth-Key: $CF_KEY" -H "Content-Type: application/json")
CURLCMD="curl -sS -X"
# Make sure we have all needed programs
command -v curl >/dev/null 2>&1 || { echo "This script requires curl but I cannot find it on this system... Exiting now." >&2; exit 1; }
# Log start of script
printf "%s\n" "$(date) Beginning DNS update process" >> $CF_LOG
# This is a pause to wait for tentative addresses to disappear. You may comment this out if you have reason to do so, I include it because in my networks, these addresses go away.
while ip -6 addr show tentative | grep . > /dev/null ; do sleep 1 ; done
# Try to get the best global address from the chosen interface
CF_IPV6=$(ip addr show dev $CF_IF | grep -Ev "deprecated" | sed -e's/^.*inet6 \([^ ]*\)\/.*$/\1/;t;d' | grep -Ev "^fd|^fe80" | sed -n '$p')
# Look up CloudFlare zone identifier
CF_ZONE_ID=$($CURLCMD GET "$CF_URL/zones?name=$CF_ZONE" "${CF_AUTH[@]}" | cut -d '"' -f6)
# Get record identifier from CloudFlare
CF_RECORD_ID=$($CURLCMD GET "$CF_URL/zones/$CF_ZONE_ID/dns_records?type=AAAA&name=$CF_HOST.$CF_ZONE" "${CF_AUTH[@]}" | cut -d '"' -f6)
# Log what we are going to try
printf "%s\n" "$(date) IP on $CF_IF is $CF_IPV6" >> $CF_LOG
printf "%s\n" "$(date) Record to update is $CF_HOST.$CF_ZONE ($CF_RECORD_ID)" >> $CF_LOG
# Build the request string and submit
CF_REQUEST=$($CURLCMD PUT "$CF_URL/zones/$CF_ZONE_ID/dns_records/$CF_RECORD_ID" "${CF_AUTH[@]}" \
--data "{\"id\":\"$CF_RECORD_ID\",\"type\":\"AAAA\",\"name\":\"$CF_HOST\",\"content\":\"$CF_IPV6\",\"zone_id\":\"$CF_ZONE_ID\",\"zone_name\":\"$CF_ZONE\",\"data\":{}}" | sed -n -e 's/^.*\("success":\)/\1/p')
# Log output
printf "%s\n" "$(date) Result string is $CF_REQUEST" >> $CF_LOG
# All done
exit 0
# EOF