mirror of
https://github.com/plantroon/acme.sh.git
synced 2024-12-22 13:11:41 +00:00
commit
c33e5bc40f
4
.github/workflows/DNS.yml
vendored
4
.github/workflows/DNS.yml
vendored
@ -184,7 +184,7 @@ jobs:
|
|||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
- name: Clone acmetest
|
- name: Clone acmetest
|
||||||
run: cd .. && git clone https://github.com/acmesh-official/acmetest.git && cp -r acme.sh acmetest/
|
run: cd .. && git clone https://github.com/acmesh-official/acmetest.git && cp -r acme.sh acmetest/
|
||||||
- uses: vmactions/freebsd-vm@v0.0.7
|
- uses: vmactions/freebsd-vm@v0.1.2
|
||||||
with:
|
with:
|
||||||
envs: 'TEST_DNS TestingDomain TEST_DNS_NO_WILDCARD TEST_DNS_SLEEP CASE TEST_LOCAL DEBUG ${{ secrets.TokenName1}} ${{ secrets.TokenName2}} ${{ secrets.TokenName3}} ${{ secrets.TokenName4}} ${{ secrets.TokenName5}}'
|
envs: 'TEST_DNS TestingDomain TEST_DNS_NO_WILDCARD TEST_DNS_SLEEP CASE TEST_LOCAL DEBUG ${{ secrets.TokenName1}} ${{ secrets.TokenName2}} ${{ secrets.TokenName3}} ${{ secrets.TokenName4}} ${{ secrets.TokenName5}}'
|
||||||
prepare: pkg install -y socat curl
|
prepare: pkg install -y socat curl
|
||||||
@ -223,7 +223,7 @@ jobs:
|
|||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
- name: Clone acmetest
|
- name: Clone acmetest
|
||||||
run: cd .. && git clone https://github.com/acmesh-official/acmetest.git && cp -r acme.sh acmetest/
|
run: cd .. && git clone https://github.com/acmesh-official/acmetest.git && cp -r acme.sh acmetest/
|
||||||
- uses: vmactions/solaris-vm@v0.0.1
|
- uses: vmactions/solaris-vm@v0.0.3
|
||||||
with:
|
with:
|
||||||
envs: 'TEST_DNS TestingDomain TEST_DNS_NO_WILDCARD TEST_DNS_SLEEP CASE TEST_LOCAL DEBUG ${{ secrets.TokenName1}} ${{ secrets.TokenName2}} ${{ secrets.TokenName3}} ${{ secrets.TokenName4}} ${{ secrets.TokenName5}}'
|
envs: 'TEST_DNS TestingDomain TEST_DNS_NO_WILDCARD TEST_DNS_SLEEP CASE TEST_LOCAL DEBUG ${{ secrets.TokenName1}} ${{ secrets.TokenName2}} ${{ secrets.TokenName3}} ${{ secrets.TokenName4}} ${{ secrets.TokenName5}}'
|
||||||
prepare: pkgutil -y -i socat curl
|
prepare: pkgutil -y -i socat curl
|
||||||
|
13
acme.sh
13
acme.sh
@ -1132,7 +1132,11 @@ _createkey() {
|
|||||||
fi
|
fi
|
||||||
else
|
else
|
||||||
_debug "Using RSA: $length"
|
_debug "Using RSA: $length"
|
||||||
if _opkey="$(${ACME_OPENSSL_BIN:-openssl} genrsa "$length" 2>/dev/null)"; then
|
__traditional=""
|
||||||
|
if _contains "$(${ACME_OPENSSL_BIN:-openssl} help genrsa 2>&1)" "-traditional"; then
|
||||||
|
__traditional="-traditional"
|
||||||
|
fi
|
||||||
|
if _opkey="$(${ACME_OPENSSL_BIN:-openssl} genrsa $__traditional "$length" 2>/dev/null)"; then
|
||||||
echo "$_opkey" >"$f"
|
echo "$_opkey" >"$f"
|
||||||
else
|
else
|
||||||
_err "error rsa key: $length"
|
_err "error rsa key: $length"
|
||||||
@ -2279,6 +2283,13 @@ _clearaccountconf() {
|
|||||||
_clear_conf "$ACCOUNT_CONF_PATH" "$1"
|
_clear_conf "$ACCOUNT_CONF_PATH" "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#key
|
||||||
|
_clearaccountconf_mutable() {
|
||||||
|
_clearaccountconf "SAVED_$1"
|
||||||
|
#remove later
|
||||||
|
_clearaccountconf "$1"
|
||||||
|
}
|
||||||
|
|
||||||
#_savecaconf key value
|
#_savecaconf key value
|
||||||
_savecaconf() {
|
_savecaconf() {
|
||||||
_save_conf "$CA_CONF" "$1" "$2"
|
_save_conf "$CA_CONF" "$1" "$2"
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
# HUAWEICLOUD_ProjectID
|
# HUAWEICLOUD_ProjectID
|
||||||
|
|
||||||
iam_api="https://iam.myhuaweicloud.com"
|
iam_api="https://iam.myhuaweicloud.com"
|
||||||
dns_api="https://dns.ap-southeast-1.myhuaweicloud.com"
|
dns_api="https://dns.ap-southeast-1.myhuaweicloud.com" # Should work
|
||||||
|
|
||||||
######## Public functions #####################
|
######## Public functions #####################
|
||||||
|
|
||||||
@ -29,16 +29,27 @@ dns_huaweicloud_add() {
|
|||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
unset token # Clear token
|
||||||
token="$(_get_token "${HUAWEICLOUD_Username}" "${HUAWEICLOUD_Password}" "${HUAWEICLOUD_ProjectID}")"
|
token="$(_get_token "${HUAWEICLOUD_Username}" "${HUAWEICLOUD_Password}" "${HUAWEICLOUD_ProjectID}")"
|
||||||
_debug2 "${token}"
|
if [ -z "${token}" ]; then # Check token
|
||||||
|
_err "dns_api(dns_huaweicloud): Error getting token."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
_debug "Access token is: ${token}"
|
||||||
|
|
||||||
|
unset zoneid
|
||||||
zoneid="$(_get_zoneid "${token}" "${fulldomain}")"
|
zoneid="$(_get_zoneid "${token}" "${fulldomain}")"
|
||||||
_debug "${zoneid}"
|
if [ -z "${zoneid}" ]; then
|
||||||
|
_err "dns_api(dns_huaweicloud): Error getting zone id."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
_debug "Zone ID is: ${zoneid}"
|
||||||
|
|
||||||
_debug "Adding Record"
|
_debug "Adding Record"
|
||||||
_add_record "${token}" "${fulldomain}" "${txtvalue}"
|
_add_record "${token}" "${fulldomain}" "${txtvalue}"
|
||||||
ret="$?"
|
ret="$?"
|
||||||
if [ "${ret}" != "0" ]; then
|
if [ "${ret}" != "0" ]; then
|
||||||
_err "dns_huaweicloud: Error adding record."
|
_err "dns_api(dns_huaweicloud): Error adding record."
|
||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@ -69,12 +80,21 @@ dns_huaweicloud_rm() {
|
|||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
unset token # Clear token
|
||||||
token="$(_get_token "${HUAWEICLOUD_Username}" "${HUAWEICLOUD_Password}" "${HUAWEICLOUD_ProjectID}")"
|
token="$(_get_token "${HUAWEICLOUD_Username}" "${HUAWEICLOUD_Password}" "${HUAWEICLOUD_ProjectID}")"
|
||||||
_debug2 "${token}"
|
if [ -z "${token}" ]; then # Check token
|
||||||
|
_err "dns_api(dns_huaweicloud): Error getting token."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
_debug "Access token is: ${token}"
|
||||||
|
|
||||||
|
unset zoneid
|
||||||
zoneid="$(_get_zoneid "${token}" "${fulldomain}")"
|
zoneid="$(_get_zoneid "${token}" "${fulldomain}")"
|
||||||
_debug "${zoneid}"
|
if [ -z "${zoneid}" ]; then
|
||||||
record_id="$(_get_recordset_id "${token}" "${fulldomain}" "${zoneid}")"
|
_err "dns_api(dns_huaweicloud): Error getting zone id."
|
||||||
_debug "Record Set ID is: ${record_id}"
|
return 1
|
||||||
|
fi
|
||||||
|
_debug "Zone ID is: ${zoneid}"
|
||||||
|
|
||||||
# Remove all records
|
# Remove all records
|
||||||
# Therotically HuaweiCloud does not allow more than one record set
|
# Therotically HuaweiCloud does not allow more than one record set
|
||||||
|
400
notify/smtp.sh
400
notify/smtp.sh
@ -2,14 +2,398 @@
|
|||||||
|
|
||||||
# support smtp
|
# support smtp
|
||||||
|
|
||||||
smtp_send() {
|
# Please report bugs to https://github.com/acmesh-official/acme.sh/issues/3358
|
||||||
_subject="$1"
|
|
||||||
_content="$2"
|
|
||||||
_statusCode="$3" #0: success, 1: error 2($RENEW_SKIP): skipped
|
|
||||||
_debug "_subject" "$_subject"
|
|
||||||
_debug "_content" "$_content"
|
|
||||||
_debug "_statusCode" "$_statusCode"
|
|
||||||
|
|
||||||
_err "Not implemented yet."
|
# This implementation uses either curl or Python (3 or 2.7).
|
||||||
|
# (See also the "mail" notify hook, which supports other ways to send mail.)
|
||||||
|
|
||||||
|
# SMTP_FROM="from@example.com" # required
|
||||||
|
# SMTP_TO="to@example.com" # required
|
||||||
|
# SMTP_HOST="smtp.example.com" # required
|
||||||
|
# SMTP_PORT="25" # defaults to 25, 465 or 587 depending on SMTP_SECURE
|
||||||
|
# SMTP_SECURE="tls" # one of "none", "ssl" (implicit TLS, TLS Wrapper), "tls" (explicit TLS, STARTTLS)
|
||||||
|
# SMTP_USERNAME="" # set if SMTP server requires login
|
||||||
|
# SMTP_PASSWORD="" # set if SMTP server requires login
|
||||||
|
# SMTP_TIMEOUT="30" # seconds for SMTP operations to timeout
|
||||||
|
# SMTP_BIN="/path/to/python_or_curl" # default finds first of python3, python2.7, python, pypy3, pypy, curl on PATH
|
||||||
|
|
||||||
|
SMTP_SECURE_DEFAULT="tls"
|
||||||
|
SMTP_TIMEOUT_DEFAULT="30"
|
||||||
|
|
||||||
|
# subject content statuscode
|
||||||
|
smtp_send() {
|
||||||
|
SMTP_SUBJECT="$1"
|
||||||
|
SMTP_CONTENT="$2"
|
||||||
|
# UNUSED: _statusCode="$3" # 0: success, 1: error 2($RENEW_SKIP): skipped
|
||||||
|
|
||||||
|
# Load and validate config:
|
||||||
|
SMTP_BIN="$(_readaccountconf_mutable_default SMTP_BIN)"
|
||||||
|
if [ -n "$SMTP_BIN" ] && ! _exists "$SMTP_BIN"; then
|
||||||
|
_err "SMTP_BIN '$SMTP_BIN' does not exist."
|
||||||
return 1
|
return 1
|
||||||
|
fi
|
||||||
|
if [ -z "$SMTP_BIN" ]; then
|
||||||
|
# Look for a command that can communicate with an SMTP server.
|
||||||
|
# (Please don't add sendmail, ssmtp, mutt, mail, or msmtp here.
|
||||||
|
# Those are already handled by the "mail" notify hook.)
|
||||||
|
for cmd in python3 python2.7 python pypy3 pypy curl; do
|
||||||
|
if _exists "$cmd"; then
|
||||||
|
SMTP_BIN="$cmd"
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
if [ -z "$SMTP_BIN" ]; then
|
||||||
|
_err "The smtp notify-hook requires curl or Python, but can't find any."
|
||||||
|
_err 'If you have one of them, define SMTP_BIN="/path/to/curl_or_python".'
|
||||||
|
_err 'Otherwise, see if you can use the "mail" notify-hook instead.'
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
_debug SMTP_BIN "$SMTP_BIN"
|
||||||
|
_saveaccountconf_mutable_default SMTP_BIN "$SMTP_BIN"
|
||||||
|
|
||||||
|
SMTP_FROM="$(_readaccountconf_mutable_default SMTP_FROM)"
|
||||||
|
SMTP_FROM="$(_clean_email_header "$SMTP_FROM")"
|
||||||
|
if [ -z "$SMTP_FROM" ]; then
|
||||||
|
_err "You must define SMTP_FROM as the sender email address."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
if _email_has_display_name "$SMTP_FROM"; then
|
||||||
|
_err "SMTP_FROM must be only a simple email address (sender@example.com)."
|
||||||
|
_err "Change your SMTP_FROM='$SMTP_FROM' to remove the display name."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
_debug SMTP_FROM "$SMTP_FROM"
|
||||||
|
_saveaccountconf_mutable_default SMTP_FROM "$SMTP_FROM"
|
||||||
|
|
||||||
|
SMTP_TO="$(_readaccountconf_mutable_default SMTP_TO)"
|
||||||
|
SMTP_TO="$(_clean_email_header "$SMTP_TO")"
|
||||||
|
if [ -z "$SMTP_TO" ]; then
|
||||||
|
_err "You must define SMTP_TO as the recipient email address(es)."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
if _email_has_display_name "$SMTP_TO"; then
|
||||||
|
_err "SMTP_TO must be only simple email addresses (to@example.com,to2@example.com)."
|
||||||
|
_err "Change your SMTP_TO='$SMTP_TO' to remove the display name(s)."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
_debug SMTP_TO "$SMTP_TO"
|
||||||
|
_saveaccountconf_mutable_default SMTP_TO "$SMTP_TO"
|
||||||
|
|
||||||
|
SMTP_HOST="$(_readaccountconf_mutable_default SMTP_HOST)"
|
||||||
|
if [ -z "$SMTP_HOST" ]; then
|
||||||
|
_err "You must define SMTP_HOST as the SMTP server hostname."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
_debug SMTP_HOST "$SMTP_HOST"
|
||||||
|
_saveaccountconf_mutable_default SMTP_HOST "$SMTP_HOST"
|
||||||
|
|
||||||
|
SMTP_SECURE="$(_readaccountconf_mutable_default SMTP_SECURE "$SMTP_SECURE_DEFAULT")"
|
||||||
|
case "$SMTP_SECURE" in
|
||||||
|
"none") smtp_port_default="25" ;;
|
||||||
|
"ssl") smtp_port_default="465" ;;
|
||||||
|
"tls") smtp_port_default="587" ;;
|
||||||
|
*)
|
||||||
|
_err "Invalid SMTP_SECURE='$SMTP_SECURE'. It must be 'ssl', 'tls' or 'none'."
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
_debug SMTP_SECURE "$SMTP_SECURE"
|
||||||
|
_saveaccountconf_mutable_default SMTP_SECURE "$SMTP_SECURE" "$SMTP_SECURE_DEFAULT"
|
||||||
|
|
||||||
|
SMTP_PORT="$(_readaccountconf_mutable_default SMTP_PORT "$smtp_port_default")"
|
||||||
|
case "$SMTP_PORT" in
|
||||||
|
*[!0-9]*)
|
||||||
|
_err "Invalid SMTP_PORT='$SMTP_PORT'. It must be a port number."
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
_debug SMTP_PORT "$SMTP_PORT"
|
||||||
|
_saveaccountconf_mutable_default SMTP_PORT "$SMTP_PORT" "$smtp_port_default"
|
||||||
|
|
||||||
|
SMTP_USERNAME="$(_readaccountconf_mutable_default SMTP_USERNAME)"
|
||||||
|
_debug SMTP_USERNAME "$SMTP_USERNAME"
|
||||||
|
_saveaccountconf_mutable_default SMTP_USERNAME "$SMTP_USERNAME"
|
||||||
|
|
||||||
|
SMTP_PASSWORD="$(_readaccountconf_mutable_default SMTP_PASSWORD)"
|
||||||
|
_secure_debug SMTP_PASSWORD "$SMTP_PASSWORD"
|
||||||
|
_saveaccountconf_mutable_default SMTP_PASSWORD "$SMTP_PASSWORD"
|
||||||
|
|
||||||
|
SMTP_TIMEOUT="$(_readaccountconf_mutable_default SMTP_TIMEOUT "$SMTP_TIMEOUT_DEFAULT")"
|
||||||
|
_debug SMTP_TIMEOUT "$SMTP_TIMEOUT"
|
||||||
|
_saveaccountconf_mutable_default SMTP_TIMEOUT "$SMTP_TIMEOUT" "$SMTP_TIMEOUT_DEFAULT"
|
||||||
|
|
||||||
|
SMTP_X_MAILER="$(_clean_email_header "$PROJECT_NAME $VER --notify-hook smtp")"
|
||||||
|
|
||||||
|
# Run with --debug 2 (or above) to echo the transcript of the SMTP session.
|
||||||
|
# Careful: this may include SMTP_PASSWORD in plaintext!
|
||||||
|
if [ "${DEBUG:-$DEBUG_LEVEL_NONE}" -ge "$DEBUG_LEVEL_2" ]; then
|
||||||
|
SMTP_SHOW_TRANSCRIPT="True"
|
||||||
|
else
|
||||||
|
SMTP_SHOW_TRANSCRIPT=""
|
||||||
|
fi
|
||||||
|
|
||||||
|
SMTP_SUBJECT=$(_clean_email_header "$SMTP_SUBJECT")
|
||||||
|
_debug SMTP_SUBJECT "$SMTP_SUBJECT"
|
||||||
|
_debug SMTP_CONTENT "$SMTP_CONTENT"
|
||||||
|
|
||||||
|
# Send the message:
|
||||||
|
case "$(basename "$SMTP_BIN")" in
|
||||||
|
curl) _smtp_send=_smtp_send_curl ;;
|
||||||
|
py*) _smtp_send=_smtp_send_python ;;
|
||||||
|
*)
|
||||||
|
_err "Can't figure out how to invoke '$SMTP_BIN'."
|
||||||
|
_err "Check your SMTP_BIN setting."
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
if ! smtp_output="$($_smtp_send)"; then
|
||||||
|
_err "Error sending message with $SMTP_BIN."
|
||||||
|
if [ -n "$smtp_output" ]; then
|
||||||
|
_err "$smtp_output"
|
||||||
|
fi
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
# Strip CR and NL from text to prevent MIME header injection
|
||||||
|
# text
|
||||||
|
_clean_email_header() {
|
||||||
|
printf "%s" "$(echo "$1" | tr -d "\r\n")"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Simple check for display name in an email address (< > or ")
|
||||||
|
# email
|
||||||
|
_email_has_display_name() {
|
||||||
|
_email="$1"
|
||||||
|
expr "$_email" : '^.*[<>"]' >/dev/null
|
||||||
|
}
|
||||||
|
|
||||||
|
##
|
||||||
|
## curl smtp sending
|
||||||
|
##
|
||||||
|
|
||||||
|
# Send the message via curl using SMTP_* variables
|
||||||
|
_smtp_send_curl() {
|
||||||
|
# Build curl args in $@
|
||||||
|
case "$SMTP_SECURE" in
|
||||||
|
none)
|
||||||
|
set -- --url "smtp://${SMTP_HOST}:${SMTP_PORT}"
|
||||||
|
;;
|
||||||
|
ssl)
|
||||||
|
set -- --url "smtps://${SMTP_HOST}:${SMTP_PORT}"
|
||||||
|
;;
|
||||||
|
tls)
|
||||||
|
set -- --url "smtp://${SMTP_HOST}:${SMTP_PORT}" --ssl-reqd
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
# This will only occur if someone adds a new SMTP_SECURE option above
|
||||||
|
# without updating this code for it.
|
||||||
|
_err "Unhandled SMTP_SECURE='$SMTP_SECURE' in _smtp_send_curl"
|
||||||
|
_err "Please re-run with --debug and report a bug."
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
set -- "$@" \
|
||||||
|
--upload-file - \
|
||||||
|
--mail-from "$SMTP_FROM" \
|
||||||
|
--max-time "$SMTP_TIMEOUT"
|
||||||
|
|
||||||
|
# Burst comma-separated $SMTP_TO into individual --mail-rcpt args.
|
||||||
|
_to="${SMTP_TO},"
|
||||||
|
while [ -n "$_to" ]; do
|
||||||
|
_rcpt="${_to%%,*}"
|
||||||
|
_to="${_to#*,}"
|
||||||
|
set -- "$@" --mail-rcpt "$_rcpt"
|
||||||
|
done
|
||||||
|
|
||||||
|
_smtp_login="${SMTP_USERNAME}:${SMTP_PASSWORD}"
|
||||||
|
if [ "$_smtp_login" != ":" ]; then
|
||||||
|
set -- "$@" --user "$_smtp_login"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "$SMTP_SHOW_TRANSCRIPT" = "True" ]; then
|
||||||
|
set -- "$@" --verbose
|
||||||
|
else
|
||||||
|
set -- "$@" --silent --show-error
|
||||||
|
fi
|
||||||
|
|
||||||
|
raw_message="$(_smtp_raw_message)"
|
||||||
|
|
||||||
|
_debug2 "curl command:" "$SMTP_BIN" "$*"
|
||||||
|
_debug2 "raw_message:\n$raw_message"
|
||||||
|
|
||||||
|
echo "$raw_message" | "$SMTP_BIN" "$@"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Output an RFC-822 / RFC-5322 email message using SMTP_* variables.
|
||||||
|
# (This assumes variables have already been cleaned for use in email headers.)
|
||||||
|
_smtp_raw_message() {
|
||||||
|
echo "From: $SMTP_FROM"
|
||||||
|
echo "To: $SMTP_TO"
|
||||||
|
echo "Subject: $(_mime_encoded_word "$SMTP_SUBJECT")"
|
||||||
|
echo "Date: $(_rfc2822_date)"
|
||||||
|
echo "Content-Type: text/plain; charset=utf-8"
|
||||||
|
echo "X-Mailer: $SMTP_X_MAILER"
|
||||||
|
echo
|
||||||
|
echo "$SMTP_CONTENT"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Convert text to RFC-2047 MIME "encoded word" format if it contains non-ASCII chars
|
||||||
|
# text
|
||||||
|
_mime_encoded_word() {
|
||||||
|
_text="$1"
|
||||||
|
# (regex character ranges like [a-z] can be locale-dependent; enumerate ASCII chars to avoid that)
|
||||||
|
_ascii='] $`"'"[!#%&'()*+,./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ~^_abcdefghijklmnopqrstuvwxyz{|}~-"
|
||||||
|
if expr "$_text" : "^.*[^$_ascii]" >/dev/null; then
|
||||||
|
# At least one non-ASCII char; convert entire thing to encoded word
|
||||||
|
printf "%s" "=?UTF-8?B?$(printf "%s" "$_text" | _base64)?="
|
||||||
|
else
|
||||||
|
# Just printable ASCII, no conversion needed
|
||||||
|
printf "%s" "$_text"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# Output current date in RFC-2822 Section 3.3 format as required in email headers
|
||||||
|
# (e.g., "Mon, 15 Feb 2021 14:22:01 -0800")
|
||||||
|
_rfc2822_date() {
|
||||||
|
# Notes:
|
||||||
|
# - this is deliberately not UTC, because it "SHOULD express local time" per spec
|
||||||
|
# - the spec requires weekday and month in the C locale (English), not localized
|
||||||
|
# - this date format specifier has been tested on Linux, Mac, Solaris and FreeBSD
|
||||||
|
_old_lc_time="$LC_TIME"
|
||||||
|
LC_TIME=C
|
||||||
|
date +'%a, %-d %b %Y %H:%M:%S %z'
|
||||||
|
LC_TIME="$_old_lc_time"
|
||||||
|
}
|
||||||
|
|
||||||
|
##
|
||||||
|
## Python smtp sending
|
||||||
|
##
|
||||||
|
|
||||||
|
# Send the message via Python using SMTP_* variables
|
||||||
|
_smtp_send_python() {
|
||||||
|
_debug "Python version" "$("$SMTP_BIN" --version 2>&1)"
|
||||||
|
|
||||||
|
# language=Python
|
||||||
|
"$SMTP_BIN" <<PYTHON
|
||||||
|
# This code is meant to work with either Python 2.7.x or Python 3.4+.
|
||||||
|
try:
|
||||||
|
try:
|
||||||
|
from email.message import EmailMessage
|
||||||
|
from email.policy import default as email_policy_default
|
||||||
|
except ImportError:
|
||||||
|
# Python 2 (or < 3.3)
|
||||||
|
from email.mime.text import MIMEText as EmailMessage
|
||||||
|
email_policy_default = None
|
||||||
|
from email.utils import formatdate as rfc2822_date
|
||||||
|
from smtplib import SMTP, SMTP_SSL, SMTPException
|
||||||
|
from socket import error as SocketError
|
||||||
|
except ImportError as err:
|
||||||
|
print("A required Python standard package is missing. This system may have"
|
||||||
|
" a reduced version of Python unsuitable for sending mail: %s" % err)
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
show_transcript = """$SMTP_SHOW_TRANSCRIPT""" == "True"
|
||||||
|
|
||||||
|
smtp_host = """$SMTP_HOST"""
|
||||||
|
smtp_port = int("""$SMTP_PORT""")
|
||||||
|
smtp_secure = """$SMTP_SECURE"""
|
||||||
|
username = """$SMTP_USERNAME"""
|
||||||
|
password = """$SMTP_PASSWORD"""
|
||||||
|
timeout=int("""$SMTP_TIMEOUT""") # seconds
|
||||||
|
x_mailer="""$SMTP_X_MAILER"""
|
||||||
|
|
||||||
|
from_email="""$SMTP_FROM"""
|
||||||
|
to_emails="""$SMTP_TO""" # can be comma-separated
|
||||||
|
subject="""$SMTP_SUBJECT"""
|
||||||
|
content="""$SMTP_CONTENT"""
|
||||||
|
|
||||||
|
try:
|
||||||
|
msg = EmailMessage(policy=email_policy_default)
|
||||||
|
msg.set_content(content)
|
||||||
|
except (AttributeError, TypeError):
|
||||||
|
# Python 2 MIMEText
|
||||||
|
msg = EmailMessage(content)
|
||||||
|
msg["Subject"] = subject
|
||||||
|
msg["From"] = from_email
|
||||||
|
msg["To"] = to_emails
|
||||||
|
msg["Date"] = rfc2822_date(localtime=True)
|
||||||
|
msg["X-Mailer"] = x_mailer
|
||||||
|
|
||||||
|
smtp = None
|
||||||
|
try:
|
||||||
|
if smtp_secure == "ssl":
|
||||||
|
smtp = SMTP_SSL(smtp_host, smtp_port, timeout=timeout)
|
||||||
|
else:
|
||||||
|
smtp = SMTP(smtp_host, smtp_port, timeout=timeout)
|
||||||
|
smtp.set_debuglevel(show_transcript)
|
||||||
|
if smtp_secure == "tls":
|
||||||
|
smtp.starttls()
|
||||||
|
if username or password:
|
||||||
|
smtp.login(username, password)
|
||||||
|
smtp.sendmail(msg["From"], msg["To"].split(","), msg.as_string())
|
||||||
|
|
||||||
|
except SMTPException as err:
|
||||||
|
# Output just the error (skip the Python stack trace) for SMTP errors
|
||||||
|
print("Error sending: %r" % err)
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
except SocketError as err:
|
||||||
|
print("Error connecting to %s:%d: %r" % (smtp_host, smtp_port, err))
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
finally:
|
||||||
|
if smtp is not None:
|
||||||
|
smtp.quit()
|
||||||
|
PYTHON
|
||||||
|
}
|
||||||
|
|
||||||
|
##
|
||||||
|
## Conf helpers
|
||||||
|
##
|
||||||
|
|
||||||
|
#_readaccountconf_mutable_default name default_value
|
||||||
|
# Given a name like MY_CONF:
|
||||||
|
# - if MY_CONF is set and non-empty, output $MY_CONF
|
||||||
|
# - if MY_CONF is set _empty_, output $default_value
|
||||||
|
# (lets user `export MY_CONF=` to clear previous saved value
|
||||||
|
# and return to default, without user having to know default)
|
||||||
|
# - otherwise if _readaccountconf_mutable MY_CONF is non-empty, return that
|
||||||
|
# (value of SAVED_MY_CONF from account.conf)
|
||||||
|
# - otherwise output $default_value
|
||||||
|
_readaccountconf_mutable_default() {
|
||||||
|
_name="$1"
|
||||||
|
_default_value="$2"
|
||||||
|
|
||||||
|
eval "_value=\"\$$_name\""
|
||||||
|
eval "_name_is_set=\"\${${_name}+true}\""
|
||||||
|
# ($_name_is_set is "true" if $$_name is set to anything, including empty)
|
||||||
|
if [ -z "${_value}" ] && [ "${_name_is_set:-}" != "true" ]; then
|
||||||
|
_value="$(_readaccountconf_mutable "$_name")"
|
||||||
|
fi
|
||||||
|
if [ -z "${_value}" ]; then
|
||||||
|
_value="$_default_value"
|
||||||
|
fi
|
||||||
|
printf "%s" "$_value"
|
||||||
|
}
|
||||||
|
|
||||||
|
#_saveaccountconf_mutable_default name value default_value base64encode
|
||||||
|
# Like _saveaccountconf_mutable, but if value is default_value
|
||||||
|
# then _clearaccountconf_mutable instead
|
||||||
|
_saveaccountconf_mutable_default() {
|
||||||
|
_name="$1"
|
||||||
|
_value="$2"
|
||||||
|
_default_value="$3"
|
||||||
|
_base64encode="$4"
|
||||||
|
|
||||||
|
if [ "$_value" != "$_default_value" ]; then
|
||||||
|
_saveaccountconf_mutable "$_name" "$_value" "$_base64encode"
|
||||||
|
else
|
||||||
|
_clearaccountconf_mutable "$_name"
|
||||||
|
fi
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user