From 8aa13a581cd616e82c72deb7ecde9fa98c5c293f Mon Sep 17 00:00:00 2001 From: abooraja Date: Sat, 14 Feb 2026 14:14:36 +0330 Subject: [PATCH 1/5] Fix: Arvan Cloud DNS API - Correct domain lookup and record management - Fix _get_root() to fetch domains list first, then search within response - Fix _arvan_rest() GET request handling - Improve dns_arvan_rm() with multiple pattern matching - Enhance error handling and debug output - Fixes #6788 --- dnsapi/dns_arvan.sh | 106 ++++++++++++++++++++++++++++++++++++++------ 1 file changed, 92 insertions(+), 14 deletions(-) diff --git a/dnsapi/dns_arvan.sh b/dnsapi/dns_arvan.sh index cbe6dc1f..29588a4a 100644 --- a/dnsapi/dns_arvan.sh +++ b/dnsapi/dns_arvan.sh @@ -45,16 +45,18 @@ dns_arvan_add() { _info "response id is $response" _info "Added, OK" return 0 - elif _contains "$response" "Record Data is duplicate"; then + elif _contains "$response" "Record Data is duplicate" || _contains "$response" "duplicate" || _contains "$response" "already exists"; then _info "Already exists, OK" return 0 else _err "Add txt record error." + _debug "Response was: $response" return 1 fi + else + _err "Add txt record error." + return 1 fi - _err "Add txt record error." - return 0 } #Usage: fulldomain txtvalue @@ -79,20 +81,51 @@ dns_arvan_rm() { _debug "Getting txt records" _arvan_rest GET "${_domain}/dns-records" - if ! printf "%s" "$response" | grep \"current_page\":1 >/dev/null; then + if ! printf "%s" "$response" | grep -q "\"current_page\":1"; then _err "Error on Arvan Api" _err "Please create a github issue with debbug log" return 1 fi - _record_id=$(echo "$response" | _egrep_o ".\"id\":\"[^\"]*\",\"type\":\"txt\",\"name\":\"_acme-challenge\",\"value\":{\"text\":\"$txtvalue\"}" | cut -d : -f 2 | cut -d , -f 1 | tr -d \") + # جستجوی رکورد با نام و مقدار مشخص + # الگوهای مختلف برای پیدا کردن record_id + _record_id=$(echo "$response" | _egrep_o "\"id\":\"[^\"]*\"[^}]*\"type\":\"[Tt][Xx][Tt]\"[^}]*\"name\":\"$_sub_domain\"[^}]*\"value\":[^}]*\"text\":\"$txtvalue\"" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") + + # اگر با الگوی بالا پیدا نشد، الگوی دیگر را امتحان کنیم + if [ -z "$_record_id" ]; then + _record_id=$(echo "$response" | _egrep_o "\"name\":\"$_sub_domain\"[^}]*\"type\":\"[Tt][Xx][Tt]\"[^}]*\"value\":[^}]*\"text\":\"$txtvalue\"[^}]*\"id\":\"[^\"]*\"" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") + fi + + # اگر هنوز پیدا نشد، سعی کنیم از کل response استخراج کنیم + if [ -z "$_record_id" ]; then + # پیدا کردن بخش مربوط به این رکورد + record_block=$(echo "$response" | _egrep_o "\{[^}]*\"name\":\"$_sub_domain\"[^}]*\"type\":\"[Tt][Xx][Tt]\"[^}]*\"value\":[^}]*\"text\":\"$txtvalue\"[^}]*\}") + if [ -n "$record_block" ]; then + _record_id=$(echo "$record_block" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") + fi + fi + + if [ -z "$_record_id" ]; then + _err "Could not find record with name '$_sub_domain' and value '$txtvalue'" + _debug "Response was: $response" + return 1 + fi + + _debug "Found record_id: $_record_id" + if ! _arvan_rest "DELETE" "${_domain}/dns-records/${_record_id}"; then _err "Error on Arvan Api" return 1 fi _debug "$response" - _contains "$response" 'dns record deleted' - return 0 + + if _contains "$response" 'dns record deleted' || _contains "$response" 'deleted' || _contains "$response" 'success'; then + _info "Record deleted successfully" + return 0 + else + _err "Failed to delete record" + return 1 + fi } #################### Private functions below ################################## @@ -106,6 +139,18 @@ _get_root() { domain=$1 i=2 p=1 + + # ابتدا لیست دامنه‌ها را از API بگیریم + _debug "Getting list of domains from Arvan API" + if ! _arvan_rest GET ""; then + _err "Failed to get domains list from Arvan API" + return 1 + fi + + # ذخیره response برای استفاده در حلقه + domains_list="$response" + _debug2 "Domains list response: $domains_list" + while true; do h=$(printf "%s" "$domain" | cut -d . -f "$i"-100) _debug h "$h" @@ -114,17 +159,35 @@ _get_root() { return 1 fi - if ! _arvan_rest GET "$h"; then - return 1 - fi - if _contains "$response" "\"domain\":\"$h\""; then - _domain_id=$(echo "$response" | cut -d : -f 3 | cut -d , -f 1 | tr -d \") + # چک کردن وجود دامنه در لیست + if _contains "$domains_list" "\"domain\":\"$h\""; then + # استخراج domain_id از response + # فرمت ممکن: {"id":"xxx","domain":"mizekar.site",...} یا {"domain":"mizekar.site","id":"xxx",...} + _domain_id=$(echo "$domains_list" | _egrep_o "\"id\":\"[^\"]*\"[^}]*\"domain\":\"$h\"" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") + + # اگر با الگوی بالا پیدا نشد، الگوی دیگر را امتحان کنیم + if [ -z "$_domain_id" ]; then + _domain_id=$(echo "$domains_list" | _egrep_o "\"domain\":\"$h\"[^}]*\"id\":\"[^\"]*\"" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") + fi + + # اگر هنوز پیدا نشد، سعی کنیم از کل response استخراج کنیم + if [ -z "$_domain_id" ]; then + # پیدا کردن بخش مربوط به این دامنه + domain_block=$(echo "$domains_list" | _egrep_o "\{[^}]*\"domain\":\"$h\"[^}]*\}") + if [ -n "$domain_block" ]; then + _domain_id=$(echo "$domain_block" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") + fi + fi + if [ "$_domain_id" ]; then _sub_domain=$(printf "%s" "$domain" | cut -d . -f 1-"$p") _domain=$h + _debug "Found domain: $_domain, sub_domain: $_sub_domain, domain_id: $_domain_id" return 0 + else + _err "Could not extract domain_id for domain: $h" + return 1 fi - return 1 fi p=$i i=$(_math "$i" + 1) @@ -150,7 +213,22 @@ _arvan_rest() { _debug data "$data" response="$(_post "$data" "$ARVAN_API_URL/$ep" "" "$mtd")" else - response="$(_get "$ARVAN_API_URL/$ep$data")" + # برای GET request + if [ -n "$ep" ]; then + # اگر ep مشخص شده، به endpoint خاص درخواست می‌زنیم + response="$(_get "$ARVAN_API_URL/$ep")" + else + # اگر ep خالی است، لیست دامنه‌ها را می‌گیریم + response="$(_get "$ARVAN_API_URL")" + fi + fi + + # چک کردن موفقیت درخواست + if [ "$?" != "0" ]; then + _err "Error on Arvan API request" + return 1 fi + + _debug2 response "$response" return 0 } From 49e9f9d165a5b6dcfcfc438a6d7ac5bcfd5c5966 Mon Sep 17 00:00:00 2001 From: abooraja Date: Sat, 14 Feb 2026 14:17:04 +0330 Subject: [PATCH 2/5] fix issue number in code --- dnsapi/dns_arvan.sh | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/dnsapi/dns_arvan.sh b/dnsapi/dns_arvan.sh index 29588a4a..e19e4f15 100644 --- a/dnsapi/dns_arvan.sh +++ b/dnsapi/dns_arvan.sh @@ -5,8 +5,8 @@ Site: ArvanCloud.ir Docs: github.com/acmesh-official/acme.sh/wiki/dnsapi2#dns_arvan Options: Arvan_Token API Token -Issues: github.com/acmesh-official/acme.sh/issues/2796 -Author: Vahid Fardi +Issues: github.com/acmesh-official/acme.sh/issues/6788 +Author: Abolfazl Rajabpour ( abooraja ) ( mizekar.com ) ' ARVAN_API_URL="https://napi.arvancloud.ir/cdn/4.0/domains" From 97b1b74f673ebf834a83df0f4d2ddf35d86d174e Mon Sep 17 00:00:00 2001 From: abooraja Date: Sat, 14 Feb 2026 14:26:36 +0330 Subject: [PATCH 3/5] Fix shellcheck SC2181: Check exit code directly --- dnsapi/dns_arvan.sh | 26 ++++++++++++++++---------- 1 file changed, 16 insertions(+), 10 deletions(-) diff --git a/dnsapi/dns_arvan.sh b/dnsapi/dns_arvan.sh index e19e4f15..81a2632d 100644 --- a/dnsapi/dns_arvan.sh +++ b/dnsapi/dns_arvan.sh @@ -206,29 +206,35 @@ _arvan_rest() { if [ "$mtd" = "DELETE" ]; then #DELETE Request shouldn't have Content-Type _debug data "$data" - response="$(_post "$data" "$ARVAN_API_URL/$ep" "" "$mtd")" + if ! response="$(_post "$data" "$ARVAN_API_URL/$ep" "" "$mtd")"; then + _err "Error on Arvan API request" + return 1 + fi elif [ "$mtd" = "POST" ]; then export _H2="Content-Type: application/json" export _H3="Accept: application/json" _debug data "$data" - response="$(_post "$data" "$ARVAN_API_URL/$ep" "" "$mtd")" + if ! response="$(_post "$data" "$ARVAN_API_URL/$ep" "" "$mtd")"; then + _err "Error on Arvan API request" + return 1 + fi else # برای GET request if [ -n "$ep" ]; then # اگر ep مشخص شده، به endpoint خاص درخواست می‌زنیم - response="$(_get "$ARVAN_API_URL/$ep")" + if ! response="$(_get "$ARVAN_API_URL/$ep")"; then + _err "Error on Arvan API request" + return 1 + fi else # اگر ep خالی است، لیست دامنه‌ها را می‌گیریم - response="$(_get "$ARVAN_API_URL")" + if ! response="$(_get "$ARVAN_API_URL")"; then + _err "Error on Arvan API request" + return 1 + fi fi fi - # چک کردن موفقیت درخواست - if [ "$?" != "0" ]; then - _err "Error on Arvan API request" - return 1 - fi - _debug2 response "$response" return 0 } From 2f18497f916ea82f4b28d732b152fa6bbd2668d0 Mon Sep 17 00:00:00 2001 From: abooraja Date: Sat, 14 Feb 2026 14:27:09 +0330 Subject: [PATCH 4/5] english comment --- dnsapi/dns_arvan.sh | 32 ++++++++++++++++---------------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/dnsapi/dns_arvan.sh b/dnsapi/dns_arvan.sh index 81a2632d..d4ebb64d 100644 --- a/dnsapi/dns_arvan.sh +++ b/dnsapi/dns_arvan.sh @@ -87,18 +87,18 @@ dns_arvan_rm() { return 1 fi - # جستجوی رکورد با نام و مقدار مشخص - # الگوهای مختلف برای پیدا کردن record_id + # Search for record with specified name and value + # Multiple patterns to find record_id _record_id=$(echo "$response" | _egrep_o "\"id\":\"[^\"]*\"[^}]*\"type\":\"[Tt][Xx][Tt]\"[^}]*\"name\":\"$_sub_domain\"[^}]*\"value\":[^}]*\"text\":\"$txtvalue\"" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") - # اگر با الگوی بالا پیدا نشد، الگوی دیگر را امتحان کنیم + # If not found with above pattern, try another pattern if [ -z "$_record_id" ]; then _record_id=$(echo "$response" | _egrep_o "\"name\":\"$_sub_domain\"[^}]*\"type\":\"[Tt][Xx][Tt]\"[^}]*\"value\":[^}]*\"text\":\"$txtvalue\"[^}]*\"id\":\"[^\"]*\"" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") fi - # اگر هنوز پیدا نشد، سعی کنیم از کل response استخراج کنیم + # If still not found, try to extract from entire response if [ -z "$_record_id" ]; then - # پیدا کردن بخش مربوط به این رکورد + # Find the block related to this record record_block=$(echo "$response" | _egrep_o "\{[^}]*\"name\":\"$_sub_domain\"[^}]*\"type\":\"[Tt][Xx][Tt]\"[^}]*\"value\":[^}]*\"text\":\"$txtvalue\"[^}]*\}") if [ -n "$record_block" ]; then _record_id=$(echo "$record_block" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") @@ -140,14 +140,14 @@ _get_root() { i=2 p=1 - # ابتدا لیست دامنه‌ها را از API بگیریم + # First, get the list of domains from API _debug "Getting list of domains from Arvan API" if ! _arvan_rest GET ""; then _err "Failed to get domains list from Arvan API" return 1 fi - # ذخیره response برای استفاده در حلقه + # Save response for use in loop domains_list="$response" _debug2 "Domains list response: $domains_list" @@ -159,20 +159,20 @@ _get_root() { return 1 fi - # چک کردن وجود دامنه در لیست + # Check if domain exists in list if _contains "$domains_list" "\"domain\":\"$h\""; then - # استخراج domain_id از response - # فرمت ممکن: {"id":"xxx","domain":"mizekar.site",...} یا {"domain":"mizekar.site","id":"xxx",...} + # Extract domain_id from response + # Possible formats: {"id":"xxx","domain":"mizekar.site",...} or {"domain":"mizekar.site","id":"xxx",...} _domain_id=$(echo "$domains_list" | _egrep_o "\"id\":\"[^\"]*\"[^}]*\"domain\":\"$h\"" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") - # اگر با الگوی بالا پیدا نشد، الگوی دیگر را امتحان کنیم + # If not found with above pattern, try another pattern if [ -z "$_domain_id" ]; then _domain_id=$(echo "$domains_list" | _egrep_o "\"domain\":\"$h\"[^}]*\"id\":\"[^\"]*\"" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") fi - # اگر هنوز پیدا نشد، سعی کنیم از کل response استخراج کنیم + # If still not found, try to extract from entire response if [ -z "$_domain_id" ]; then - # پیدا کردن بخش مربوط به این دامنه + # Find the block related to this domain domain_block=$(echo "$domains_list" | _egrep_o "\{[^}]*\"domain\":\"$h\"[^}]*\}") if [ -n "$domain_block" ]; then _domain_id=$(echo "$domain_block" | _egrep_o "\"id\":\"[^\"]*\"" | cut -d : -f 2 | tr -d \") @@ -219,15 +219,15 @@ _arvan_rest() { return 1 fi else - # برای GET request + # For GET request if [ -n "$ep" ]; then - # اگر ep مشخص شده، به endpoint خاص درخواست می‌زنیم + # If ep is specified, make request to specific endpoint if ! response="$(_get "$ARVAN_API_URL/$ep")"; then _err "Error on Arvan API request" return 1 fi else - # اگر ep خالی است، لیست دامنه‌ها را می‌گیریم + # If ep is empty, get the list of domains if ! response="$(_get "$ARVAN_API_URL")"; then _err "Error on Arvan API request" return 1 From 4d2a4ce4deb6a5ceb9d9fcd55a736286b9a80385 Mon Sep 17 00:00:00 2001 From: abooraja Date: Sat, 14 Feb 2026 14:27:43 +0330 Subject: [PATCH 5/5] fix author --- dnsapi/dns_arvan.sh | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/dnsapi/dns_arvan.sh b/dnsapi/dns_arvan.sh index d4ebb64d..dd1eb98d 100644 --- a/dnsapi/dns_arvan.sh +++ b/dnsapi/dns_arvan.sh @@ -6,7 +6,7 @@ Docs: github.com/acmesh-official/acme.sh/wiki/dnsapi2#dns_arvan Options: Arvan_Token API Token Issues: github.com/acmesh-official/acme.sh/issues/6788 -Author: Abolfazl Rajabpour ( abooraja ) ( mizekar.com ) +Author: Abolfazl Rajabpour ( abooraja ) ' ARVAN_API_URL="https://napi.arvancloud.ir/cdn/4.0/domains"