373 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
			
		
		
	
	
			373 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
| #!/usr/bin/env sh
 | |
| # shellcheck disable=SC2034
 | |
| dns_freedns_info='FreeDNS
 | |
| Site: FreeDNS.afraid.org
 | |
| Docs: github.com/acmesh-official/acme.sh/wiki/dnsapi#dns_freedns
 | |
| Options:
 | |
|  FREEDNS_User Username
 | |
|  FREEDNS_Password Password
 | |
| Issues: github.com/acmesh-official/acme.sh/issues/2305
 | |
| Author: David Kerr <https://github.com/dkerr64>
 | |
| '
 | |
| 
 | |
| ########  Public functions #####################
 | |
| 
 | |
| # Export FreeDNS userid and password in following variables...
 | |
| #  FREEDNS_User=username
 | |
| #  FREEDNS_Password=password
 | |
| # login cookie is saved in acme account config file so userid / pw
 | |
| # need to be set only when changed.
 | |
| 
 | |
| #Usage: dns_freedns_add   _acme-challenge.www.domain.com   "XKrxpRBosdIKFzxW_CT3KLZNf6q0HG9i01zxXp5CPBs"
 | |
| dns_freedns_add() {
 | |
|   fulldomain="$1"
 | |
|   txtvalue="$2"
 | |
| 
 | |
|   _info "Add TXT record using FreeDNS"
 | |
|   _debug "fulldomain: $fulldomain"
 | |
|   _debug "txtvalue: $txtvalue"
 | |
| 
 | |
|   if [ -z "$FREEDNS_User" ] || [ -z "$FREEDNS_Password" ]; then
 | |
|     FREEDNS_User=""
 | |
|     FREEDNS_Password=""
 | |
|     if [ -z "$FREEDNS_COOKIE" ]; then
 | |
|       _err "You did not specify the FreeDNS username and password yet."
 | |
|       _err "Please export as FREEDNS_User / FREEDNS_Password and try again."
 | |
|       return 1
 | |
|     fi
 | |
|     using_cached_cookies="true"
 | |
|   else
 | |
|     FREEDNS_COOKIE="$(_freedns_login "$FREEDNS_User" "$FREEDNS_Password")"
 | |
|     if [ -z "$FREEDNS_COOKIE" ]; then
 | |
|       return 1
 | |
|     fi
 | |
|     using_cached_cookies="false"
 | |
|   fi
 | |
| 
 | |
|   _debug "FreeDNS login cookies: $FREEDNS_COOKIE (cached = $using_cached_cookies)"
 | |
| 
 | |
|   _saveaccountconf FREEDNS_COOKIE "$FREEDNS_COOKIE"
 | |
| 
 | |
|   # We may have to cycle through the domain name to find the
 | |
|   # TLD that we own...
 | |
|   i=1
 | |
|   wmax="$(echo "$fulldomain" | tr '.' ' ' | wc -w)"
 | |
|   while [ "$i" -lt "$wmax" ]; do
 | |
|     # split our full domain name into two parts...
 | |
|     sub_domain="$(echo "$fulldomain" | cut -d. -f -"$i")"
 | |
|     i="$(_math "$i" + 1)"
 | |
|     top_domain="$(echo "$fulldomain" | cut -d. -f "$i"-100)"
 | |
|     _debug "sub_domain: $sub_domain"
 | |
|     _debug "top_domain: $top_domain"
 | |
| 
 | |
|     DNSdomainid="$(_freedns_domain_id "$top_domain")"
 | |
|     if [ "$?" = "0" ]; then
 | |
|       _info "Domain $top_domain found at FreeDNS, domain_id $DNSdomainid"
 | |
|       break
 | |
|     else
 | |
|       _info "Domain $top_domain not found at FreeDNS, try with next level of TLD"
 | |
|     fi
 | |
|   done
 | |
| 
 | |
|   if [ -z "$DNSdomainid" ]; then
 | |
|     # If domain ID is empty then something went wrong (top level
 | |
|     # domain not found at FreeDNS).
 | |
|     _err "Domain $top_domain not found at FreeDNS"
 | |
|     return 1
 | |
|   fi
 | |
| 
 | |
|   # Add in new TXT record with the value provided
 | |
|   _debug "Adding TXT record for $fulldomain, $txtvalue"
 | |
|   _freedns_add_txt_record "$FREEDNS_COOKIE" "$DNSdomainid" "$sub_domain" "$txtvalue"
 | |
|   return $?
 | |
| }
 | |
| 
 | |
| #Usage: fulldomain txtvalue
 | |
| #Remove the txt record after validation.
 | |
| dns_freedns_rm() {
 | |
|   fulldomain="$1"
 | |
|   txtvalue="$2"
 | |
| 
 | |
|   _info "Delete TXT record using FreeDNS"
 | |
|   _debug "fulldomain: $fulldomain"
 | |
|   _debug "txtvalue: $txtvalue"
 | |
| 
 | |
|   # Need to read cookie from conf file again in case new value set
 | |
|   # during login to FreeDNS when TXT record was created.
 | |
|   FREEDNS_COOKIE="$(_readaccountconf "FREEDNS_COOKIE")"
 | |
|   _debug "FreeDNS login cookies: $FREEDNS_COOKIE"
 | |
| 
 | |
|   TXTdataid="$(_freedns_data_id "$fulldomain" "TXT")"
 | |
|   if [ "$?" != "0" ]; then
 | |
|     _info "Cannot delete TXT record for $fulldomain, record does not exist at FreeDNS"
 | |
|     return 1
 | |
|   fi
 | |
|   _debug "Data ID's found, $TXTdataid"
 | |
| 
 | |
|   # now we have one (or more) TXT record data ID's. Load the page
 | |
|   # for that record and search for the record txt value.  If match
 | |
|   # then we can delete it.
 | |
|   lines="$(echo "$TXTdataid" | wc -l)"
 | |
|   _debug "Found $lines TXT data records for $fulldomain"
 | |
|   i=0
 | |
|   while [ "$i" -lt "$lines" ]; do
 | |
|     i="$(_math "$i" + 1)"
 | |
|     dataid="$(echo "$TXTdataid" | sed -n "${i}p")"
 | |
|     _debug "$dataid"
 | |
| 
 | |
|     htmlpage="$(_freedns_retrieve_data_page "$FREEDNS_COOKIE" "$dataid")"
 | |
|     if [ "$?" != "0" ]; then
 | |
|       if [ "$using_cached_cookies" = "true" ]; then
 | |
|         _err "Has your FreeDNS username and password changed?  If so..."
 | |
|         _err "Please export as FREEDNS_User / FREEDNS_Password and try again."
 | |
|       fi
 | |
|       return 1
 | |
|     fi
 | |
| 
 | |
|     echo "$htmlpage" | grep "value=\""$txtvalue"\"" >/dev/null
 | |
|     if [ "$?" = "0" ]; then
 | |
|       # Found a match... delete the record and return
 | |
|       _info "Deleting TXT record for $fulldomain, $txtvalue"
 | |
|       _freedns_delete_txt_record "$FREEDNS_COOKIE" "$dataid"
 | |
|       return $?
 | |
|     fi
 | |
|   done
 | |
| 
 | |
|   # If we get this far we did not find a match
 | |
|   # Not necessarily an error, but log anyway.
 | |
|   _info "Cannot delete TXT record for $fulldomain, $txtvalue. Does not exist at FreeDNS"
 | |
|   return 0
 | |
| }
 | |
| 
 | |
| ####################  Private functions below ##################################
 | |
| 
 | |
| # usage: _freedns_login username password
 | |
| # print string "cookie=value" etc.
 | |
| # returns 0 success
 | |
| _freedns_login() {
 | |
|   export _H1="Accept-Language:en-US"
 | |
|   username="$1"
 | |
|   password="$2"
 | |
|   url="https://freedns.afraid.org/zc.php?step=2"
 | |
| 
 | |
|   _debug "Login to FreeDNS as user $username"
 | |
| 
 | |
|   htmlpage="$(_post "username=$(printf '%s' "$username" | _url_encode)&password=$(printf '%s' "$password" | _url_encode)&submit=Login&action=auth" "$url")"
 | |
| 
 | |
|   if [ "$?" != "0" ]; then
 | |
|     _err "FreeDNS login failed for user $username bad RC from _post"
 | |
|     return 1
 | |
|   fi
 | |
| 
 | |
|   cookies="$(grep -i '^Set-Cookie.*dns_cookie.*$' "$HTTP_HEADER" | _head_n 1 | tr -d "\r\n" | cut -d " " -f 2)"
 | |
| 
 | |
|   # if cookies is not empty then logon successful
 | |
|   if [ -z "$cookies" ]; then
 | |
|     _debug3 "htmlpage: $htmlpage"
 | |
|     _err "FreeDNS login failed for user $username. Check $HTTP_HEADER file"
 | |
|     return 1
 | |
|   fi
 | |
| 
 | |
|   printf "%s" "$cookies"
 | |
|   return 0
 | |
| }
 | |
| 
 | |
| # usage _freedns_retrieve_subdomain_page login_cookies
 | |
| # echo page retrieved (html)
 | |
| # returns 0 success
 | |
| _freedns_retrieve_subdomain_page() {
 | |
|   export _H1="Cookie:$1"
 | |
|   export _H2="Accept-Language:en-US"
 | |
|   url="https://freedns.afraid.org/subdomain/"
 | |
| 
 | |
|   _debug "Retrieve subdomain page from FreeDNS"
 | |
| 
 | |
|   htmlpage="$(_get "$url")"
 | |
| 
 | |
|   if [ "$?" != "0" ]; then
 | |
|     _err "FreeDNS retrieve subdomains failed bad RC from _get"
 | |
|     return 1
 | |
|   elif [ -z "$htmlpage" ]; then
 | |
|     _err "FreeDNS returned empty subdomain page"
 | |
|     return 1
 | |
|   fi
 | |
| 
 | |
|   _debug3 "htmlpage: $htmlpage"
 | |
| 
 | |
|   printf "%s" "$htmlpage"
 | |
|   return 0
 | |
| }
 | |
| 
 | |
| # usage _freedns_retrieve_data_page login_cookies data_id
 | |
| # echo page retrieved (html)
 | |
| # returns 0 success
 | |
| _freedns_retrieve_data_page() {
 | |
|   export _H1="Cookie:$1"
 | |
|   export _H2="Accept-Language:en-US"
 | |
|   data_id="$2"
 | |
|   url="https://freedns.afraid.org/subdomain/edit.php?data_id=$2"
 | |
| 
 | |
|   _debug "Retrieve data page for ID $data_id from FreeDNS"
 | |
| 
 | |
|   htmlpage="$(_get "$url")"
 | |
| 
 | |
|   if [ "$?" != "0" ]; then
 | |
|     _err "FreeDNS retrieve data page failed bad RC from _get"
 | |
|     return 1
 | |
|   elif [ -z "$htmlpage" ]; then
 | |
|     _err "FreeDNS returned empty data page"
 | |
|     return 1
 | |
|   fi
 | |
| 
 | |
|   _debug3 "htmlpage: $htmlpage"
 | |
| 
 | |
|   printf "%s" "$htmlpage"
 | |
|   return 0
 | |
| }
 | |
| 
 | |
| # usage _freedns_add_txt_record login_cookies domain_id subdomain value
 | |
| # returns 0 success
 | |
| _freedns_add_txt_record() {
 | |
|   export _H1="Cookie:$1"
 | |
|   export _H2="Accept-Language:en-US"
 | |
|   domain_id="$2"
 | |
|   subdomain="$3"
 | |
|   value="$(printf '%s' "$4" | _url_encode)"
 | |
|   url="https://freedns.afraid.org/subdomain/save.php?step=2"
 | |
| 
 | |
|   htmlpage="$(_post "type=TXT&domain_id=$domain_id&subdomain=$subdomain&address=%22$value%22&send=Save%21" "$url")"
 | |
| 
 | |
|   if [ "$?" != "0" ]; then
 | |
|     _err "FreeDNS failed to add TXT record for $subdomain bad RC from _post"
 | |
|     return 1
 | |
|   elif ! grep "200 OK" "$HTTP_HEADER" >/dev/null; then
 | |
|     _debug3 "htmlpage: $htmlpage"
 | |
|     _err "FreeDNS failed to add TXT record for $subdomain. Check $HTTP_HEADER file"
 | |
|     return 1
 | |
|   elif _contains "$htmlpage" "security code was incorrect"; then
 | |
|     _debug3 "htmlpage: $htmlpage"
 | |
|     _err "FreeDNS failed to add TXT record for $subdomain as FreeDNS requested security code"
 | |
|     _err "Note that you cannot use automatic DNS validation for FreeDNS public domains"
 | |
|     return 1
 | |
|   fi
 | |
| 
 | |
|   _debug3 "htmlpage: $htmlpage"
 | |
|   _info "Added acme challenge TXT record for $fulldomain at FreeDNS"
 | |
|   return 0
 | |
| }
 | |
| 
 | |
| # usage _freedns_delete_txt_record login_cookies data_id
 | |
| # returns 0 success
 | |
| _freedns_delete_txt_record() {
 | |
|   export _H1="Cookie:$1"
 | |
|   export _H2="Accept-Language:en-US"
 | |
|   data_id="$2"
 | |
|   url="https://freedns.afraid.org/subdomain/delete2.php"
 | |
| 
 | |
|   htmlheader="$(_get "$url?data_id%5B%5D=$data_id&submit=delete+selected" "onlyheader")"
 | |
| 
 | |
|   if [ "$?" != "0" ]; then
 | |
|     _err "FreeDNS failed to delete TXT record for $data_id bad RC from _get"
 | |
|     return 1
 | |
|   elif ! _contains "$htmlheader" "200 OK"; then
 | |
|     _debug2 "htmlheader: $htmlheader"
 | |
|     _err "FreeDNS failed to delete TXT record $data_id"
 | |
|     return 1
 | |
|   fi
 | |
| 
 | |
|   _info "Deleted acme challenge TXT record for $fulldomain at FreeDNS"
 | |
|   return 0
 | |
| }
 | |
| 
 | |
| # usage _freedns_domain_id domain_name
 | |
| # echo the domain_id if found
 | |
| # return 0 success
 | |
| _freedns_domain_id() {
 | |
|   # Start by escaping the dots in the domain name
 | |
|   search_domain="$(echo "$1" | sed 's/\./\\./g')"
 | |
| 
 | |
|   # Sometimes FreeDNS does not return the subdomain page but rather
 | |
|   # returns a page regarding becoming a premium member.  This usually
 | |
|   # happens after a period of inactivity.  Immediately trying again
 | |
|   # returns the correct subdomain page.  So, we will try twice to
 | |
|   # load the page and obtain our domain ID
 | |
|   attempts=2
 | |
|   while [ "$attempts" -gt "0" ]; do
 | |
|     attempts="$(_math "$attempts" - 1)"
 | |
| 
 | |
|     htmlpage="$(_freedns_retrieve_subdomain_page "$FREEDNS_COOKIE")"
 | |
|     if [ "$?" != "0" ]; then
 | |
|       if [ "$using_cached_cookies" = "true" ]; then
 | |
|         _err "Has your FreeDNS username and password changed?  If so..."
 | |
|         _err "Please export as FREEDNS_User / FREEDNS_Password and try again."
 | |
|       fi
 | |
|       return 1
 | |
|     fi
 | |
| 
 | |
|     domain_id="$(echo "$htmlpage" | tr -d " \t\r\n\v\f" | sed 's/<tr>/@<tr>/g' | tr '@' '\n' |
 | |
|       grep "<td>$search_domain</td>\|<td>$search_domain(.*)</td>" |
 | |
|       sed -n 's/.*\(edit\.php?edit_domain_id=[0-9a-zA-Z]*\).*/\1/p' |
 | |
|       cut -d = -f 2)"
 | |
|     # The above beauty extracts domain ID from the html page...
 | |
|     # strip out all blank space and new lines. Then insert newlines
 | |
|     # before each table row <tr>
 | |
|     # search for the domain within each row (which may or may not have
 | |
|     # a text string in brackets (.*) after it.
 | |
|     # And finally extract the domain ID.
 | |
|     if [ -n "$domain_id" ]; then
 | |
|       printf "%s" "$domain_id"
 | |
|       return 0
 | |
|     fi
 | |
|     _debug "Domain $search_domain not found. Retry loading subdomain page ($attempts attempts remaining)"
 | |
|   done
 | |
|   _debug "Domain $search_domain not found after retry"
 | |
|   return 1
 | |
| }
 | |
| 
 | |
| # usage _freedns_data_id domain_name record_type
 | |
| # echo the data_id(s) if found
 | |
| # return 0 success
 | |
| _freedns_data_id() {
 | |
|   # Start by escaping the dots in the domain name
 | |
|   search_domain="$(echo "$1" | sed 's/\./\\./g')"
 | |
|   record_type="$2"
 | |
| 
 | |
|   # Sometimes FreeDNS does not return the subdomain page but rather
 | |
|   # returns a page regarding becoming a premium member.  This usually
 | |
|   # happens after a period of inactivity.  Immediately trying again
 | |
|   # returns the correct subdomain page.  So, we will try twice to
 | |
|   # load the page and obtain our domain ID
 | |
|   attempts=2
 | |
|   while [ "$attempts" -gt "0" ]; do
 | |
|     attempts="$(_math "$attempts" - 1)"
 | |
| 
 | |
|     htmlpage="$(_freedns_retrieve_subdomain_page "$FREEDNS_COOKIE")"
 | |
|     if [ "$?" != "0" ]; then
 | |
|       if [ "$using_cached_cookies" = "true" ]; then
 | |
|         _err "Has your FreeDNS username and password changed?  If so..."
 | |
|         _err "Please export as FREEDNS_User / FREEDNS_Password and try again."
 | |
|       fi
 | |
|       return 1
 | |
|     fi
 | |
| 
 | |
|     data_id="$(echo "$htmlpage" | tr -d " \t\r\n\v\f" | sed 's/<tr>/@<tr>/g' | tr '@' '\n' |
 | |
|       grep "<td[a-zA-Z=#]*>$record_type</td>" |
 | |
|       grep "<ahref.*>$search_domain</a>" |
 | |
|       sed -n 's/.*\(edit\.php?data_id=[0-9a-zA-Z]*\).*/\1/p' |
 | |
|       cut -d = -f 2)"
 | |
|     # The above beauty extracts data ID from the html page...
 | |
|     # strip out all blank space and new lines. Then insert newlines
 | |
|     # before each table row <tr>
 | |
|     # search for the record type withing each row (e.g. TXT)
 | |
|     # search for the domain within each row (which is within a <a..>
 | |
|     # </a> anchor. And finally extract the domain ID.
 | |
|     if [ -n "$data_id" ]; then
 | |
|       printf "%s" "$data_id"
 | |
|       return 0
 | |
|     fi
 | |
|     _debug "Domain $search_domain not found. Retry loading subdomain page ($attempts attempts remaining)"
 | |
|   done
 | |
|   _debug "Domain $search_domain not found after retry"
 | |
|   return 1
 | |
| }
 |