198 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
			
		
		
	
	
			198 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
#!/usr/bin/env sh
 | 
						|
 | 
						|
# Author: Radek Sprta <sprta@vshosting.cz>
 | 
						|
 | 
						|
#CLOUDDNS_EMAIL=XXXXX
 | 
						|
#CLOUDDNS_PASSWORD="YYYYYYYYY"
 | 
						|
#CLOUDDNS_CLIENT_ID=XXXXX
 | 
						|
 | 
						|
CLOUDDNS_API='https://admin.vshosting.cloud/clouddns'
 | 
						|
CLOUDDNS_LOGIN_API='https://admin.vshosting.cloud/api/public/auth/login'
 | 
						|
 | 
						|
########  Public functions #####################
 | 
						|
 | 
						|
# Usage: add _acme-challenge.www.domain.com "XKrxpRBosdIKFzxW_CT3KLZNf6q0HG9i01zxXp5CPBs"
 | 
						|
dns_clouddns_add() {
 | 
						|
  fulldomain=$1
 | 
						|
  txtvalue=$2
 | 
						|
  _debug "fulldomain" "$fulldomain"
 | 
						|
 | 
						|
  CLOUDDNS_CLIENT_ID="${CLOUDDNS_CLIENT_ID:-$(_readaccountconf_mutable CLOUDDNS_CLIENT_ID)}"
 | 
						|
  CLOUDDNS_EMAIL="${CLOUDDNS_EMAIL:-$(_readaccountconf_mutable CLOUDDNS_EMAIL)}"
 | 
						|
  CLOUDDNS_PASSWORD="${CLOUDDNS_PASSWORD:-$(_readaccountconf_mutable CLOUDDNS_PASSWORD)}"
 | 
						|
 | 
						|
  if [ -z "$CLOUDDNS_PASSWORD" ] || [ -z "$CLOUDDNS_EMAIL" ] || [ -z "$CLOUDDNS_CLIENT_ID" ]; then
 | 
						|
    CLOUDDNS_CLIENT_ID=""
 | 
						|
    CLOUDDNS_EMAIL=""
 | 
						|
    CLOUDDNS_PASSWORD=""
 | 
						|
    _err "You didn't specify a CloudDNS password, email and client ID yet."
 | 
						|
    return 1
 | 
						|
  fi
 | 
						|
  if ! _contains "$CLOUDDNS_EMAIL" "@"; then
 | 
						|
    _err "It seems that the CLOUDDNS_EMAIL=$CLOUDDNS_EMAIL is not a valid email address."
 | 
						|
    _err "Please check and retry."
 | 
						|
    return 1
 | 
						|
  fi
 | 
						|
  # Save CloudDNS client id, email and password to config file
 | 
						|
  _saveaccountconf_mutable CLOUDDNS_CLIENT_ID "$CLOUDDNS_CLIENT_ID"
 | 
						|
  _saveaccountconf_mutable CLOUDDNS_EMAIL "$CLOUDDNS_EMAIL"
 | 
						|
  _saveaccountconf_mutable CLOUDDNS_PASSWORD "$CLOUDDNS_PASSWORD"
 | 
						|
 | 
						|
  _debug "First detect the root zone"
 | 
						|
  if ! _get_root "$fulldomain"; then
 | 
						|
    _err "Invalid domain"
 | 
						|
    return 1
 | 
						|
  fi
 | 
						|
  _debug _domain_id "$_domain_id"
 | 
						|
  _debug _sub_domain "$_sub_domain"
 | 
						|
  _debug _domain "$_domain"
 | 
						|
 | 
						|
  # Add TXT record
 | 
						|
  data="{\"type\":\"TXT\",\"name\":\"$fulldomain.\",\"value\":\"$txtvalue\",\"domainId\":\"$_domain_id\"}"
 | 
						|
  if _clouddns_api POST "record-txt" "$data"; then
 | 
						|
    if _contains "$response" "$txtvalue"; then
 | 
						|
      _info "Added, OK"
 | 
						|
    elif _contains "$response" '"code":4136'; then
 | 
						|
      _info "Already exists, OK"
 | 
						|
    else
 | 
						|
      _err "Add TXT record error."
 | 
						|
      return 1
 | 
						|
    fi
 | 
						|
  fi
 | 
						|
 | 
						|
  _debug "Publishing record changes"
 | 
						|
  _clouddns_api PUT "domain/$_domain_id/publish" "{\"soaTtl\":300}"
 | 
						|
}
 | 
						|
 | 
						|
# Usage: rm _acme-challenge.www.domain.com
 | 
						|
dns_clouddns_rm() {
 | 
						|
  fulldomain=$1
 | 
						|
  _debug "fulldomain" "$fulldomain"
 | 
						|
 | 
						|
  CLOUDDNS_CLIENT_ID="${CLOUDDNS_CLIENT_ID:-$(_readaccountconf_mutable CLOUDDNS_CLIENT_ID)}"
 | 
						|
  CLOUDDNS_EMAIL="${CLOUDDNS_EMAIL:-$(_readaccountconf_mutable CLOUDDNS_EMAIL)}"
 | 
						|
  CLOUDDNS_PASSWORD="${CLOUDDNS_PASSWORD:-$(_readaccountconf_mutable CLOUDDNS_PASSWORD)}"
 | 
						|
 | 
						|
  _debug "First detect the root zone"
 | 
						|
  if ! _get_root "$fulldomain"; then
 | 
						|
    _err "Invalid domain"
 | 
						|
    return 1
 | 
						|
  fi
 | 
						|
  _debug _domain_id "$_domain_id"
 | 
						|
  _debug _sub_domain "$_sub_domain"
 | 
						|
  _debug _domain "$_domain"
 | 
						|
 | 
						|
  # Get record ID
 | 
						|
  _clouddns_api GET "domain/$_domain_id"
 | 
						|
  if _contains "$response" "lastDomainRecordList"; then
 | 
						|
    re="\"lastDomainRecordList\".*\"id\":\"([^\"}]*)\"[^}]*\"name\":\"$fulldomain.\","
 | 
						|
    _last_domains=$(echo "$response" | _egrep_o "$re")
 | 
						|
    re2="\"id\":\"([^\"}]*)\"[^}]*\"name\":\"$fulldomain.\","
 | 
						|
    _record_id=$(echo "$_last_domains" | _egrep_o "$re2" | _head_n 1 | cut -d : -f 2 | cut -d , -f 1 | tr -d "\"")
 | 
						|
    _debug _record_id "$_record_id"
 | 
						|
  else
 | 
						|
    _err "Could not retrieve record ID"
 | 
						|
    return 1
 | 
						|
  fi
 | 
						|
 | 
						|
  _info "Removing record"
 | 
						|
  if _clouddns_api DELETE "record/$_record_id"; then
 | 
						|
    if _contains "$response" "\"error\":"; then
 | 
						|
      _err "Could not remove record"
 | 
						|
      return 1
 | 
						|
    fi
 | 
						|
  fi
 | 
						|
 | 
						|
  _debug "Publishing record changes"
 | 
						|
  _clouddns_api PUT "domain/$_domain_id/publish" "{\"soaTtl\":300}"
 | 
						|
}
 | 
						|
 | 
						|
####################  Private functions below ##################################
 | 
						|
 | 
						|
# Usage: _get_root _acme-challenge.www.domain.com
 | 
						|
# Returns:
 | 
						|
# _sub_domain=_acme-challenge.www
 | 
						|
# _domain=domain.com
 | 
						|
# _domain_id=sdjkglgdfewsdfg
 | 
						|
_get_root() {
 | 
						|
  domain=$1
 | 
						|
 | 
						|
  # Get domain root
 | 
						|
  data="{\"search\": [{\"name\": \"clientId\", \"operator\": \"eq\", \"value\": \"$CLOUDDNS_CLIENT_ID\"}]}"
 | 
						|
  _clouddns_api "POST" "domain/search" "$data"
 | 
						|
  domain_slice="$domain"
 | 
						|
  while [ -z "$domain_root" ]; do
 | 
						|
    if _contains "$response" "\"domainName\":\"$domain_slice\.\""; then
 | 
						|
      domain_root="$domain_slice"
 | 
						|
      _debug domain_root "$domain_root"
 | 
						|
    fi
 | 
						|
    domain_slice="$(echo "$domain_slice" | cut -d . -f 2-)"
 | 
						|
  done
 | 
						|
 | 
						|
  # Get domain id
 | 
						|
  data="{\"search\": [{\"name\": \"clientId\", \"operator\": \"eq\", \"value\": \"$CLOUDDNS_CLIENT_ID\"}, \
 | 
						|
      {\"name\": \"domainName\", \"operator\": \"eq\", \"value\": \"$domain_root.\"}]}"
 | 
						|
  _clouddns_api "POST" "domain/search" "$data"
 | 
						|
  if _contains "$response" "\"id\":\""; then
 | 
						|
    re='domainType\":\"[^\"]*\",\"id\":\"([^\"]*)\",' # Match domain id
 | 
						|
    _domain_id=$(echo "$response" | _egrep_o "$re" | _head_n 1 | cut -d : -f 3 | tr -d "\",")
 | 
						|
    if [ "$_domain_id" ]; then
 | 
						|
      _sub_domain=$(printf "%s" "$domain" | sed "s/.$domain_root//")
 | 
						|
      _domain="$domain_root"
 | 
						|
      return 0
 | 
						|
    fi
 | 
						|
    _err 'Domain name not found on your CloudDNS account'
 | 
						|
    return 1
 | 
						|
  fi
 | 
						|
  return 1
 | 
						|
}
 | 
						|
 | 
						|
# Usage: _clouddns_api GET domain/search '{"data": "value"}'
 | 
						|
# Returns:
 | 
						|
#  response='{"message": "api response"}'
 | 
						|
_clouddns_api() {
 | 
						|
  method=$1
 | 
						|
  endpoint="$2"
 | 
						|
  data="$3"
 | 
						|
  _debug endpoint "$endpoint"
 | 
						|
 | 
						|
  if [ -z "$CLOUDDNS_TOKEN" ]; then
 | 
						|
    _clouddns_login
 | 
						|
  fi
 | 
						|
  _debug CLOUDDNS_TOKEN "$CLOUDDNS_TOKEN"
 | 
						|
 | 
						|
  export _H1="Content-Type: application/json"
 | 
						|
  export _H2="Authorization: Bearer $CLOUDDNS_TOKEN"
 | 
						|
 | 
						|
  if [ "$method" != "GET" ]; then
 | 
						|
    _debug data "$data"
 | 
						|
    response="$(_post "$data" "$CLOUDDNS_API/$endpoint" "" "$method" | tr -d '\t\r\n ')"
 | 
						|
  else
 | 
						|
    response="$(_get "$CLOUDDNS_API/$endpoint" | tr -d '\t\r\n ')"
 | 
						|
  fi
 | 
						|
 | 
						|
  # shellcheck disable=SC2181
 | 
						|
  if [ "$?" != "0" ]; then
 | 
						|
    _err "Error $endpoint"
 | 
						|
    return 1
 | 
						|
  fi
 | 
						|
  _debug2 response "$response"
 | 
						|
  return 0
 | 
						|
}
 | 
						|
 | 
						|
# Returns:
 | 
						|
#  CLOUDDNS_TOKEN=dslfje2rj23l
 | 
						|
_clouddns_login() {
 | 
						|
  login_data="{\"email\": \"$CLOUDDNS_EMAIL\", \"password\": \"$CLOUDDNS_PASSWORD\"}"
 | 
						|
  response="$(_post "$login_data" "$CLOUDDNS_LOGIN_API" "" "POST" "Content-Type: application/json")"
 | 
						|
 | 
						|
  if _contains "$response" "\"accessToken\":\""; then
 | 
						|
    CLOUDDNS_TOKEN=$(echo "$response" | _egrep_o "\"accessToken\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \")
 | 
						|
    export CLOUDDNS_TOKEN
 | 
						|
  else
 | 
						|
    echo 'Could not get CloudDNS access token; check your credentials'
 | 
						|
    return 1
 | 
						|
  fi
 | 
						|
  return 0
 | 
						|
}
 |