123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200 |
- #!/usr/bin/env sh
- # shellcheck disable=SC2034
- dns_linode_v4_info='Linode.com
- Site: Linode.com
- Docs: github.com/acmesh-official/acme.sh/wiki/dnsapi#dns_linode_v4
- Options:
- LINODE_V4_API_KEY API Key
- Author: Philipp Grosswiler <[email protected]>, Aaron W. Swenson <[email protected]>
- '
- LINODE_V4_API_URL="https://api.linode.com/v4/domains"
- ######## Public functions #####################
- #Usage: dns_linode_add _acme-challenge.www.domain.com "XKrxpRBosdIKFzxW_CT3KLZNf6q0HG9i01zxXp5CPBs"
- dns_linode_v4_add() {
- fulldomain="${1}"
- txtvalue="${2}"
- if ! _Linode_API; then
- return 1
- fi
- _info "Using Linode"
- _debug "Calling: dns_linode_add() '${fulldomain}' '${txtvalue}'"
- _debug "First detect the root zone"
- if ! _get_root "$fulldomain"; then
- _err "Domain does not exist."
- return 1
- fi
- _debug _domain_id "$_domain_id"
- _debug _sub_domain "$_sub_domain"
- _debug _domain "$_domain"
- _payload="{
- \"type\": \"TXT\",
- \"name\": \"$_sub_domain\",
- \"target\": \"$txtvalue\",
- \"ttl_sec\": 300
- }"
- if _rest POST "/$_domain_id/records" "$_payload" && [ -n "$response" ]; then
- _resource_id=$(printf "%s\n" "$response" | _egrep_o "\"id\": *[0-9]+" | cut -d : -f 2 | tr -d " " | _head_n 1)
- _debug _resource_id "$_resource_id"
- if [ -z "$_resource_id" ]; then
- _err "Error adding the domain resource."
- return 1
- fi
- _info "Domain resource successfully added."
- return 0
- fi
- return 1
- }
- #Usage: dns_linode_rm _acme-challenge.www.domain.com
- dns_linode_v4_rm() {
- fulldomain="${1}"
- if ! _Linode_API; then
- return 1
- fi
- _info "Using Linode"
- _debug "Calling: dns_linode_rm() '${fulldomain}'"
- _debug "First detect the root zone"
- if ! _get_root "$fulldomain"; then
- _err "Domain does not exist."
- return 1
- fi
- _debug _domain_id "$_domain_id"
- _debug _sub_domain "$_sub_domain"
- _debug _domain "$_domain"
- if _H4="X-Filter: { \"type\": \"TXT\", \"name\": \"$_sub_domain\" }" _rest GET "/$_domain_id/records" && [ -n "$response" ]; then
- response="$(echo "$response" | tr -d "\n" | tr '{' "|" | sed 's/|/&{/g' | tr "|" "\n")"
- resource="$(echo "$response" | _egrep_o "\{.*\"name\": *\"$_sub_domain\".*}")"
- if [ "$resource" ]; then
- _resource_id=$(printf "%s\n" "$resource" | _egrep_o "\"id\": *[0-9]+" | _head_n 1 | cut -d : -f 2 | tr -d \ )
- if [ "$_resource_id" ]; then
- _debug _resource_id "$_resource_id"
- if _rest DELETE "/$_domain_id/records/$_resource_id" && [ -n "$response" ]; then
- # On 200/OK, empty set is returned. Check for error, if any.
- _error_response=$(printf "%s\n" "$response" | _egrep_o "\"errors\"" | cut -d : -f 2 | tr -d " " | _head_n 1)
- if [ -n "$_error_response" ]; then
- _err "Error deleting the domain resource: $_error_response"
- return 1
- fi
- _info "Domain resource successfully deleted."
- return 0
- fi
- fi
- return 1
- fi
- return 0
- fi
- return 1
- }
- #################### Private functions below ##################################
- _Linode_API() {
- LINODE_V4_API_KEY="${LINODE_V4_API_KEY:-$(_readaccountconf_mutable LINODE_V4_API_KEY)}"
- if [ -z "$LINODE_V4_API_KEY" ]; then
- LINODE_V4_API_KEY=""
- _err "You didn't specify the Linode v4 API key yet."
- _err "Please create your key and try again."
- return 1
- fi
- _saveaccountconf_mutable LINODE_V4_API_KEY "$LINODE_V4_API_KEY"
- }
- #################### Private functions below ##################################
- #_acme-challenge.www.domain.com
- #returns
- # _sub_domain=_acme-challenge.www
- # _domain=domain.com
- # _domain_id=12345
- _get_root() {
- full_host_str="$1"
- i=2
- p=1
- while true; do
- # loop through the received string (e.g. _acme-challenge.sub3.sub2.sub1.domain.tld),
- # starting from the lowest subdomain, and check if it's a hosted domain
- tst_hosted_domain=$(printf "%s" "$full_host_str" | cut -d . -f "$i"-100)
- _debug tst_hosted_domain "$tst_hosted_domain"
- if [ -z "$tst_hosted_domain" ]; then
- #not valid
- _err "Couldn't get domain from string '$full_host_str'."
- return 1
- fi
- _debug "Querying Linode APIv4 for hosted zone: $tst_hosted_domain"
- if _H4="X-Filter: {\"domain\":\"$tst_hosted_domain\"}" _rest GET; then
- _debug "Got response from API: $response"
- response="$(echo "$response" | tr -d "\n" | tr '{' "|" | sed 's/|/&{/g' | tr "|" "\n")"
- hostedzone="$(echo "$response" | _egrep_o "\{.*\"domain\": *\"$tst_hosted_domain\".*}")"
- if [ "$hostedzone" ]; then
- _domain_id=$(printf "%s\n" "$hostedzone" | _egrep_o "\"id\": *[0-9]+" | _head_n 1 | cut -d : -f 2 | tr -d \ )
- _debug "Found domain hosted on Linode DNS. Zone: $tst_hosted_domain, id: $_domain_id"
- if [ "$_domain_id" ]; then
- _sub_domain=$(printf "%s" "$full_host_str" | cut -d . -f 1-"$p")
- _domain=$tst_hosted_domain
- return 0
- fi
- return 1
- fi
- p=$i
- i=$(_math "$i" + 1)
- fi
- done
- return 1
- }
- #method method action data
- _rest() {
- mtd="$1"
- ep="$2"
- data="$3"
- _debug mtd "$mtd"
- _debug ep "$ep"
- export _H1="Accept: application/json"
- export _H2="Content-Type: application/json"
- export _H3="Authorization: Bearer $LINODE_V4_API_KEY"
- if [ "$mtd" != "GET" ]; then
- # both POST and DELETE.
- _debug data "$data"
- response="$(_post "$data" "$LINODE_V4_API_URL$ep" "" "$mtd")"
- else
- response="$(_get "$LINODE_V4_API_URL$ep$data")"
- fi
- if [ "$?" != "0" ]; then
- _err "error $ep"
- return 1
- fi
- _debug2 response "$response"
- return 0
- }
|