6b7b5caf54
Instead of using comments declare info in a special variable. Then the variable can be used to print the DNS API provider usage. The usage can be parsed on UI and show all needed inputs for options. The info is stored in plain string that it's both human-readable and easy to parse: dns_example_info='API name An extended description. Multiline. Domains: list of alternative domains to find Site: the dns provider website e.g. example.com Docs: Link to ACME.sh wiki for the provider Options: VARIABLE1 Title for the option1. VARIABLE2 Title for the option2. Default "default value". VARIABLE3 Title for the option3. Description to show on UI. Optional. Issues: Link to a support ticket on https://github.com/acmesh-official/acme.sh Author: First Lastname <authoremail@example.com>, Another Author <https://github.com/example>; ' Here: VARIABLE1 will be required. VARIABLE2 will be required too but will be populated with a "default value". VARIABLE3 is optional and can be empty. A DNS provider may have alternative options like CloudFlare may use API KEY or API Token. You can use a second section OptionsAlt: section. Some providers may have alternative names or domains e.g. Aliyun and AlibabaCloud. Add them to Domains: section. Signed-off-by: Sergey Ponomarev <stokito@gmail.com>
177 lines
5.1 KiB
Bash
Executable File
177 lines
5.1 KiB
Bash
Executable File
#!/usr/bin/env sh
|
|
# shellcheck disable=SC2034
|
|
dns_gcloud_info='Google Cloud DNS
|
|
Site: Cloud.Google.com/dns
|
|
Docs: github.com/acmesh-official/acme.sh/wiki/dnsapi#dns_gcloud
|
|
Options:
|
|
CLOUDSDK_ACTIVE_CONFIG_NAME Active config name. E.g. "default"
|
|
Author: Janos Lenart <janos@lenart.io>
|
|
'
|
|
|
|
######## Public functions #####################
|
|
|
|
# Usage: dns_gcloud_add _acme-challenge.www.domain.com "XKrxpRBosdIKFzxW_CT3KLZNf6q0HG9i01zxXp5CPBs"
|
|
dns_gcloud_add() {
|
|
fulldomain=$1
|
|
txtvalue=$2
|
|
_info "Using gcloud"
|
|
_debug fulldomain "$fulldomain"
|
|
_debug txtvalue "$txtvalue"
|
|
|
|
_dns_gcloud_find_zone || return $?
|
|
|
|
# Add an extra RR
|
|
_dns_gcloud_start_tr || return $?
|
|
_dns_gcloud_get_rrdatas || return $?
|
|
echo "$rrdatas" | _dns_gcloud_remove_rrs || return $?
|
|
printf "%s\n%s\n" "$rrdatas" "\"$txtvalue\"" | grep -v '^$' | _dns_gcloud_add_rrs || return $?
|
|
_dns_gcloud_execute_tr || return $?
|
|
|
|
_info "$fulldomain record added"
|
|
}
|
|
|
|
# Usage: dns_gcloud_rm _acme-challenge.www.domain.com "XKrxpRBosdIKFzxW_CT3KLZNf6q0HG9i01zxXp5CPBs"
|
|
# Remove the txt record after validation.
|
|
dns_gcloud_rm() {
|
|
fulldomain=$1
|
|
txtvalue=$2
|
|
_info "Using gcloud"
|
|
_debug fulldomain "$fulldomain"
|
|
_debug txtvalue "$txtvalue"
|
|
|
|
_dns_gcloud_find_zone || return $?
|
|
|
|
# Remove one RR
|
|
_dns_gcloud_start_tr || return $?
|
|
_dns_gcloud_get_rrdatas || return $?
|
|
echo "$rrdatas" | _dns_gcloud_remove_rrs || return $?
|
|
echo "$rrdatas" | grep -F -v -- "\"$txtvalue\"" | _dns_gcloud_add_rrs || return $?
|
|
_dns_gcloud_execute_tr || return $?
|
|
|
|
_info "$fulldomain record removed"
|
|
}
|
|
|
|
#################### Private functions below ##################################
|
|
|
|
_dns_gcloud_start_tr() {
|
|
if ! trd=$(mktemp -d); then
|
|
_err "_dns_gcloud_start_tr: failed to create temporary directory"
|
|
return 1
|
|
fi
|
|
tr="$trd/tr.yaml"
|
|
_debug tr "$tr"
|
|
|
|
if ! gcloud dns record-sets transaction start \
|
|
--transaction-file="$tr" \
|
|
--zone="$managedZone"; then
|
|
rm -r "$trd"
|
|
_err "_dns_gcloud_start_tr: failed to execute transaction"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
_dns_gcloud_execute_tr() {
|
|
if ! gcloud dns record-sets transaction execute \
|
|
--transaction-file="$tr" \
|
|
--zone="$managedZone"; then
|
|
_debug tr "$(cat "$tr")"
|
|
rm -r "$trd"
|
|
_err "_dns_gcloud_execute_tr: failed to execute transaction"
|
|
return 1
|
|
fi
|
|
rm -r "$trd"
|
|
|
|
for i in $(seq 1 120); do
|
|
if gcloud dns record-sets changes list \
|
|
--zone="$managedZone" \
|
|
--filter='status != done' |
|
|
grep -q '^.*'; then
|
|
_info "_dns_gcloud_execute_tr: waiting for transaction to be comitted ($i/120)..."
|
|
sleep 5
|
|
else
|
|
return 0
|
|
fi
|
|
done
|
|
|
|
_err "_dns_gcloud_execute_tr: transaction is still pending after 10 minutes"
|
|
rm -r "$trd"
|
|
return 1
|
|
}
|
|
|
|
_dns_gcloud_remove_rrs() {
|
|
if ! xargs -r gcloud dns record-sets transaction remove \
|
|
--name="$fulldomain." \
|
|
--ttl="$ttl" \
|
|
--type=TXT \
|
|
--zone="$managedZone" \
|
|
--transaction-file="$tr" --; then
|
|
_debug tr "$(cat "$tr")"
|
|
rm -r "$trd"
|
|
_err "_dns_gcloud_remove_rrs: failed to remove RRs"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
_dns_gcloud_add_rrs() {
|
|
ttl=60
|
|
if ! xargs -r gcloud dns record-sets transaction add \
|
|
--name="$fulldomain." \
|
|
--ttl="$ttl" \
|
|
--type=TXT \
|
|
--zone="$managedZone" \
|
|
--transaction-file="$tr" --; then
|
|
_debug tr "$(cat "$tr")"
|
|
rm -r "$trd"
|
|
_err "_dns_gcloud_add_rrs: failed to add RRs"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
_dns_gcloud_find_zone() {
|
|
# Prepare a filter that matches zones that are suiteable for this entry.
|
|
# For example, _acme-challenge.something.domain.com might need to go into something.domain.com or domain.com;
|
|
# this function finds the longest postfix that has a managed zone.
|
|
part="$fulldomain"
|
|
filter="dnsName=( "
|
|
while [ "$part" != "" ]; do
|
|
filter="$filter$part. "
|
|
part="$(echo "$part" | sed 's/[^.]*\.*//')"
|
|
done
|
|
filter="$filter) AND visibility=public"
|
|
_debug filter "$filter"
|
|
|
|
# List domains and find the zone with the deepest sub-domain (in case of some levels of delegation)
|
|
if ! match=$(gcloud dns managed-zones list \
|
|
--format="value(name, dnsName)" \
|
|
--filter="$filter" |
|
|
while read -r dnsName name; do
|
|
printf "%s\t%s\t%s\n" "$(echo "$name" | awk -F"." '{print NF-1}')" "$dnsName" "$name"
|
|
done |
|
|
sort -n -r | _head_n 1 | cut -f2,3 | grep '^.*'); then
|
|
_err "_dns_gcloud_find_zone: Can't find a matching managed zone! Perhaps wrong project or gcloud credentials?"
|
|
return 1
|
|
fi
|
|
|
|
dnsName=$(echo "$match" | cut -f2)
|
|
_debug dnsName "$dnsName"
|
|
managedZone=$(echo "$match" | cut -f1)
|
|
_debug managedZone "$managedZone"
|
|
}
|
|
|
|
_dns_gcloud_get_rrdatas() {
|
|
if ! rrdatas=$(gcloud dns record-sets list \
|
|
--zone="$managedZone" \
|
|
--name="$fulldomain." \
|
|
--type=TXT \
|
|
--format="value(ttl,rrdatas)"); then
|
|
_err "_dns_gcloud_get_rrdatas: Failed to list record-sets"
|
|
rm -r "$trd"
|
|
return 1
|
|
fi
|
|
ttl=$(echo "$rrdatas" | cut -f1)
|
|
# starting with version 353.0.0 gcloud seems to
|
|
# separate records with a semicolon instead of commas
|
|
# see also https://cloud.google.com/sdk/docs/release-notes#35300_2021-08-17
|
|
rrdatas=$(echo "$rrdatas" | cut -f2 | sed 's/"[,;]"/"\n"/g')
|
|
}
|