mirror of
https://github.com/drwetter/testssl.sh.git
synced 2025-01-19 15:09:30 +01:00
- fix heartbleed detection which sometimes case false psoitives over slow connections like sattelite links, partially addressing #352
- start revamping run)ccs_injection - fix missing space in BEAST after protocol
This commit is contained in:
parent
67fb3feff8
commit
a22e4e5228
134
testssl.sh
134
testssl.sh
@ -9111,7 +9111,7 @@ run_heartbleed(){
|
|||||||
local -i n ret lines_returned
|
local -i n ret lines_returned
|
||||||
local -i hb_rounds=3
|
local -i hb_rounds=3
|
||||||
local append=""
|
local append=""
|
||||||
local found_500_oops=false
|
local tls_hello_ascii=""
|
||||||
local cve="CVE-2014-0160"
|
local cve="CVE-2014-0160"
|
||||||
local cwe="CWE-119"
|
local cwe="CWE-119"
|
||||||
local hint=""
|
local hint=""
|
||||||
@ -9191,26 +9191,37 @@ run_heartbleed(){
|
|||||||
# extension: heartbeat
|
# extension: heartbeat
|
||||||
x00, x0f, x00, x01, x01"
|
x00, x0f, x00, x01, x01"
|
||||||
|
|
||||||
for (( n=1; n <= hb_rounds; n++)); do
|
fd_socket 5 || return 6
|
||||||
fd_socket 5 || return 6
|
debugme out "\nsending client hello (TLS version $tls_hexcode)"
|
||||||
debugme out "\nsending client hello (TLS version $tls_hexcode)"
|
debugme outln " ($n of $hb_rounds)"
|
||||||
debugme outln " ($n of $hb_rounds)"
|
socksend "$client_hello" 1
|
||||||
socksend "$client_hello" 1
|
|
||||||
|
|
||||||
debugme outln "\nreading server hello"
|
debugme outln "\nreading server hello"
|
||||||
sockread_serverhello 32768
|
sockread_serverhello 32768
|
||||||
if [[ $DEBUG -ge 4 ]]; then
|
if [[ $DEBUG -ge 4 ]]; then
|
||||||
hexdump -C "$SOCK_REPLY_FILE" | head -20
|
hexdump -C "$SOCK_REPLY_FILE" | head -20
|
||||||
outln "[...]"
|
outln "[...]"
|
||||||
outln "\nsending payload with TLS version $tls_hexcode:"
|
outln "\nsending payload with TLS version $tls_hexcode:"
|
||||||
fi
|
fi
|
||||||
rm "$SOCK_REPLY_FILE"
|
rm "$SOCK_REPLY_FILE"
|
||||||
|
|
||||||
socksend "$heartbleed_payload" 1
|
socksend "$heartbleed_payload" 1
|
||||||
sockread_serverhello 16384 $HEARTBLEED_MAX_WAITSOCK
|
sockread_serverhello 16384 $HEARTBLEED_MAX_WAITSOCK
|
||||||
[[ $? -eq 3 ]] && append=", timed out"
|
if [[ $? -eq 3 ]]; then
|
||||||
|
append=", timed out"
|
||||||
|
pr_done_best "not vulnerable (OK)"; out "$append"
|
||||||
|
fileout "heartbleed" "OK" "Heartbleed: not vulnerable $append" "$cve" "$cwe"
|
||||||
|
ret=0
|
||||||
|
else
|
||||||
|
|
||||||
|
# server reply should be (>=SSLv3): 18030x in case of a heartBEAT reply -- which we take as a positive result
|
||||||
|
tls_hello_ascii=$(hexdump -v -e '16/1 "%02X"' "$SOCK_REPLY_FILE")
|
||||||
|
debugme echo "tls_content_type: ${tls_hello_ascii:0:2}"
|
||||||
|
debugme echo "tls_protocol: ${tls_hello_ascii:2:4}"
|
||||||
|
|
||||||
|
lines_returned=$(count_lines "$(hexdump -ve '16/1 "%02x " " \n"' "$SOCK_REPLY_FILE")")
|
||||||
|
debugme echo "lines HB reply: $lines_returned"
|
||||||
|
|
||||||
lines_returned=$(hexdump -ve '16/1 "%02x " " \n"' "$SOCK_REPLY_FILE" | wc -l | sed 's/ //g')
|
|
||||||
if [[ $DEBUG -ge 3 ]]; then
|
if [[ $DEBUG -ge 3 ]]; then
|
||||||
outln "\nheartbleed reply: "
|
outln "\nheartbleed reply: "
|
||||||
hexdump -C "$SOCK_REPLY_FILE" | head -20
|
hexdump -C "$SOCK_REPLY_FILE" | head -20
|
||||||
@ -9218,55 +9229,37 @@ run_heartbleed(){
|
|||||||
outln
|
outln
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ $lines_returned -gt 1 ]]; then
|
if [[ $lines_returned -gt 1 ]] && [[ "${tls_hello_ascii:0:4}" == "1803" ]]; then
|
||||||
if [[ "$STARTTLS_PROTOCOL" == "ftp" ]] || [[ "$STARTTLS_PROTOCOL" == "ftps" ]]; then
|
if [[ "$STARTTLS_PROTOCOL" == "ftp" ]] || [[ "$STARTTLS_PROTOCOL" == "ftps" ]]; then
|
||||||
# check possibility of weird vsftpd reply, see #426
|
# check possibility of weird vsftpd reply, see #426, despite "1803" seems very unlikely...
|
||||||
saved_sockreply[n]="$(hexdump -ve '1/1 "%.2x"' "$SOCK_REPLY_FILE")"
|
if grep -q '500 OOPS' "$SOCK_REPLY_FILE" ; then
|
||||||
[[ $n -eq 1 ]] && grep -q '500 OOPS' "$SOCK_REPLY_FILE" && found_500_oops=true
|
append=", successful weeded out vsftpd false positive"
|
||||||
rm "$SOCK_REPLY_FILE"
|
pr_done_best "not vulnerable (OK)"; out "$append"
|
||||||
#debugme out "${saved_sockreply[n]}"
|
fileout "heartbleed" "OK" "Heartbleed: not vulnerable $append" "$cve" "$cwe"
|
||||||
#TMPFILE="${saved_sockreply[n]}"
|
ret=0
|
||||||
close_socket
|
else
|
||||||
#tmpfile_handle "$FUNCNAME,$n.txt"
|
out "likely "
|
||||||
|
pr_svrty_critical "VULNERABLE (NOT ok)"
|
||||||
|
[[ $DEBUG -lt 3 ]] && out ", use debug >=3 to confirm"
|
||||||
|
fileout "heartbleed" "CRITICAL" "Heartbleed: VULNERABLE $cve" "$cwe" "$hint"
|
||||||
|
ret=1
|
||||||
|
fi
|
||||||
else
|
else
|
||||||
rm "$SOCK_REPLY_FILE"
|
|
||||||
pr_svrty_critical "VULNERABLE (NOT ok)"
|
pr_svrty_critical "VULNERABLE (NOT ok)"
|
||||||
fileout "heartbleed" "CRITICAL" "Heartbleed: VULNERABLE $append" "$cve" "$cwe" "$hint"
|
fileout "heartbleed" "CRITICAL" "Heartbleed: VULNERABLE $cve" "$cwe" "$hint"
|
||||||
ret=1
|
ret=1
|
||||||
break
|
|
||||||
fi
|
fi
|
||||||
else
|
else
|
||||||
rm "$SOCK_REPLY_FILE"
|
|
||||||
pr_done_best "not vulnerable (OK)"
|
pr_done_best "not vulnerable (OK)"
|
||||||
fileout "heartbleed" "OK" "Heartbleed: not vulnerable $append" "$cve" "$cwe"
|
fileout "heartbleed" "OK" "Heartbleed: not vulnerable $cve" "$cwe"
|
||||||
ret=0
|
ret=0
|
||||||
break
|
|
||||||
fi
|
fi
|
||||||
done
|
|
||||||
|
|
||||||
if [[ $n -gt 1 ]]; then
|
|
||||||
# more than one round of heartbleed checks --> vsftpd probably.
|
|
||||||
# This is the robust approach. According to a few tests it could also suffice # to check for "500 OOPS" only.
|
|
||||||
# Checking for the same socket reply DOES NOT suffice -- server can be idle and return the same memory
|
|
||||||
if [[ "${saved_sockreply[1]}" == "${saved_sockreply[2]}" ]] && [[ "${saved_sockreply[2]}" == "${saved_sockreply[3]}" ]] \
|
|
||||||
&& "$found_500_oops"; then
|
|
||||||
pr_done_best "not vulnerable (OK)$append"
|
|
||||||
[[ $DEBUG -ge 1 ]] && out ", successful weeded out vsftpd false positive"
|
|
||||||
fileout "heartbleed" "OK" "Heartbleed: not vulnerable $append" "$cve" "$cwe"
|
|
||||||
else
|
|
||||||
out "likely "
|
|
||||||
pr_svrty_critical "VULNERABLE (NOT ok)"
|
|
||||||
[[ $DEBUG -ge 1 ]] && out " use debug >=2 to confirm"
|
|
||||||
fileout "heartbleed" "CRITICAL" "Heartbleed: likely VULNERABLE $append" "$cve" "$cwe" "$hint"
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
# for the repeated tries we did that already
|
|
||||||
#TMPFILE="$SOCKREPLY"
|
|
||||||
close_socket 2>/dev/null
|
|
||||||
#tmpfile_handle $FUNCNAME.txt
|
|
||||||
fi
|
fi
|
||||||
outln "$append"
|
outln
|
||||||
|
|
||||||
|
TMPFILE="$SOCK_REPLY_FILE"
|
||||||
|
close_socket
|
||||||
|
tmpfile_handle $FUNCNAME.dd
|
||||||
return $ret
|
return $ret
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -9279,7 +9272,8 @@ ok_ids(){
|
|||||||
#FIXME: At a certain point heartbleed and ccs needs to be changed and make use of code2network using a file, then tls_sockets
|
#FIXME: At a certain point heartbleed and ccs needs to be changed and make use of code2network using a file, then tls_sockets
|
||||||
run_ccs_injection(){
|
run_ccs_injection(){
|
||||||
local tls_proto_offered tls_hexcode ccs_message client_hello byte6 sockreply
|
local tls_proto_offered tls_hexcode ccs_message client_hello byte6 sockreply
|
||||||
local -i retval ret lines
|
local -i retval ret lines_returned
|
||||||
|
local tls_hello_ascii=""
|
||||||
local cve="CVE-2014-0224"
|
local cve="CVE-2014-0224"
|
||||||
local cwe="CWE-310"
|
local cwe="CWE-310"
|
||||||
local hint=""
|
local hint=""
|
||||||
@ -9369,22 +9363,27 @@ run_ccs_injection(){
|
|||||||
sockread_serverhello 2048 $CCS_MAX_WAITSOCK
|
sockread_serverhello 2048 $CCS_MAX_WAITSOCK
|
||||||
retval=$?
|
retval=$?
|
||||||
|
|
||||||
|
tls_hello_ascii=$(hexdump -v -e '16/1 "%02X"' "$SOCK_REPLY_FILE")
|
||||||
|
debugme echo "tls_content_type: ${tls_hello_ascii:0:2}"
|
||||||
|
debugme echo "tls_protocol: ${tls_hello_ascii:2:4}"
|
||||||
|
lines_returned=$(count_lines "$(hexdump -ve '16/1 "%02x " " \n"' "$SOCK_REPLY_FILE")")
|
||||||
|
debugme echo "lines new: $lines_returned, byte6: ${tls_hello_ascii:12:2}"
|
||||||
|
|
||||||
if [[ $DEBUG -ge 3 ]]; then
|
if [[ $DEBUG -ge 3 ]]; then
|
||||||
outln "\n2nd reply: "
|
outln "\n2nd reply: "
|
||||||
printf -- "$(hexdump -C "$SOCK_REPLY_FILE")"
|
hexdump -C "$SOCK_REPLY_FILE"
|
||||||
# not ok: 15 | 0301 | 02 | 02 | 15
|
|
||||||
# ALERT | TLS 1.0 | Length=2 | Decryption failed (21)
|
|
||||||
# ok: 0a or nothing: ==> RST
|
|
||||||
outln
|
outln
|
||||||
fi
|
fi
|
||||||
sockreply=$(cat "$SOCK_REPLY_FILE" 2>/dev/null)
|
sockreply=$(cat "$SOCK_REPLY_FILE" 2>/dev/null)
|
||||||
rm "$SOCK_REPLY_FILE"
|
|
||||||
|
|
||||||
byte6=$(echo "$sockreply" | "${HEXDUMPPLAIN[@]}" | sed 's/^..........//')
|
byte6=$(echo "$sockreply" | "${HEXDUMPPLAIN[@]}" | sed 's/^..........//')
|
||||||
lines=$(echo "$sockreply" | "${HEXDUMP[@]}" | count_lines )
|
debugme echo "lines: $lines_returned, byte6: $byte6"
|
||||||
debugme echo "lines: $lines, byte6: $byte6"
|
# not ok: 15 | 0301 | 02 | 02 | 15
|
||||||
|
# ALERT | TLS 1.0 | Length=2 | Decryption failed (21)
|
||||||
|
#
|
||||||
|
# ok: 0a or nothing: ==> RST
|
||||||
|
|
||||||
if [[ "$byte6" == "0a" ]] || [[ "$lines" -gt 1 ]]; then
|
if [[ "$byte6" == "0a" ]] || [[ "$lines_returned" -gt 1 ]]; then
|
||||||
pr_done_best "not vulnerable (OK)"
|
pr_done_best "not vulnerable (OK)"
|
||||||
if [[ $retval -eq 3 ]]; then
|
if [[ $retval -eq 3 ]]; then
|
||||||
fileout "ccs" "OK" "CCS: not vulnerable (timed out)" "$cve" "$cwe"
|
fileout "ccs" "OK" "CCS: not vulnerable (timed out)" "$cve" "$cwe"
|
||||||
@ -9404,8 +9403,9 @@ run_ccs_injection(){
|
|||||||
[[ $retval -eq 3 ]] && out ", timed out"
|
[[ $retval -eq 3 ]] && out ", timed out"
|
||||||
outln
|
outln
|
||||||
|
|
||||||
|
TMPFILE="$SOCK_REPLY_FILE"
|
||||||
close_socket
|
close_socket
|
||||||
tmpfile_handle $FUNCNAME.txt
|
tmpfile_handle $FUNCNAME.dd
|
||||||
return $ret
|
return $ret
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -10465,7 +10465,7 @@ run_beast(){
|
|||||||
if [[ -n "$detected_cbc_ciphers" ]]; then
|
if [[ -n "$detected_cbc_ciphers" ]]; then
|
||||||
fileout "cbc_$proto" "MEDIUM" "BEAST: CBC ciphers for $(toupper $proto): $detected_cbc_ciphers" "$cve" "$cwe" "$hint"
|
fileout "cbc_$proto" "MEDIUM" "BEAST: CBC ciphers for $(toupper $proto): $detected_cbc_ciphers" "$cve" "$cwe" "$hint"
|
||||||
! "$first" && out "$spaces"
|
! "$first" && out "$spaces"
|
||||||
out "$(toupper $proto):"
|
out "$(toupper $proto): "
|
||||||
[[ -n "$higher_proto_supported" ]] && \
|
[[ -n "$higher_proto_supported" ]] && \
|
||||||
out_row_aligned_max_width "$detected_cbc_ciphers" " " $TERM_WIDTH pr_svrty_low || \
|
out_row_aligned_max_width "$detected_cbc_ciphers" " " $TERM_WIDTH pr_svrty_low || \
|
||||||
out_row_aligned_max_width "$detected_cbc_ciphers" " " $TERM_WIDTH pr_svrty_medium
|
out_row_aligned_max_width "$detected_cbc_ciphers" " " $TERM_WIDTH pr_svrty_medium
|
||||||
|
Loading…
Reference in New Issue
Block a user