Merge pull request #1697 from drwetter/no_starttls_detection2

Trying to address no STARTTLS offerings (2)
This commit is contained in:
Dirk Wetter 2020-08-11 16:27:24 +02:00 committed by GitHub
commit a2929211b2
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 95 additions and 64 deletions

View File

@ -15,6 +15,8 @@
* Security fix: DNS input * Security fix: DNS input
* Don't use external pwd anymore * Don't use external pwd anymore
* STARTTLS: XMPP server support * STARTTLS: XMPP server support
* Code improvements to STARTTLS
* Detect better when no STARTTLS is offered
* Rating (SSL Labs, not complete) * Rating (SSL Labs, not complete)
* Don't penalize missing trust in rating when CA not in Java store * Don't penalize missing trust in rating when CA not in Java store
* Added support for certificates with EdDSA signatures and pubilc keys * Added support for certificates with EdDSA signatures and pubilc keys

View File

@ -197,6 +197,7 @@ IGN_OCSP_PROXY=${IGN_OCSP_PROXY:-false} # Also when --proxy is supplied it is ig
HEADER_MAXSLEEP=${HEADER_MAXSLEEP:-5} # we wait this long before killing the process to retrieve a service banner / http header HEADER_MAXSLEEP=${HEADER_MAXSLEEP:-5} # we wait this long before killing the process to retrieve a service banner / http header
MAX_SOCKET_FAIL=${MAX_SOCKET_FAIL:-2} # If this many failures for TCP socket connects are reached we terminate MAX_SOCKET_FAIL=${MAX_SOCKET_FAIL:-2} # If this many failures for TCP socket connects are reached we terminate
MAX_OSSL_FAIL=${MAX_OSSL_FAIL:-2} # If this many failures for s_client connects are reached we terminate MAX_OSSL_FAIL=${MAX_OSSL_FAIL:-2} # If this many failures for s_client connects are reached we terminate
MAX_STARTTLS_FAIL=${MAX_STARTTLS_FAIL:-2} # max number of STARTTLS handshake failures in plaintext phase
MAX_HEADER_FAIL=${MAX_HEADER_FAIL:-2} # If this many failures for HTTP GET are encountered we don't try again to get the header MAX_HEADER_FAIL=${MAX_HEADER_FAIL:-2} # If this many failures for HTTP GET are encountered we don't try again to get the header
MAX_WAITSOCK=${MAX_WAITSOCK:-10} # waiting at max 10 seconds for socket reply. There shouldn't be any reason to change this. MAX_WAITSOCK=${MAX_WAITSOCK:-10} # waiting at max 10 seconds for socket reply. There shouldn't be any reason to change this.
CCS_MAX_WAITSOCK=${CCS_MAX_WAITSOCK:-5} # for the two CCS payload (each). There shouldn't be any reason to change this. CCS_MAX_WAITSOCK=${CCS_MAX_WAITSOCK:-5} # for the two CCS payload (each). There shouldn't be any reason to change this.
@ -252,6 +253,7 @@ TIMEOUT_CMD=""
HAD_SLEPT=0 HAD_SLEPT=0
NR_SOCKET_FAIL=0 # Counter for socket failures NR_SOCKET_FAIL=0 # Counter for socket failures
NR_OSSL_FAIL=0 # .. for OpenSSL connects NR_OSSL_FAIL=0 # .. for OpenSSL connects
NR_STARTTLS_FAIL=0 # .. for STARTTLS failures
NR_HEADER_FAIL=0 # .. for HTTP_GET NR_HEADER_FAIL=0 # .. for HTTP_GET
PROTOS_OFFERED="" # This keeps which protocol is being offered. See has_server_protocol(). PROTOS_OFFERED="" # This keeps which protocol is being offered. See has_server_protocol().
TLS12_CIPHER_OFFERED="" # This contains the hexcode of a cipher known to be supported by the server with TLS 1.2 TLS12_CIPHER_OFFERED="" # This contains the hexcode of a cipher known to be supported by the server with TLS 1.2
@ -10415,16 +10417,19 @@ starttls_io() {
# Line-based send with newline characters appended (arg2 empty) # Line-based send with newline characters appended (arg2 empty)
# Stream-based send (not in use currently): arg2: <any>. # arg2: debug_string -- what we had in the caller previously
starttls_just_send(){ starttls_just_send(){
if [[ -z "$2" ]] ; then local -i ret=0
debugme echo "C: $1\r\n" debugme echo "C: $1\r\n"
echo -ne "$1\r\n" >&5 echo -ne "$1\r\n" >&5
ret=$?
if [[ $ret -eq 0 ]]; then
debugme echo " > succeeded: $2"
else else
debugme echo -e "C: $1" debugme echo " > failed: $2 ($ret)"
echo -ne "$1" >&5
fi fi
return $? return $ret
} }
# arg1: (optional): wait time # arg1: (optional): wait time
@ -10444,66 +10449,74 @@ starttls_just_read(){
starttls_full_read(){ starttls_full_read(){
local cont_pattern="$1" local cont_pattern="$1"
local end_pattern="$2" local end_pattern="$2"
local regex="$3" local starttls_regex="$3" # optional: pattern we search for in the server's response
local debug_str="$4" # optional
local starttls_read_data=() local starttls_read_data=()
local one_line="" local one_line=""
local ret=0 local ret=0
local ret_found=0 local ret_found=0
local debugpad=" > found: " local debugpad=" > found: "
local oldIFS="$IFS"
debugme echo "=== reading banner ... ===" debugme echo "=== reading banner ... ==="
if [[ $# -ge 3 ]]; then if [[ -n "$starttls_regex" ]]; then
debugme echo "=== we'll have to search for \"$regex\" pattern ===" debugme echo "=== we'll have to search for \"$starttls_regex\" pattern ==="
# pre-set an error if we won't find the ~regex
ret_found=3 ret_found=3
fi fi
local oldIFS="$IFS"
IFS='' IFS=''
# Now read handshake line by line and act on the args supplied.
# Exit the subshell if timeout has been hit (-t $STARTTLS_SLEEP)
while read -r -t $STARTTLS_SLEEP one_line; ret=$?; (exit $ret); do while read -r -t $STARTTLS_SLEEP one_line; ret=$?; (exit $ret); do
debugme tmln_out "S: ${one_line}" debugme tmln_out "S: ${one_line}"
if [[ $DEBUG -ge 5 ]]; then if [[ $DEBUG -ge 5 ]]; then
echo "end_pattern/cont_pattern: ${end_pattern} / ${cont_pattern}" echo "end_pattern/cont_pattern: ${end_pattern} / ${cont_pattern}"
fi fi
if [[ $# -ge 3 ]]; then if [[ -n "$starttls_regex" ]]; then
if [[ ${one_line} =~ $regex ]]; then if [[ ${one_line} =~ $starttls_regex ]]; then
ret_found=0
debugme tmln_out "${debugpad} ${one_line} " debugme tmln_out "${debugpad} ${one_line} "
# We don't exit here as the buffer is not empty. So we continue reading but save the status:
ret_found=0
fi fi
fi fi
starttls_read_data+=("${one_line}") starttls_read_data+=("${one_line}")
if [[ ${one_line} =~ ${end_pattern} ]]; then if [[ ${one_line} =~ ${end_pattern} ]]; then
debugme tmln_out "${debugpad} ${one_line} " debugme tmln_out "${debugpad} ${one_line} "
IFS="${oldIFS}" IFS="${oldIFS}"
return ${ret_found} break
fi fi
if [[ ! ${one_line} =~ ${cont_pattern} ]]; then if [[ ! ${one_line} =~ ${cont_pattern} ]]; then
debugme echo "=== full read syntax error, expected regex pattern ${cont_pattern} (cont) or ${end_pattern} (end) ===" debugme echo "=== full read syntax error, expected regex pattern ${cont_pattern} (cont) or ${end_pattern} (end) ==="
IFS="${oldIFS}" IFS="${oldIFS}"
return 2 ret_found=2
break
fi fi
done <&5 done <&5
if [[ $DEBUG -ge 2 ]]; then if [[ $ret_found -eq 0 ]]; then
if [[ $ret -ge 128 ]]; then # Print the debug statement we previously had in the caller function
echo "=== timeout reading ===" [[ -n "$debug_str" ]] && debugme echo " >> $debug_str"
else else
echo "=== full read error (no timeout) ===" if [[ $ret -ge 128 ]]; then
debugme echo "=== timeout reading ==="
$ret_found=$ret
fi fi
fi fi
IFS="${oldIFS}" IFS="${oldIFS}"
return $ret return $ret_found
} }
starttls_ftp_dialog() { starttls_ftp_dialog() {
local debugpad=" > " local -i ret=0
local reAUTHTLS='^ AUTH TLS' local reSTARTTLS='^ AUTH TLS'
debugme echo "=== starting ftp STARTTLS dialog ===" debugme echo "=== starting ftp STARTTLS dialog ==="
starttls_full_read '^220-' '^220 ' && debugme echo "${debugpad}received server greeting" && starttls_full_read '^220-' '^220 ' '' "received server greeting" &&
starttls_just_send 'FEAT' && debugme echo "${debugpad}sent FEAT" && starttls_just_send 'FEAT' "sent FEAT" &&
starttls_full_read '^(211-| )' '^211 ' "${reAUTHTLS}" && debugme echo "${debugpad}received server features and checked STARTTLS availability" && starttls_full_read '^(211-| )' '^211 ' "${reSTARTTLS}" "received server features and checked STARTTLS availability" &&
starttls_just_send 'AUTH TLS' && debugme echo "${debugpad}initiated STARTTLS" && starttls_just_send 'AUTH TLS' "initiated STARTTLS" &&
starttls_full_read '^234-' '^234 ' && debugme echo "${debugpad}received ack for STARTTLS" starttls_full_read '^234-' '^234 ' '' "received ack for STARTTLS"
local ret=$? ret=$?
debugme echo "=== finished ftp STARTTLS dialog with ${ret} ===" debugme echo "=== finished ftp STARTTLS dialog with ${ret} ==="
return $ret return $ret
} }
@ -10513,59 +10526,61 @@ starttls_ftp_dialog() {
starttls_smtp_dialog() { starttls_smtp_dialog() {
local greet_str="EHLO testssl.sh" local greet_str="EHLO testssl.sh"
local proto="smtp" local proto="smtp"
local re250STARTTLS='^250[ -]STARTTLS' local reSTARTTLS='^250[ -]STARTTLS'
local debugpad=" > " local -i ret=0
if [[ "$1" == lmtp ]]; then if [[ "$1" == lmtp ]]; then
proto="lmtp" proto="lmtp"
greet_str="LHLO" greet_str="LHLO"
fi fi
if [[ -n "$2" ]]; then if [[ -n "$2" ]]; then
# Here we can "add" commands in the future. Please note \r\n will automatically appended # Here we can "add" commands in the future. Please note \r\n will automatically be appended
greet_str="$2" greet_str="$2"
elif "$SNEAKY"; then elif "$SNEAKY"; then
greet_str="EHLO google.com" greet_str="EHLO google.com"
fi fi
debugme echo "=== starting $proto STARTTLS dialog ===" debugme echo "=== starting $proto STARTTLS dialog ==="
starttls_full_read '^220-' '^220 ' && debugme echo "${debugpad}received server greeting" && starttls_full_read '^220-' '^220 ' '' "received server greeting" &&
starttls_just_send "$greet_str" && debugme echo "${debugpad}sent $greet_str" && starttls_just_send "$greet_str" "sent $greet_str" &&
starttls_full_read '^250-' '^250 ' "${re250STARTTLS}" && debugme echo "${debugpad}received server capabilities and checked STARTTLS availability" && starttls_full_read '^250-' '^250 ' "${reSTARTTLS}" "received server capabilities and checked STARTTLS availability" &&
starttls_just_send 'STARTTLS' && debugme echo "${debugpad}initiated STARTTLS" && starttls_just_send 'STARTTLS' "initiated STARTTLS" &&
starttls_full_read '^220-' '^220 ' && debugme echo "${debugpad}received ack for STARTTLS" starttls_full_read '^220-' '^220 ' '' "received ack for STARTTLS"
local ret=$? ret=$?
debugme echo "=== finished $proto STARTTLS dialog with ${ret} ===" debugme echo "=== finished $proto STARTTLS dialog with ${ret} ==="
return $ret return $ret
} }
starttls_pop3_dialog() { starttls_pop3_dialog() {
local debugpad=" > " local -i ret=0
debugme echo "=== starting pop3 STARTTLS dialog ===" debugme echo "=== starting pop3 STARTTLS dialog ==="
starttls_full_read '^\+OK' '^\+OK' && debugme echo "${debugpad}received server greeting" && starttls_full_read '^\+OK' '^\+OK' '' "received server greeting" &&
starttls_just_send 'STLS' && debugme echo "${debugpad}initiated STARTTLS" && starttls_just_send 'STLS' "initiated STARTTLS" &&
starttls_full_read '^\+OK' '^\+OK' && debugme echo "${debugpad}received ack for STARTTLS" starttls_full_read '^\+OK' '^\+OK' '' "received ack for STARTTLS"
local ret=$? ret=$?
debugme echo "=== finished pop3 STARTTLS dialog with ${ret} ===" debugme echo "=== finished pop3 STARTTLS dialog with ${ret} ==="
return $ret return $ret
} }
starttls_imap_dialog() { starttls_imap_dialog() {
local -i ret=0
local reSTARTTLS='^\* CAPABILITY(( .*)? IMAP4rev1( .*)? STARTTLS(.*)?|( .*)? STARTTLS( .*)? IMAP4rev1(.*)?)$' local reSTARTTLS='^\* CAPABILITY(( .*)? IMAP4rev1( .*)? STARTTLS(.*)?|( .*)? STARTTLS( .*)? IMAP4rev1(.*)?)$'
local debugpad=" > "
debugme echo "=== starting imap STARTTLS dialog ===" debugme echo "=== starting imap STARTTLS dialog ==="
starttls_full_read '^\* ' '^\* OK ' && debugme echo "${debugpad}received server greeting" && starttls_full_read '^\* ' '^\* OK ' '' "received server greeting" &&
starttls_just_send 'a001 CAPABILITY' && debugme echo "${debugpad}sent CAPABILITY" && starttls_just_send 'a001 CAPABILITY' "sent CAPABILITY" &&
starttls_full_read '^\* ' '^a001 OK ' "${reSTARTTLS}" && debugme echo "${debugpad}received server capabilities and checked STARTTLS availability" && starttls_full_read '^\* ' '^a001 OK ' "${reSTARTTLS}" "received server capabilities and checked STARTTLS availability" &&
starttls_just_send 'a002 STARTTLS' && debugme echo "${debugpad}initiated STARTTLS" && starttls_just_send 'a002 STARTTLS' "initiated STARTTLS" &&
starttls_full_read '^\* ' '^a002 OK ' && debugme echo "${debugpad}received ack for STARTTLS" starttls_full_read '^\* ' '^a002 OK ' '' "received ack for STARTTLS"
local ret=$? ret=$?
debugme echo "=== finished imap STARTTLS dialog with ${ret} ===" debugme echo "=== finished imap STARTTLS dialog with ${ret} ==="
return $ret return $ret
} }
starttls_xmpp_dialog() { starttls_xmpp_dialog() {
local -i ret=0
debugme echo "=== starting xmpp STARTTLS dialog ===" debugme echo "=== starting xmpp STARTTLS dialog ==="
[[ -z $XMPP_HOST ]] && XMPP_HOST="$NODE" [[ -z $XMPP_HOST ]] && XMPP_HOST="$NODE"
@ -10575,36 +10590,40 @@ starttls_xmpp_dialog() {
starttls_io "<stream:stream xmlns:stream='http://etherx.jabber.org/streams' xmlns='"$namespace"' to='"$XMPP_HOST"' version='1.0'>" 'starttls(.*)features' 1 && starttls_io "<stream:stream xmlns:stream='http://etherx.jabber.org/streams' xmlns='"$namespace"' to='"$XMPP_HOST"' version='1.0'>" 'starttls(.*)features' 1 &&
starttls_io "<starttls xmlns='urn:ietf:params:xml:ns:xmpp-tls'/>" '<proceed' 1 starttls_io "<starttls xmlns='urn:ietf:params:xml:ns:xmpp-tls'/>" '<proceed' 1
# starttls_io "<stream:stream xmlns:stream='http://etherx.jabber.org/streams' xmlns='"$namespace"' to='"$XMPP_HOST"' version='1.0'>" 'JUSTSEND' 2 # starttls_io "<stream:stream xmlns:stream='http://etherx.jabber.org/streams' xmlns='"$namespace"' to='"$XMPP_HOST"' version='1.0'>" 'JUSTSEND' 2
local ret=$? ret=$?
debugme echo "=== finished xmpp STARTTLS dialog with ${ret} ===" debugme echo "=== finished xmpp STARTTLS dialog with ${ret} ==="
return $ret return $ret
} }
starttls_nntp_dialog() { starttls_nntp_dialog() {
local debugpad=" > " local -i ret=0
debugme echo "=== starting nntp STARTTLS dialog ===" debugme echo "=== starting nntp STARTTLS dialog ==="
starttls_full_read '$^' '^20[01] ' && debugme echo "${debugpad}received server greeting" && starttls_full_read '$^' '^20[01] ' '' "received server greeting" &&
starttls_just_send 'STARTTLS' && debugme echo "${debugpad}initiated STARTTLS" && starttls_just_send 'STARTTLS' "initiated STARTTLS" &&
starttls_full_read '$^' '^382 ' && debugme echo "${debugpad}received ack for STARTTLS" starttls_full_read '$^' '^382 ' '' "received ack for STARTTLS"
local ret=$? ret=$?
debugme echo "=== finished nntp STARTTLS dialog with ${ret} ===" debugme echo "=== finished nntp STARTTLS dialog with ${ret} ==="
return $ret return $ret
} }
starttls_postgres_dialog() { starttls_postgres_dialog() {
local -i ret=0
local debugpad=" > "
local starttls_init=", x00, x00 ,x00 ,x08 ,x04 ,xD2 ,x16 ,x2F"
debugme echo "=== starting postgres STARTTLS dialog ===" debugme echo "=== starting postgres STARTTLS dialog ==="
local init_tls=", x00, x00 ,x00 ,x08 ,x04 ,xD2 ,x16 ,x2F" socksend "${starttls_init}" 0 && debugme echo "${debugpad}initiated STARTTLS" &&
socksend "${init_tls}" 0 && debugme echo "${debugpad}initiated STARTTLS" &&
starttls_io "" S 1 && debugme echo "${debugpad}received ack (="S") for STARTTLS" starttls_io "" S 1 && debugme echo "${debugpad}received ack (="S") for STARTTLS"
local ret=$? ret=$?
debugme echo "=== finished postgres STARTTLS dialog with ${ret} ===" debugme echo "=== finished postgres STARTTLS dialog with ${ret} ==="
return $ret return $ret
} }
starttls_mysql_dialog() { starttls_mysql_dialog() {
local debugpad=" > " local debugpad=" > "
local login_request=" local -i ret=0
local starttls_init="
, x20, x00, x00, x01, # payload_length, sequence_id , x20, x00, x00, x01, # payload_length, sequence_id
x85, xae, xff, x00, # capability flags, CLIENT_SSL always set x85, xae, xff, x00, # capability flags, CLIENT_SSL always set
x00, x00, x00, x01, # max-packet size x00, x00, x00, x01, # max-packet size
@ -10614,8 +10633,8 @@ starttls_mysql_dialog() {
x00, x00, x00, x00, x00, x00, x00" x00, x00, x00, x00, x00, x00, x00"
debugme echo "=== starting mysql STARTTLS dialog ===" debugme echo "=== starting mysql STARTTLS dialog ==="
socksend "${login_request}" 0 socksend "${starttls_init}" 0 && debugme echo "${debugpad}initiated STARTTLS" &&
starttls_just_read 1 && debugme echo "${debugpad}read succeeded" starttls_just_read 1 "read succeeded"
# 1 is the timeout value which only MySQL needs. Note, there seems no response whether STARTTLS # 1 is the timeout value which only MySQL needs. Note, there seems no response whether STARTTLS
# succeeded. We could try harder, see https://github.com/openssl/openssl/blob/master/apps/s_client.c # succeeded. We could try harder, see https://github.com/openssl/openssl/blob/master/apps/s_client.c
# but atm this seems sufficient as later we will fail if there's no STARTTLS. # but atm this seems sufficient as later we will fail if there's no STARTTLS.
@ -10623,7 +10642,7 @@ starttls_mysql_dialog() {
# also there's a banner in the reply "<version><somebytes>mysql_native_password" # also there's a banner in the reply "<version><somebytes>mysql_native_password"
# TODO: We could detect if the server supports STARTTLS via the "Server Capabilities" # TODO: We could detect if the server supports STARTTLS via the "Server Capabilities"
# bit field, but we'd need to parse the binary stream, with greater precision than regex. # bit field, but we'd need to parse the binary stream, with greater precision than regex.
local ret=$? ret=$?
debugme echo "=== finished mysql STARTTLS dialog with ${ret} ===" debugme echo "=== finished mysql STARTTLS dialog with ${ret} ==="
return $ret return $ret
} }
@ -10731,9 +10750,19 @@ fd_socket() {
*) # we need to throw an error here -- otherwise testssl.sh treats the STARTTLS protocol as plain SSL/TLS which leads to FP *) # we need to throw an error here -- otherwise testssl.sh treats the STARTTLS protocol as plain SSL/TLS which leads to FP
fatal "FIXME: STARTTLS protocol $STARTTLS_PROTOCOL is not yet supported" $ERR_NOSUPPORT fatal "FIXME: STARTTLS protocol $STARTTLS_PROTOCOL is not yet supported" $ERR_NOSUPPORT
esac esac
ret=$?
case $ret in
0) return 0 ;;
3) fatal "No STARTTLS found in handshake" $ERR_CONNECT ;;
*) ((NR_STARTTLS_FAIL++))
# This are mostly timeouts here (code >=128). We give the client a chance to try again later. For cases
# where we have no STARTTLS in the server banner however - ret code=3 - we don't neet to try again
connectivity_problem $NR_STARTTLS_FAIL $MAX_STARTTLS_FAIL "STARTTLS handshake failed (code: $ret)" "repeated STARTTLS problems, giving up ($ret)"
return 6 ;;
esac
fi fi
# Plain socket ok, yes or no?
[[ $? -eq 0 ]] && return 0 [[ $? -eq 0 ]] && return 0
prln_warning " STARTTLS handshake failed"
return 1 return 1
} }