2021-05-08 04:00:15 -06:00
#!/bin/sh
#
2021-05-09 00:42:53 -06:00
# update Cloudflare DNS records with current (dynamic) IP address
2021-05-08 04:00:15 -06:00
# Script by Asif Bacchus <asif@bacchus.cloud>
# Last modified: May 7, 2021
#
### text formatting presets using tput
if command -v tput >/dev/null; then
bold = $( tput bold)
cyan = $( tput setaf 6)
err = $( tput bold) $( tput setaf 1)
magenta = $( tput setaf 5)
norm = $( tput sgr0)
ok = $( tput setaf 2)
warn = $( tput bold) $( tput setaf 3)
yellow = $( tput setaf 3)
width = $( tput cols)
else
bold = ""
cyan = ""
err = ""
magenta = ""
norm = ""
ok = ""
warn = ""
yellow = ""
width = 80
fi
2018-09-11 23:32:42 -06:00
2021-05-08 04:00:15 -06:00
### functions
badParam( ) {
if [ " $1 " = "null" ] ; then
printf "\n%sERROR: '%s' cannot have a NULL (empty) value.\n" " $err " " $2 "
printf "%sPlease use '--help' for assistance.%s\n\n" " $cyan " " $norm "
exit 1
elif [ " $1 " = "dne" ] ; then
printf "\n%sERROR: '%s %s'\n" " $err " " $2 " " $3 "
printf "file or directory does not exist or is empty.%s\n\n" " $norm "
exit 1
elif [ " $1 " = "errMsg" ] ; then
printf "\n%sERROR: %s%s\n\n" " $err " " $2 " " $norm "
exit 1
fi
}
2018-09-11 23:32:42 -06:00
2021-05-08 04:00:15 -06:00
exitError( ) {
case " $1 " in
2021-05-09 04:33:41 -06:00
3)
errMsg = "Unable to connect to Cloudflare servers. This is probably a temporary networking issue. Please try again later."
; ;
2021-05-08 04:00:15 -06:00
10)
errMsg = "Unable to auto-detect IP address. Try again later or supply the IP address to be used."
; ;
20)
2021-05-09 00:42:53 -06:00
errMsg = "Cloudflare authorized email address (cfEmail) is either null or undefined. Please check your Cloudflare credentials file."
2021-05-08 04:00:15 -06:00
; ;
21)
2021-05-09 00:42:53 -06:00
errMsg = "Cloudflare authorized API key (cfKey) is either null or undefined. Please check your Cloudflare credentials file."
2021-05-08 04:00:15 -06:00
; ;
22)
2021-05-09 00:42:53 -06:00
errMsg = "Cloudflare zone id (cfZoneId) is either null or undefined. Please check your Cloudflare credentials file."
2021-05-08 04:00:15 -06:00
; ;
25)
2021-05-09 07:08:44 -06:00
errMsg = "Cloudflare API error. Please review any 'CF-ERR:' lines in this log for details."
2021-05-08 04:00:15 -06:00
; ;
2021-05-09 04:33:41 -06:00
26)
2021-05-09 08:48:56 -06:00
errMsg = " ${ failedHostCount } host update(s) failed. Any 'CF-ERR:' lines noted in this log may help determine what went wrong. "
2021-05-08 04:00:15 -06:00
; ;
*)
2021-05-09 09:46:15 -06:00
writeLog error "An unspecified error occurred. (code: 99)"
printf "%s[%s] -- Cloudflare DDNS update-script: completed with error(s) --%s\n" " $err " " $( stamp) " " $norm " >>" $logFile "
2021-05-08 04:00:15 -06:00
exit 99
; ;
esac
2021-05-09 09:46:15 -06:00
writeLog error " $errMsg " " $1 "
printf "%s[%s] -- Cloudflare DDNS update-script: completed with error(s) --%s\n" " $err " " $( stamp) " " $norm " >>" $logFile "
2021-05-08 04:00:15 -06:00
exit " $1 "
}
2018-09-12 06:07:55 -06:00
2021-05-08 04:00:15 -06:00
exitOK( ) {
2021-05-09 09:46:15 -06:00
printf "%s[%s] -- Cloudflare DDNS update-script: completed successfully --%s\n" " $ok " " $( stamp) " " $norm " >>" $logFile "
2021-05-08 04:00:15 -06:00
exit 0
2018-09-12 06:07:55 -06:00
}
2021-05-09 06:49:49 -06:00
listCFErrors( ) {
# extract error codes and messages in separate variables, replace newlines with underscores
codes = " $( printf "%s" " $1 " | jq -r '.errors | .[] | .code' | tr '\n' '_' ) "
messages = " $( printf "%s" " $1 " | jq -r '.errors | .[] | .message' | tr '\n' '_' ) "
# iterate codes and messages and assemble into coherent messages in log
while [ -n " $codes " ] && [ -n " $messages " ] ; do
# get first code and message in respective sets
code = " ${ codes %%_* } "
message = " ${ messages %%_* } "
# update codes and messages sets by removing first item in each set
codes = " ${ codes #*_ } "
messages = " ${ messages #*_ } "
# output to log
2021-05-09 09:46:15 -06:00
writeLog cf " $message " " $code "
2021-05-09 06:49:49 -06:00
done
2021-05-08 04:00:15 -06:00
}
2018-09-12 06:07:55 -06:00
2021-05-09 06:39:55 -06:00
scriptExamples( ) {
newline
printf "Update Cloudflare DNS host A/AAAA records with current IP address.\n"
printf "%sUsage: %s --records host.domain.tld[,host2.domain.tld,...] [parameters]%s\n\n" " $bold " " $scriptName " " $norm "
2021-05-09 08:45:24 -06:00
textBlock " ${ magenta } --- usage examples --- ${ norm } "
2021-05-09 06:39:55 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches " ${ scriptName } -r myserver.mydomain.net "
textBlock "Update Cloudflare DNS records for myserver.mydomain.net with the auto-detected public IP4 address. Credentials will be expected in the default location and the log will be written in the default location also."
2021-05-09 06:39:55 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches " ${ scriptName } -r myserver.mydomain.net -6 "
textBlock "Same as above, but update AAAA host records with the auto-detected public IP6 address."
2021-05-09 06:39:55 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches " ${ scriptName } -r myserver.mydomain.net,myserver2.mydomain.net -l /var/log/cfddns.log --nc "
textBlock "Update DNS entries for both listed hosts using auto-detected IP4 address. Write a non-coloured log to '/var/log/cfddns.log'."
2021-05-09 06:39:55 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches " ${ scriptName } -r myserver.mydomain.net,myserver2.mydomain.net -l /var/log/cfddns.log --ip6 --ip fd21:7a62:2737:9c3a::a151 "
textBlock "Update DNS AAAA entries for listed hosts using the *specified* IP address. Write a colourful log to the location specified."
2021-05-09 06:39:55 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches " ${ scriptName } -r myserver.mydomain.net -c /root/cloudflare.creds -l /var/log/cfddns.log --ip 1.2.3.4 "
textBlock "Update DNS A entry for listed hostname with the provided IP address. Read cloudflare credentials file from specified location, save log in specified location."
2021-05-09 06:39:55 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches " ${ scriptName } -r myserver.mydomain.net -c /root/cloudflare.creds -l /var/log/cfddns.log -6 -i fd21:7a62:2737:9c3a::a151 "
textBlock "Exact same as above, but change the AAAA record. This is how you run the script once for IP4 and again for IP6."
2021-05-09 06:39:55 -06:00
exit 0
}
2021-05-08 04:00:15 -06:00
scriptHelp( ) {
newline
2021-05-09 00:42:53 -06:00
printf "Update Cloudflare DNS host A/AAAA records with current IP address.\n"
2021-05-08 04:00:15 -06:00
printf "%sUsage: %s --records host.domain.tld[,host2.domain.tld,...] [parameters]%s\n\n" " $bold " " $scriptName " " $norm "
2021-05-09 08:45:24 -06:00
textBlock "The only required parameter is '--records' which is a comma-delimited list of hostnames to update. However, there are several other options which may be useful to implement."
textBlock "Parameters are listed below and followed by a description of their effect. If a default value exists, it will be listed on the following line in (parentheses)."
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlock " ${ magenta } --- script related parameters --- ${ norm } "
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "-c | --cred | --creds | --credentials | -f (deprecated, backward-compatibility)"
textBlock "Path to file containing your Cloudflare *token* credentials. Please refer to the repo README for more information on format, etc."
textBlockDefaults " ( ${ accountFile } ) "
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "-l | --log"
textBlock "Path where the log file should be written."
textBlockDefaults " ( ${ logFile } ) "
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "--nc | --no-color | --no-colour"
textBlock "Switch value. Disables ANSI colours in the log. Useful if you review the logs using a reader that does not parse ANSI colour codes."
textBlockDefaults "(disabled: print logs in colour)"
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "--log-console"
textBlock "Switch value. Output log to console (stdout) instead of a log file. Can be combined with --nc if desired."
textBlockDefaults "(disabled: output to log file)"
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "--no-log"
textBlock "Switch value. Do not create a log (i.e. no console, no file). You will not have *any* output from the script if you choose this option, so you will not know if updates succeeded or failed."
textBlockDefaults "(disabled: output to log file)"
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "-h | --help | -?"
textBlock "Display this help screen."
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "--examples"
textBlock "Show some usage examples."
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlock " ${ magenta } --- DNS related parameters --- ${ norm } "
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "-r | --record | --records"
textBlock "Comma-delimited list of hostnames for which IP addresses should be updated in Cloudflare DNS. This parameter is REQUIRED. Note that this script will only *update* records, it will not create new ones. If you supply hostnames that are not already defined in DNS, the script will log a warning and will skip those hostnames."
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "-i | --ip | --ip-address | -a | --address"
textBlock "New IP address for DNS host records. If you omit this, the script will attempt to auto-detect your public IP address and use that."
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "-4 | --ip4 | --ipv4"
textBlock "Switch value. Update Host 'A' records (IP4) only. Note that this script can only update either A *or* AAAA records. If you need to update both, you'll have to run the script once in IP4 mode and again in IP6 mode. If you specify both this switch and the IP6 switch, the last one specified will take effect."
textBlockDefaults "(enabled: update A records)"
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlockSwitches "-6 | --ip6 | --ipv6"
textBlock "Switch value. Update Host 'AAAA' records (IP6) only. Note that this script can only update either A *or* AAAA records. If you need to update both, you'll have to run the script once in IP4 mode and again in IP6 mode. If you specify both this switch and the IP4 switch, the last one specified will take effect."
textBlockDefaults "(disabled: update A records)"
2021-05-08 04:00:15 -06:00
newline
2021-05-09 08:45:24 -06:00
textBlock "Please refer to the repo README for more detailed information regarding this script and how to automate and monitor it."
2021-05-08 04:00:15 -06:00
newline
exit 0
}
2018-09-12 06:07:55 -06:00
2021-05-09 06:39:55 -06:00
stamp( ) {
( date +%F" " %T)
}
newline( ) {
printf "\n"
2018-09-12 06:07:55 -06:00
}
2021-05-09 08:45:24 -06:00
textBlock( ) {
2021-05-08 04:00:15 -06:00
printf "%s\n" " $1 " | fold -w " $width " -s
}
2018-09-12 06:07:55 -06:00
2021-05-09 08:45:24 -06:00
textBlockDefaults( ) {
2021-05-08 04:00:15 -06:00
printf "%s%s%s\n" " $yellow " " $1 " " $norm "
2018-09-11 23:32:42 -06:00
}
2021-05-09 08:45:24 -06:00
textBlockSwitches( ) {
2021-05-08 04:00:15 -06:00
printf "%s%s%s\n" " $cyan " " $1 " " $norm "
}
2018-09-11 23:32:42 -06:00
2021-05-09 09:46:15 -06:00
writeLog( ) {
case " $1 " in
2021-05-09 10:24:15 -06:00
cf)
printf "[%s] CF-ERR: %s (code: %s)\n" " $( stamp) " " $2 " " $3 " >>" $logFile "
; ;
err)
printf "%s[%s] ERR: %s%s\n" " $err " " $( stamp) " " $2 " " $norm " >>" $logFile "
; ;
error)
printf "%s[%s] ERROR: %s (code: %s)%s\n" " $err " " $( stamp) " " $2 " " $3 " " $norm " >>" $logFile "
; ;
process)
printf "%s[%s] %s... %s" " $cyan " " $( stamp) " " $2 " " $norm " >>" $logFile "
; ;
process-done)
printf "%s%s%s\n" " $cyan " " $2 " " $norm " >>" $logFile "
; ;
process-error)
printf "%sERROR%s\n" " $err " " $norm " >>" $logFile "
; ;
process-warning)
printf "%s%s%s\n" " $warn " " $2 " " $norm " >>" $logFile "
; ;
stamped)
printf "[%s] %s\n" " $( stamp) " " $2 " >>" $logFile "
; ;
success)
printf "%s[%s] SUCCESS: %s%s\n" " $ok " " $( stamp) " " $2 " " $norm " >>" $logFile "
; ;
warn)
printf "%s[%s] WARN: %s%s\n" " $warn " " $( stamp) " " $2 " " $norm " >>" $logFile "
; ;
warning)
printf "%s[%s] WARNING: %s%s\n" " $warn " " $( stamp) " " $2 " " $norm " >>" $logFile "
; ;
*)
printf "%s\n" " $2 " >>" $logFile "
; ;
2021-05-09 09:46:15 -06:00
esac
}
2021-05-08 04:00:15 -06:00
### default variable values
scriptPath = " $( CDPATH = '' \c d -- " $( dirname -- " $0 " ) " && pwd -P) "
scriptName = " $( basename " $0 " ) "
logFile = " $scriptPath / ${ scriptName %.* } .log "
accountFile = " $scriptPath /cloudflare.credentials "
colourizeLogFile = 1
dnsRecords = ""
dnsSeparator = ","
ipAddress = ""
2018-09-12 02:11:47 -06:00
ip4 = 1
ip6 = 0
2021-05-08 04:00:15 -06:00
ip4DetectionSvc = "http://ipv4.icanhazip.com"
ip6DetectionSvc = "http://ipv6.icanhazip.com"
invalidDomainCount = 0
2021-05-09 08:48:56 -06:00
failedHostCount = 0
2018-09-12 00:48:58 -06:00
2021-05-08 04:00:15 -06:00
### process startup parameters
2018-09-26 00:35:33 -06:00
if [ -z " $1 " ] ; then
2021-05-08 04:00:15 -06:00
scriptHelp
2018-09-11 23:32:42 -06:00
fi
2021-05-08 04:00:15 -06:00
while [ $# -gt 0 ] ; do
case " $1 " in
-h | -\? | --help)
# display help
scriptHelp
; ;
--examples)
# display sample commands
scriptExamples
; ;
-l | --log)
# set log file location
if [ -n " $2 " ] ; then
logFile = " ${ 2 %/ } "
shift
else
badParam null " $@ "
fi
; ;
--log-console)
# log to the console instead of a file
logFile = "/dev/stdout"
; ;
--no-log)
# do not log anything
logFile = "/dev/null"
; ;
--nc | --no-color | --no-colour)
# do not colourize log file
colourizeLogFile = 0
; ;
-c | --cred* | -f)
2021-05-09 00:42:53 -06:00
# path to Cloudflare credentials file
2021-05-08 04:00:15 -06:00
if [ -n " $2 " ] ; then
if [ -f " $2 " ] && [ -s " $2 " ] ; then
accountFile = " ${ 2 %/ } "
shift
else
badParam dne " $@ "
fi
else
badParam null " $@ "
fi
; ;
-r | --record | --records)
# DNS records to update
if [ -n " $2 " ] ; then
dnsRecords = $( printf "%s" " $2 " | sed -e 's/ //g' )
shift
else
badParam null " $@ "
fi
; ;
-i | --ip | --ip-address | -a | --address)
# IP address to use (not parsed for correctness)
if [ -n " $2 " ] ; then
ipAddress = " $2 "
shift
else
badParam null " $@ "
fi
; ;
-4 | --ip4 | --ipv4)
# operate in IP4 mode (default)
ip4 = 1
ip6 = 0
; ;
-6 | --ip6 | --ipv6)
# operate in IP6 mode
ip6 = 1
ip4 = 0
; ;
*)
printf "\n%sUnknown option: %s\n" " $err " " $1 "
printf "%sUse '--help' for valid options.%s\n\n" " $cyan " " $norm "
exit 1
; ;
2018-09-11 23:32:42 -06:00
esac
2021-05-08 04:00:15 -06:00
shift
2018-09-11 23:32:42 -06:00
done
2021-05-08 04:00:15 -06:00
### pre-flight checks
if ! command -v curl >/dev/null; then
printf "\n%sThis script requires 'curl' be installed and accessible. Exiting.%s\n\n" " $err " " $norm "
exit 2
2018-09-12 00:08:06 -06:00
fi
2021-05-08 04:00:15 -06:00
if ! command -v jq >/dev/null; then
printf "\n%sThis script requires 'jq' be installed and accessible. Exiting.%s\n\n" " $err " " $norm "
exit 2
2018-09-14 00:01:30 -06:00
fi
2021-05-08 04:00:15 -06:00
[ -z " $dnsRecords " ] && badParam errMsg "You must specify at least one DNS record to update. Exiting."
# verify credentials file exists and is not empty (default check)
if [ ! -f " $accountFile " ] || [ ! -s " $accountFile " ] ; then
2021-05-09 00:42:53 -06:00
badParam errMsg " Cannot find Cloudflare credentials file ( ${ accountFile } ). Exiting. "
2021-05-08 04:00:15 -06:00
fi
# turn off log file colourization if parameter is set
if [ " $colourizeLogFile " -eq 0 ] ; then
bold = ""
cyan = ""
err = ""
magenta = ""
norm = ""
ok = ""
warn = ""
yellow = ""
2018-09-13 21:25:12 -06:00
fi
2021-05-08 04:00:15 -06:00
### initial log entries
{
2021-05-09 09:46:15 -06:00
printf "%s[%s] -- Cloudflare DDNS update-script: starting --%s\n" " $ok " " $( stamp) " " $norm "
2021-05-09 03:58:35 -06:00
printf "Parameters:\n"
2021-05-08 04:00:15 -06:00
printf "script path: %s\n" " $scriptPath / $scriptName "
printf "credentials file: %s\n" " $accountFile "
2018-09-11 23:32:42 -06:00
2021-05-08 04:00:15 -06:00
if [ " $ip4 " -eq 1 ] ; then
2021-05-09 09:46:15 -06:00
printf "mode: IP4\n"
elif [ " $ip6 " -eq 1 ] ; then
printf "mode: IP6\n"
2018-09-12 02:11:47 -06:00
fi
2021-05-09 09:46:15 -06:00
# detect and report IP address
if [ -z " $ipAddress " ] ; then
# detect public ip address
if [ " $ip4 " -eq 1 ] ; then
if ! ipAddress = " $( curl -s $ip4DetectionSvc ) " ; then
printf "ddns ip address:%s ERROR%s\n" " $err " " $norm "
exitError 10
fi
2021-05-08 04:00:15 -06:00
fi
2021-05-09 09:46:15 -06:00
if [ " $ip6 " -eq 1 ] ; then
if ! ipAddress = " $( curl -s $ip6DetectionSvc ) " ; then
printf "ddns ip address:%s ERROR%s\n" " $err " " $norm "
exitError 10
fi
fi
printf "ddns ip address (detected): %s\n" " $ipAddress "
else
printf "ddns ip address (supplied): %s\n" " $ipAddress "
2018-09-12 01:15:39 -06:00
fi
2018-09-12 00:48:58 -06:00
2021-05-09 09:46:15 -06:00
# iterate DNS records to update
2021-05-09 09:47:41 -06:00
dnsRecordsToUpdate = " $( printf '%s' " ${ dnsRecords } " | sed " s/ ${ dnsSeparator } * $// " ) $dnsSeparator "
2021-05-09 09:46:15 -06:00
while [ -n " $dnsRecordsToUpdate " ] && [ " $dnsRecordsToUpdate " != " $dnsSeparator " ] ; do
record = " ${ dnsRecordsToUpdate %% ${ dnsSeparator } * } "
dnsRecordsToUpdate = " ${ dnsRecordsToUpdate #* ${ dnsSeparator } } "
2021-05-09 08:44:47 -06:00
2021-05-09 09:46:15 -06:00
if [ -z " $record " ] ; then continue ; fi
printf "updating record: %s\n" " $record "
done
2018-09-12 03:00:43 -06:00
2021-05-09 09:46:15 -06:00
printf "(end of parameter list)\n"
} >>" $logFile "
2021-05-08 04:00:15 -06:00
2021-05-09 00:42:53 -06:00
### read Cloudflare credentials
2021-05-09 09:46:15 -06:00
writeLog process "Reading Cloudflare credentials"
2021-05-08 04:00:15 -06:00
case " $accountFile " in
/*)
# absolute path, use as-is
# shellcheck source=./cloudflare.credentials
. " $accountFile "
; ;
*)
# relative path, rewrite
# shellcheck source=./cloudflare.credentials
. " ./ $accountFile "
; ;
esac
2021-05-09 04:35:46 -06:00
if [ -z " $cfKey " ] ; then
2021-05-09 09:46:15 -06:00
writeLog process-error
2021-05-08 04:00:15 -06:00
exitError 21
2021-05-09 04:35:46 -06:00
fi
if [ -z " $cfZoneId " ] ; then
2021-05-09 09:46:15 -06:00
writeLog process-error
2021-05-08 04:00:15 -06:00
exitError 22
2018-09-12 06:20:06 -06:00
fi
2021-05-09 09:46:15 -06:00
writeLog process-done "DONE"
2021-05-08 04:00:15 -06:00
2021-05-09 04:33:41 -06:00
### connect to Cloudflare and do what needs to be done!
2021-05-08 04:00:15 -06:00
dnsRecordsToUpdate = " $dnsRecords $dnsSeparator "
if [ " $ip4 " -eq 1 ] ; then
recordType = "A"
elif [ " $ip6 " -eq 1 ] ; then
recordType = "AAAA"
2018-09-12 04:39:37 -06:00
fi
2021-05-09 04:33:41 -06:00
# iterate hosts to update
2021-05-09 08:44:47 -06:00
while [ -n " $dnsRecordsToUpdate " ] && [ " $dnsRecordsToUpdate " != " $dnsSeparator " ] ; do
2021-05-08 04:00:15 -06:00
record = " ${ dnsRecordsToUpdate %% ${ dnsSeparator } * } "
dnsRecordsToUpdate = " ${ dnsRecordsToUpdate #* ${ dnsSeparator } } "
2021-05-09 08:44:47 -06:00
if [ -z " $record " ] ; then continue ; fi
2021-05-09 09:46:15 -06:00
writeLog process " Processing ${ record } "
2021-05-09 04:33:41 -06:00
# exit if curl/network error
if ! cfLookup = " $( curl -s -X GET " https://api.cloudflare.com/client/v4/zones/ ${ cfZoneId } /dns_records?name= ${ record } &type= ${ recordType } " \
2021-05-08 04:00:15 -06:00
-H " Authorization: Bearer ${ cfKey } " \
-H "Content-Type: application/json" ) " ; then
2021-05-09 09:46:15 -06:00
writeLog process-error
2021-05-09 04:33:41 -06:00
exitError 3
fi
# exit if API error
# exit here since API errors on GET request probably indicates authentication error which would affect all remaining operations
# no reason to continue processing other hosts and pile-up errors which might look like a DoS attempt
cfSuccess = " $( printf "%s" " $cfLookup " | jq -r '.success' ) "
if [ " $cfSuccess " = "false" ] ; then
2021-05-09 09:46:15 -06:00
writeLog process-error
2021-05-09 06:49:49 -06:00
listCFErrors " $cfLookup "
2021-05-08 04:00:15 -06:00
exitError 25
2018-09-12 04:39:37 -06:00
fi
2021-05-09 04:33:41 -06:00
resultCount = " $( printf "%s" " $cfLookup " | jq '.result_info.count' ) "
# skip to next host if cannot find existing host record (this script *updates* only, does not create!)
2021-05-08 04:00:15 -06:00
if [ " $resultCount " = "0" ] ; then
2021-05-09 04:33:41 -06:00
# warn if record of host not found
2021-05-09 09:46:15 -06:00
writeLog process-warning "NOT FOUND"
writeLog warn " Cannot find existing record to update for DNS entry: ${ record } "
2021-05-08 04:00:15 -06:00
invalidDomainCount = $(( invalidDomainCount + 1 ))
2021-05-09 04:33:41 -06:00
continue
fi
objectId = $( printf "%s" " $cfLookup " | jq -r '.result | .[] | .id' )
currentIpAddr = $( printf "%s" " $cfLookup " | jq -r '.result | .[] | .content' )
2021-05-09 09:46:15 -06:00
writeLog process-done " FOUND: IP = ${ currentIpAddr } "
2021-05-09 04:33:41 -06:00
# skip to next hostname if record already up-to-date
if [ " $currentIpAddr " = " $ipAddress " ] ; then
2021-05-09 09:46:15 -06:00
writeLog stamped " IP address for ${ record } is already up-to-date "
2021-05-09 04:33:41 -06:00
continue
fi
# update record
2021-05-09 09:46:15 -06:00
writeLog process " Updating IP address for ${ record } "
2021-05-09 04:33:41 -06:00
updateJSON = " $( jq -n --arg key0 content --arg value0 " ${ ipAddress } " '{($key0):$value0}' ) "
# exit if curl/network error
if ! cfResult = " $( curl -s -X PATCH " https://api.cloudflare.com/client/v4/zones/ ${ cfZoneId } /dns_records/ ${ objectId } " \
-H " Authorization: Bearer ${ cfKey } " \
-H "Content-Type: application/json" \
--data " ${ updateJSON } " ) " ; then
2021-05-09 09:46:15 -06:00
writeLog process-error
2021-05-09 04:33:41 -06:00
exitError 3
fi
# note update success or failure
cfSuccess = " $( printf "%s" " $cfResult " | jq '.success' ) "
if [ " $cfSuccess " = "true" ] ; then
2021-05-09 09:46:15 -06:00
writeLog process-done "DONE"
writeLog success " IP address for ${ record } updated. "
2018-09-12 08:45:33 -06:00
else
2021-05-09 09:46:15 -06:00
writeLog process-error
2021-05-09 06:49:49 -06:00
listCFErrors " $cfResult "
2021-05-09 09:46:15 -06:00
writeLog err " Unable to update IP address for ${ record } "
2021-05-09 04:33:41 -06:00
# do not exit with error, API error here is probably an update issue specific to this host
# increment counter and note it after all processing finished
2021-05-09 08:48:56 -06:00
failedHostCount = $(( failedHostCount + 1 ))
2018-09-12 08:45:33 -06:00
fi
done
2021-05-08 04:00:15 -06:00
# exit
if [ " $invalidDomainCount " -ne 0 ] ; then
2021-05-09 10:24:15 -06:00
writeLog warning " ${ invalidDomainCount } invalid host(s) supplied for updating. "
2021-05-08 04:00:15 -06:00
fi
2021-05-09 08:48:56 -06:00
if [ " $failedHostCount " -ne 0 ] ; then
2021-05-09 04:33:41 -06:00
exitError 26
2018-09-13 20:54:08 -06:00
else
2021-05-08 04:00:15 -06:00
exitOK
2018-09-13 20:54:08 -06:00
fi
2018-09-12 04:17:48 -06:00
2021-05-08 04:00:15 -06:00
### exit return codes
# 0: normal exit, no errors
# 1: invalid or unknown parameter
# 2: cannot find or access required external program(s)
2021-05-09 04:33:41 -06:00
# 3: curl error (probably connection)
2021-05-08 04:00:15 -06:00
# 10: cannot auto-detect IP address
# 21: accountFile has a null or missing cfKey variable
# 22: accountFile has a null or missing cfZoneId variable
2021-05-09 04:33:41 -06:00
# 25: Cloudflare API error
# 26: one or more updates failed
2021-05-08 04:00:15 -06:00
# 99: unspecified error occurred