You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

140 lines
5.6 KiB

  1. #
  2. # script for sending updates to cloudflare.com
  3. # 2014-2015 Christian Schoenebeck <christian dot schoenebeck at gmail dot com>
  4. # many thanks to Paul for testing and feedback during development
  5. #
  6. # This script is parsed by dynamic_dns_functions.sh inside send_update() function
  7. #
  8. # using following options from /etc/config/ddns
  9. # option username - your cloudflare e-mail
  10. # option password - cloudflare api key, you can get it from cloudflare.com/my-account/
  11. # option domain - your full hostname to update, in cloudflare its subdomain.domain
  12. # i.e. myhost.example.com where myhost is the subdomain and example.com is your domain
  13. #
  14. # variable __IP already defined with the ip-address to use for update
  15. #
  16. [ $use_https -eq 0 ] && write_log 14 "Cloudflare only support updates via Secure HTTP (HTTPS). Please correct configuration!"
  17. [ -z "$username" ] && write_log 14 "Service section not configured correctly! Missing 'username'"
  18. [ -z "$password" ] && write_log 14 "Service section not configured correctly! Missing 'password'"
  19. local __RECID __URL __KEY __KEYS __FOUND __SUBDOM __DOMAIN __TLD
  20. # split given Host/Domain into TLD, registrable domain, and subdomain
  21. split_FQDN $domain __TLD __DOMAIN __SUBDOM
  22. [ $? -ne 0 -o -z "$__DOMAIN" ] && \
  23. write_log 14 "Wrong Host/Domain configuration ($domain). Please correct configuration!"
  24. # put together what we need
  25. __DOMAIN="$__DOMAIN.$__TLD"
  26. # parse OpenWrt script with
  27. # functions for parsing and generating json
  28. . /usr/share/libubox/jshn.sh
  29. # function copied from /usr/share/libubox/jshn.sh
  30. # from BB14.09 for backward compatibility to AA12.09
  31. grep -i "json_get_keys" /usr/share/libubox/jshn.sh >/dev/null 2>&1 || json_get_keys() {
  32. local __dest="$1"
  33. local _tbl_cur
  34. if [ -n "$2" ]; then
  35. json_get_var _tbl_cur "$2"
  36. else
  37. _json_get_var _tbl_cur JSON_CUR
  38. fi
  39. local __var="${JSON_PREFIX}KEYS_${_tbl_cur}"
  40. eval "export -- \"$__dest=\${$__var}\"; [ -n \"\${$__var+x}\" ]"
  41. }
  42. # function to "sed" unwanted string parts from DATFILE
  43. cleanup() {
  44. # based on the sample output on cloudflare.com homepage we need to do some cleanup
  45. sed -i 's/^[ \t]*//;s/[ \t]*$//' $DATFILE # remove invisible chars at beginning and end of lines
  46. sed -i '/^-$/d' $DATFILE # remove lines with "-" (dash)
  47. sed -i '/^$/d' $DATFILE # remove empty lines
  48. sed -i "#'##g" $DATFILE # remove "'" (single quote)
  49. }
  50. # build url according to cloudflare client api at https://www.cloudflare.com/docs/client-api.html
  51. # to "rec_load_all" to detect rec_id needed for update
  52. __URL="https://www.cloudflare.com/api_json.html" # https://www.cloudflare.com/api_json.html
  53. __URL="${__URL}?a=rec_load_all" # -d 'a=rec_load_all'
  54. __URL="${__URL}&tkn=$password" # -d 'tkn=8afbe6dea02407989af4dd4c97bb6e25'
  55. __URL="${__URL}&email=$username" # -d 'email=sample@example.com'
  56. __URL="${__URL}&z=$__DOMAIN" # -d 'z=example.com'
  57. # lets request the data
  58. do_transfer "$__URL" || return 1
  59. cleanup # cleanup dat file
  60. json_load "$(cat $DATFILE)" # lets extract data
  61. __FOUND=0 # found record indicator
  62. json_get_var __RES "result" # cloudflare result of last request
  63. json_get_var __MSG "msg" # cloudflare error message
  64. [ "$__RES" != "success" ] && {
  65. write_log 4 "'rec_load_all' failed with error: \n$__MSG"
  66. return 1
  67. }
  68. json_select "response"
  69. json_select "recs"
  70. json_select "objs"
  71. json_get_keys __KEYS
  72. for __KEY in $__KEYS; do
  73. local __ZONE __DISPLAY __NAME __TYPE
  74. json_select "$__KEY"
  75. # json_get_var __ZONE "zone_name" # for debugging
  76. # json_get_var __DISPLAY "display_name" # for debugging
  77. json_get_var __NAME "name"
  78. json_get_var __TYPE "type"
  79. if [ "$__NAME" = "$domain" ]; then
  80. # we must verify IPv4 and IPv6 because there might be both for the same host
  81. [ \( $use_ipv6 -eq 0 -a "$__TYPE" = "A" \) -o \( $use_ipv6 -eq 1 -a "$__TYPE" = "AAAA" \) ] && {
  82. __FOUND=1 # mark found
  83. break # found leave for loop
  84. }
  85. fi
  86. json_select ..
  87. done
  88. [ $__FOUND -eq 0 ] && {
  89. # we don't need to continue trying to update cloudflare because record to update does not exist
  90. # user has to setup record first outside ddns-scripts
  91. write_log 14 "No valid record found at Cloudflare setup. Please create first!"
  92. }
  93. json_get_var __RECID "rec_id" # last thing to do get rec_id
  94. json_cleanup # cleanup
  95. write_log 7 "rec_id '$__RECID' detected for host/domain '$domain'"
  96. # build url according to cloudflare client api at https://www.cloudflare.com/docs/client-api.html
  97. # for "rec_edit" to update IP address
  98. __URL="https://www.cloudflare.com/api_json.html" # https://www.cloudflare.com/api_json.html
  99. __URL="${__URL}?a=rec_edit" # -d 'a=rec_edit'
  100. __URL="${__URL}&tkn=$password" # -d 'tkn=8afbe6dea02407989af4dd4c97bb6e25'
  101. __URL="${__URL}&id=$__RECID" # -d 'id=9001'
  102. __URL="${__URL}&email=$username" # -d 'email=sample@example.com'
  103. __URL="${__URL}&z=$__DOMAIN" # -d 'z=example.com'
  104. [ $use_ipv6 -eq 0 ] && __URL="${__URL}&type=A" # -d 'type=A' (IPv4)
  105. [ $use_ipv6 -eq 1 ] && __URL="${__URL}&type=AAAA" # -d 'type=AAAA' (IPv6)
  106. # handle subdomain or domain record
  107. [ -n "$__SUBDOM" ] && __URL="${__URL}&name=$__SUBDOM" # -d 'name=sub' (HOST/SUBDOMAIN)
  108. [ -z "$__SUBDOM" ] && __URL="${__URL}&name=$__DOMAIN" # -d 'name=example.com'(DOMAIN)
  109. __URL="${__URL}&content=$__IP" # -d 'content=1.2.3.4'
  110. __URL="${__URL}&service_mode=0" # -d 'service_mode=0'
  111. __URL="${__URL}&ttl=1" # -d 'ttl=1'
  112. # lets do the update
  113. do_transfer "$__URL" || return 1
  114. cleanup # cleanup tmp file
  115. json_load "$(cat $DATFILE)" # lets extract data
  116. json_get_var __RES "result" # cloudflare result of last request
  117. json_get_var __MSG "msg" # cloudflare error message
  118. [ "$__RES" != "success" ] && {
  119. write_log 4 "'rec_edit' failed with error:\n$__MSG"
  120. return 1
  121. }
  122. write_log 7 "Update of rec_id '$__RECID' successful"
  123. return 0