2017-02-28 11:58:04 +00:00
|
|
|
#!/usr/bin/env sh
|
2023-11-18 16:57:12 +00:00
|
|
|
# shellcheck disable=SC2034
|
|
|
|
dns_gandi_livedns_info='Gandi.net LiveDNS
|
|
|
|
Site: Gandi.net/domain/dns
|
|
|
|
Docs: github.com/acmesh-official/acme.sh/wiki/dnsapi#dns_gandi_livedns
|
|
|
|
Options:
|
|
|
|
GANDI_LIVEDNS_KEY API Key
|
|
|
|
Issues: github.com/fcrozat/acme.sh
|
|
|
|
Author: Frédéric Crozat <fcrozat@suse.com>, Dominik Röttsches <drott@google.com>
|
|
|
|
'
|
2017-02-28 11:58:04 +00:00
|
|
|
|
|
|
|
# Gandi LiveDNS v5 API
|
2023-09-11 13:05:12 +00:00
|
|
|
# https://api.gandi.net/docs/livedns/
|
|
|
|
# https://api.gandi.net/docs/authentication/ for token + apikey (deprecated) authentication
|
2017-02-28 11:58:04 +00:00
|
|
|
# currently under beta
|
2023-11-18 16:57:12 +00:00
|
|
|
|
2017-02-28 11:58:04 +00:00
|
|
|
######## Public functions #####################
|
|
|
|
|
2024-01-12 21:38:52 +00:00
|
|
|
GANDI_LIVEDNS_API="https://api.gandi.net/v5/livedns"
|
2017-02-28 11:58:04 +00:00
|
|
|
|
|
|
|
#Usage: dns_gandi_livedns_add _acme-challenge.www.domain.com "XKrxpRBosdIKFzxW_CT3KLZNf6q0HG9i01zxXp5CPBs"
|
|
|
|
dns_gandi_livedns_add() {
|
|
|
|
fulldomain=$1
|
|
|
|
txtvalue=$2
|
|
|
|
|
2023-09-12 13:54:44 +00:00
|
|
|
if [ -z "$GANDI_LIVEDNS_KEY" ] && [ -z "$GANDI_LIVEDNS_TOKEN" ]; then
|
2023-09-11 13:05:12 +00:00
|
|
|
_err "No Token or API key (deprecated) specified for Gandi LiveDNS."
|
|
|
|
_err "Create your token or key and export it as GANDI_LIVEDNS_KEY or GANDI_LIVEDNS_TOKEN respectively"
|
2017-02-28 11:58:04 +00:00
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
|
2023-09-11 13:05:12 +00:00
|
|
|
# Keep only one secret in configuration
|
|
|
|
if [ -n "$GANDI_LIVEDNS_TOKEN" ]; then
|
|
|
|
_saveaccountconf GANDI_LIVEDNS_TOKEN "$GANDI_LIVEDNS_TOKEN"
|
|
|
|
_clearaccountconf GANDI_LIVEDNS_KEY
|
|
|
|
elif [ -n "$GANDI_LIVEDNS_KEY" ]; then
|
|
|
|
_saveaccountconf GANDI_LIVEDNS_KEY "$GANDI_LIVEDNS_KEY"
|
|
|
|
_clearaccountconf GANDI_LIVEDNS_TOKEN
|
|
|
|
fi
|
|
|
|
|
2017-02-28 11:58:04 +00:00
|
|
|
_debug "First detect the root zone"
|
|
|
|
if ! _get_root "$fulldomain"; then
|
|
|
|
_err "invalid domain"
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
_debug fulldomain "$fulldomain"
|
|
|
|
_debug txtvalue "$txtvalue"
|
|
|
|
_debug domain "$_domain"
|
|
|
|
_debug sub_domain "$_sub_domain"
|
|
|
|
|
2018-12-27 22:45:51 +00:00
|
|
|
_dns_gandi_append_record "$_domain" "$_sub_domain" "$txtvalue"
|
2017-02-28 11:58:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#Usage: fulldomain txtvalue
|
|
|
|
#Remove the txt record after validation.
|
|
|
|
dns_gandi_livedns_rm() {
|
|
|
|
fulldomain=$1
|
|
|
|
txtvalue=$2
|
|
|
|
|
|
|
|
_debug "First detect the root zone"
|
|
|
|
if ! _get_root "$fulldomain"; then
|
|
|
|
_err "invalid domain"
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
|
|
|
|
_debug fulldomain "$fulldomain"
|
|
|
|
_debug domain "$_domain"
|
|
|
|
_debug sub_domain "$_sub_domain"
|
2018-12-27 22:45:51 +00:00
|
|
|
_debug txtvalue "$txtvalue"
|
2017-02-28 11:58:04 +00:00
|
|
|
|
2018-12-27 22:45:51 +00:00
|
|
|
if ! _dns_gandi_existing_rrset_values "$_domain" "$_sub_domain"; then
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
_new_rrset_values=$(echo "$_rrset_values" | sed "s/...$txtvalue...//g")
|
|
|
|
# Cleanup dangling commata.
|
|
|
|
_new_rrset_values=$(echo "$_new_rrset_values" | sed "s/, ,/ ,/g")
|
|
|
|
_new_rrset_values=$(echo "$_new_rrset_values" | sed "s/, *\]/\]/g")
|
|
|
|
_new_rrset_values=$(echo "$_new_rrset_values" | sed "s/\[ *,/\[/g")
|
|
|
|
_debug "New rrset_values" "$_new_rrset_values"
|
|
|
|
|
|
|
|
_gandi_livedns_rest PUT \
|
|
|
|
"domains/$_domain/records/$_sub_domain/TXT" \
|
2020-08-17 14:18:20 +00:00
|
|
|
"{\"rrset_ttl\": 300, \"rrset_values\": $_new_rrset_values}" &&
|
2024-01-12 21:38:52 +00:00
|
|
|
_contains "$response" '{"message":"DNS Record Created"}' &&
|
2020-08-17 14:18:20 +00:00
|
|
|
_info "Removing record $(__green "success")"
|
2017-02-28 11:58:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#################### Private functions below ##################################
|
|
|
|
#_acme-challenge.www.domain.com
|
|
|
|
#returns
|
|
|
|
# _sub_domain=_acme-challenge.www
|
|
|
|
# _domain=domain.com
|
|
|
|
_get_root() {
|
|
|
|
domain=$1
|
|
|
|
i=2
|
|
|
|
p=1
|
|
|
|
while true; do
|
|
|
|
h=$(printf "%s" "$domain" | cut -d . -f $i-100)
|
2017-02-28 12:56:11 +00:00
|
|
|
_debug h "$h"
|
2017-02-28 11:58:04 +00:00
|
|
|
if [ -z "$h" ]; then
|
|
|
|
#not valid
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
|
|
|
|
if ! _gandi_livedns_rest GET "domains/$h"; then
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
|
2017-02-28 12:39:23 +00:00
|
|
|
if _contains "$response" '"code": 401'; then
|
|
|
|
_err "$response"
|
|
|
|
return 1
|
|
|
|
elif _contains "$response" '"code": 404'; then
|
2017-02-28 11:58:04 +00:00
|
|
|
_debug "$h not found"
|
|
|
|
else
|
|
|
|
_sub_domain=$(printf "%s" "$domain" | cut -d . -f 1-$p)
|
|
|
|
_domain="$h"
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
p="$i"
|
|
|
|
i=$(_math "$i" + 1)
|
|
|
|
done
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2018-12-27 22:45:51 +00:00
|
|
|
_dns_gandi_append_record() {
|
|
|
|
domain=$1
|
|
|
|
sub_domain=$2
|
|
|
|
txtvalue=$3
|
|
|
|
|
|
|
|
if _dns_gandi_existing_rrset_values "$domain" "$sub_domain"; then
|
|
|
|
_debug "Appending new value"
|
|
|
|
_rrset_values=$(echo "$_rrset_values" | sed "s/\"]/\",\"$txtvalue\"]/")
|
|
|
|
else
|
|
|
|
_debug "Creating new record" "$_rrset_values"
|
|
|
|
_rrset_values="[\"$txtvalue\"]"
|
|
|
|
fi
|
|
|
|
_debug new_rrset_values "$_rrset_values"
|
|
|
|
_gandi_livedns_rest PUT "domains/$_domain/records/$sub_domain/TXT" \
|
2020-08-17 14:18:20 +00:00
|
|
|
"{\"rrset_ttl\": 300, \"rrset_values\": $_rrset_values}" &&
|
2024-01-12 21:38:52 +00:00
|
|
|
_contains "$response" '{"message":"DNS Record Created"}' &&
|
2020-08-17 14:18:20 +00:00
|
|
|
_info "Adding record $(__green "success")"
|
2018-12-27 22:45:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_dns_gandi_existing_rrset_values() {
|
|
|
|
domain=$1
|
|
|
|
sub_domain=$2
|
|
|
|
if ! _gandi_livedns_rest GET "domains/$domain/records/$sub_domain"; then
|
|
|
|
return 1
|
|
|
|
fi
|
2024-01-12 21:38:52 +00:00
|
|
|
if ! _contains "$response" '"rrset_type":"TXT"'; then
|
2018-12-27 22:45:51 +00:00
|
|
|
_debug "Does not have a _acme-challenge TXT record yet."
|
|
|
|
return 1
|
|
|
|
fi
|
2024-01-12 21:38:52 +00:00
|
|
|
if _contains "$response" '"rrset_values":\[\]'; then
|
2018-12-27 22:45:51 +00:00
|
|
|
_debug "Empty rrset_values for TXT record, no previous TXT record."
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
_debug "Already has TXT record."
|
2020-08-17 14:18:20 +00:00
|
|
|
_rrset_values=$(echo "$response" | _egrep_o 'rrset_values.*\[.*\]' |
|
|
|
|
_egrep_o '\[".*\"]')
|
2018-12-27 22:45:51 +00:00
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2017-02-28 11:58:04 +00:00
|
|
|
_gandi_livedns_rest() {
|
|
|
|
m=$1
|
|
|
|
ep="$2"
|
|
|
|
data="$3"
|
|
|
|
_debug "$ep"
|
|
|
|
|
|
|
|
export _H1="Content-Type: application/json"
|
2023-09-11 13:05:12 +00:00
|
|
|
|
|
|
|
if [ -n "$GANDI_LIVEDNS_TOKEN" ]; then
|
|
|
|
export _H2="Authorization: Bearer $GANDI_LIVEDNS_TOKEN"
|
|
|
|
else
|
2024-01-12 21:38:52 +00:00
|
|
|
export _H2="Authorization: Apikey $GANDI_LIVEDNS_KEY"
|
2023-09-11 13:05:12 +00:00
|
|
|
fi
|
2017-02-28 11:58:04 +00:00
|
|
|
|
2017-02-28 12:39:23 +00:00
|
|
|
if [ "$m" = "GET" ]; then
|
|
|
|
response="$(_get "$GANDI_LIVEDNS_API/$ep")"
|
|
|
|
else
|
2017-02-28 11:58:04 +00:00
|
|
|
_debug data "$data"
|
|
|
|
response="$(_post "$data" "$GANDI_LIVEDNS_API/$ep" "" "$m")"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "$?" != "0" ]; then
|
|
|
|
_err "error $ep"
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
_debug2 response "$response"
|
|
|
|
return 0
|
|
|
|
}
|