functions.sh 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. #!/bin/bash
  2. log_f() {
  3. if [[ ${2} == "no_nl" ]]; then
  4. echo -n "$(date) - ${1}"
  5. elif [[ ${2} == "no_date" ]]; then
  6. echo "${1}"
  7. elif [[ ${2} != "redis_only" ]]; then
  8. echo "$(date) - ${1}"
  9. fi
  10. if [[ ${3} == "b64" ]]; then
  11. redis-cli -h redis LPUSH ACME_LOG "{\"time\":\"$(date +%s)\",\"message\":\"base64,$(printf '%s' "${1}")\"}" > /dev/null
  12. else
  13. redis-cli -h redis LPUSH ACME_LOG "{\"time\":\"$(date +%s)\",\"message\":\"$(printf '%s' "${1}" | \
  14. tr '%&;$"[]{}-\r\n' ' ')\"}" > /dev/null
  15. fi
  16. }
  17. verify_hash_match(){
  18. CERT_HASH=$(openssl x509 -in "${1}" -noout -pubkey | openssl md5)
  19. KEY_HASH=$(openssl pkey -in "${2}" -pubout | openssl md5)
  20. if [[ ${CERT_HASH} != ${KEY_HASH} ]]; then
  21. log_f "Certificate and key hashes do not match!"
  22. return 1
  23. else
  24. log_f "Verified hashes."
  25. return 0
  26. fi
  27. }
  28. get_ipv4(){
  29. local IPV4=
  30. local IPV4_SRCS=
  31. local TRY=
  32. IPV4_SRCS[0]="ip4.mailcow.email"
  33. IPV4_SRCS[1]="ip4.korves.net"
  34. until [[ ! -z ${IPV4} ]] || [[ ${TRY} -ge 10 ]]; do
  35. IPV4=$(curl --connect-timeout 3 -m 10 -L4s ${IPV4_SRCS[$RANDOM % ${#IPV4_SRCS[@]} ]} | grep -E "^((25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$")
  36. [[ ! -z ${TRY} ]] && sleep 1
  37. TRY=$((TRY+1))
  38. done
  39. echo ${IPV4}
  40. }
  41. get_ipv6(){
  42. local IPV6=
  43. local IPV6_SRCS=
  44. local TRY=
  45. IPV6_SRCS[0]="ip6.korves.net"
  46. IPV6_SRCS[1]="ip6.mailcow.email"
  47. until [[ ! -z ${IPV6} ]] || [[ ${TRY} -ge 10 ]]; do
  48. IPV6=$(curl --connect-timeout 3 -m 10 -L6s ${IPV6_SRCS[$RANDOM % ${#IPV6_SRCS[@]} ]} | grep "^\([0-9a-fA-F]\{0,4\}:\)\{1,7\}[0-9a-fA-F]\{0,4\}$")
  49. [[ ! -z ${TRY} ]] && sleep 1
  50. TRY=$((TRY+1))
  51. done
  52. echo ${IPV6}
  53. }
  54. check_domain(){
  55. DOMAIN=$1
  56. A_DOMAIN=$(dig A ${DOMAIN} +short | tail -n 1)
  57. AAAA_DOMAIN=$(dig AAAA ${DOMAIN} +short | tail -n 1)
  58. # Check if CNAME without v6 enabled target
  59. if [[ ! -z ${AAAA_DOMAIN} ]] && [[ -z $(echo ${AAAA_DOMAIN} | grep "^\([0-9a-fA-F]\{0,4\}:\)\{1,7\}[0-9a-fA-F]\{0,4\}$") ]]; then
  60. AAAA_DOMAIN=
  61. fi
  62. if [[ ! -z ${AAAA_DOMAIN} ]]; then
  63. log_f "Found AAAA record for ${DOMAIN}: ${AAAA_DOMAIN} - skipping A record check"
  64. if [[ $(expand ${IPV6:-"0000:0000:0000:0000:0000:0000:0000:0000"}) == $(expand ${AAAA_DOMAIN}) ]] || [[ ${SKIP_IP_CHECK} == "y" ]]; then
  65. if verify_challenge_path "${DOMAIN}" 6; then
  66. log_f "Confirmed AAAA record with IP ${AAAA_DOMAIN}"
  67. return 0
  68. else
  69. log_f "Confirmed AAAA record with IP ${AAAA_DOMAIN}, but HTTP validation failed"
  70. fi
  71. else
  72. log_f "Cannot match your IP ${IPV6:-NO_IPV6_LINK} against hostname ${DOMAIN} (DNS returned $(expand ${AAAA_DOMAIN}))"
  73. fi
  74. elif [[ ! -z ${A_DOMAIN} ]]; then
  75. log_f "Found A record for ${DOMAIN}: ${A_DOMAIN}"
  76. if [[ ${IPV4:-ERR} == ${A_DOMAIN} ]] || [[ ${SKIP_IP_CHECK} == "y" ]]; then
  77. if verify_challenge_path "${DOMAIN}" 4; then
  78. log_f "Confirmed A record ${A_DOMAIN}"
  79. return 0
  80. else
  81. log_f "Confirmed A record with IP ${A_DOMAIN}, but HTTP validation failed"
  82. fi
  83. else
  84. log_f "Cannot match your IP ${IPV4} against hostname ${DOMAIN} (DNS returned ${A_DOMAIN})"
  85. fi
  86. else
  87. log_f "No A or AAAA record found for hostname ${DOMAIN}"
  88. fi
  89. return 1
  90. }
  91. verify_challenge_path(){
  92. if [[ ${SKIP_HTTP_VERIFICATION} == "y" ]]; then
  93. echo '(skipping check, returning 0)'
  94. return 0
  95. fi
  96. # verify_challenge_path URL 4|6
  97. RANDOM_N=${RANDOM}${RANDOM}${RANDOM}
  98. echo ${RANDOM_N} > /var/www/acme/${RANDOM_N}
  99. if [[ "$(curl --insecure -${2} -L http://${1}/.well-known/acme-challenge/${RANDOM_N} --silent)" == "${RANDOM_N}" ]]; then
  100. rm /var/www/acme/${RANDOM_N}
  101. return 0
  102. else
  103. rm /var/www/acme/${RANDOM_N}
  104. return 1
  105. fi
  106. }