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.
		
		
		
		
		
			
		
			
				
					
					
						
							372 lines
						
					
					
						
							12 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							372 lines
						
					
					
						
							12 KiB
						
					
					
				| #!/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 | |
| }
 |