dns_he.sh 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. #!/usr/bin/env sh
  2. ########################################################################
  3. # Hurricane Electric hook script for acme.sh
  4. #
  5. # Environment variables:
  6. #
  7. # - $HE_Username (your dns.he.net username)
  8. # - $HE_Password (your dns.he.net password)
  9. #
  10. # Author: Ondrej Simek <me@ondrejsimek.com>
  11. # Git repo: https://github.com/angel333/acme.sh
  12. #-- dns_he_add() - Add TXT record --------------------------------------
  13. # Usage: dns_he_add _acme-challenge.subdomain.domain.com "XyZ123..."
  14. dns_he_add() {
  15. _full_domain=$1
  16. _txt_value=$2
  17. _info "Using DNS-01 Hurricane Electric hook"
  18. if [ -z "$HE_Username" ] || [ -z "$HE_Password" ]; then
  19. HE_Username=
  20. HE_Password=
  21. _err "No auth details provided. Please set user credentials using the \$HE_Username and \$HE_Password envoronment variables."
  22. return 1
  23. fi
  24. _saveaccountconf HE_Username "$HE_Username"
  25. _saveaccountconf HE_Password "$HE_Password"
  26. # fills in the $_zone_id
  27. _find_zone "$_full_domain" || return 1
  28. _debug "Zone id \"$_zone_id\" will be used."
  29. body="email=${HE_Username}&pass=${HE_Password}"
  30. body="$body&account="
  31. body="$body&account="
  32. body="$body&menu=edit_zone"
  33. body="$body&Type=TXT"
  34. body="$body&hosted_dns_zoneid=$_zone_id"
  35. body="$body&hosted_dns_recordid="
  36. body="$body&hosted_dns_editzone=1"
  37. body="$body&Priority="
  38. body="$body&Name=$_full_domain"
  39. body="$body&Content=$_txt_value"
  40. body="$body&TTL=300"
  41. body="$body&hosted_dns_editrecord=Submit"
  42. response="$(_post "$body" "https://dns.he.net/")"
  43. _debug2 response "$response"
  44. }
  45. #-- dns_he_rm() - Remove TXT record ------------------------------------
  46. # Usage: dns_he_rm _acme-challenge.subdomain.domain.com "XyZ123..."
  47. dns_he_rm() {
  48. _full_domain=$1
  49. _txt_value=$2
  50. _info "Cleaning up after DNS-01 Hurricane Electric hook"
  51. # fills in the $_zone_id
  52. _find_zone "$_full_domain" || return 1
  53. _debug "Zone id \"$_zone_id\" will be used."
  54. # Find the record id to clean
  55. body="email=${HE_Username}&pass=${HE_Password}"
  56. body="$body&hosted_dns_zoneid=$_zone_id"
  57. body="$body&menu=edit_zone"
  58. body="$body&hosted_dns_editzone="
  59. _record_id=$(_post "$body" "https://dns.he.net/" \
  60. | tr -d '\n' \
  61. | _egrep_o "data=\"&quot;${_txt_value}&quot;([^>]+>){6}[^<]+<[^;]+;deleteRecord\('[0-9]+','${_full_domain}','TXT'\)" \
  62. | _egrep_o "[0-9]+','${_full_domain}','TXT'\)$" \
  63. | _egrep_o "^[0-9]+"
  64. )
  65. # The series of egreps above could have been done a bit shorter but
  66. # I wanted to double-check whether it's the correct record (in case
  67. # HE changes their website somehow).
  68. # Remove the record
  69. body="email=${HE_Username}&pass=${HE_Password}"
  70. body="$body&menu=edit_zone"
  71. body="$body&hosted_dns_zoneid=$_zone_id"
  72. body="$body&hosted_dns_recordid=$_record_id"
  73. body="$body&hosted_dns_editzone=1"
  74. body="$body&hosted_dns_delrecord=1"
  75. body="$body&hosted_dns_delconfirm=delete"
  76. body="$body&hosted_dns_editzone=1"
  77. _post "$body" "https://dns.he.net/" \
  78. | grep '<div id="dns_status" onClick="hideThis(this);">Successfully removed record.</div>' \
  79. >/dev/null
  80. if [ $? -eq 0 ]; then
  81. _info "Record removed successfuly."
  82. else
  83. _err \
  84. "Could not clean (remove) up the record. Please go to HE" \
  85. "administration interface and clean it by hand."
  86. fi
  87. }
  88. ########################## PRIVATE FUNCTIONS ###########################
  89. #-- _find_zone() -------------------------------------------------------
  90. # Returns the most specific zone found in administration interface.
  91. #
  92. # Example:
  93. #
  94. # _find_zone first.second.third.co.uk
  95. #
  96. # ... will return the first zone that exists in admin out of these:
  97. # - "first.second.third.co.uk"
  98. # - "second.third.co.uk"
  99. # - "third.co.uk"
  100. # - "co.uk" <-- unlikely
  101. # - "uk" <-'
  102. #
  103. # (another approach would be something like this:
  104. # https://github.com/hlandau/acme/blob/master/_doc/dns.hook
  105. # - that's better if there are multiple pages. It's so much simpler.
  106. # )
  107. _find_zone() {
  108. _domain="$1"
  109. body="email=${HE_Username}&pass=${HE_Password}"
  110. _matches=$(_post "$body" "https://dns.he.net/" \
  111. | _egrep_o "delete_dom.*name=\"[^\"]+\" value=\"[0-9]+"
  112. )
  113. # Zone names and zone IDs are in same order
  114. _zone_ids=$(echo "$_matches" | cut -d '"' -f 5 --output-delimiter=":")
  115. _zone_names=$(echo "$_matches" | cut -d '"' -f 3 --output-delimiter=":")
  116. _debug2 "These are the zones on this HE account:"
  117. _debug2 "$_zone_names"
  118. _debug2 "And these are their respective IDs:"
  119. _debug2 "$_zone_ids"
  120. # Walk through all possible zone names
  121. _strip_counter=1
  122. while true; do
  123. _attempted_zone=$(echo "$_domain" | cut -d . -f ${_strip_counter}-)
  124. # All possible zone names have been tried
  125. if [ -z "$_attempted_zone" ]; then
  126. _err "No zone for domain \"$_domain\" found."
  127. return 1
  128. fi
  129. _debug "Looking for zone \"${_attempted_zone}\""
  130. _line_num=$(echo "$_zone_names" | _find_linenum "$_attempted_zone")
  131. if [ -n "$_line_num" ]; then
  132. _zone_id=$(echo "$_zone_ids" | sed "${_line_num}q;d")
  133. _debug "Found relevant zone \"$_attempted_zone\" with id" \
  134. "\"$_zone_id\" - will be used for domain \"$_domain\"."
  135. return 0
  136. fi
  137. _debug "Zone \"$_attempted_zone\" doesn't exist, let's try a less" \
  138. "specific zone."
  139. _strip_counter=$(_math "$_strip_counter" + 1)
  140. done
  141. }
  142. #-- _find_linenum()-----------------------------------------------------
  143. # Returns line number of line (supplied as an argument) in STDIN.
  144. #
  145. # Example:
  146. #
  147. # printf "a\nb\nc" | _find_linenum "b"
  148. #
  149. # This will:
  150. # - print out 2 because that's the line number of "b"
  151. # - return code 0 because it was found
  152. _find_linenum() {
  153. _current_line_num=0
  154. while read -r line; do
  155. _current_line_num=$(_math "$_current_line_num" + 1)
  156. if [ "$line" = "$1" ]; then
  157. # Found! Let's echo the line number and quit
  158. echo "$_current_line_num"
  159. return 0
  160. fi
  161. done
  162. # Not found
  163. return 1
  164. }
  165. # vim: et:ts=2:sw=2: