House keeping: consolidating socket functions

* Put all low level socket related functions close to each other

* removed socksend2 as it was not used and outdated looking forward

* socksend_sslv2_clienthello() renamed to socksend_clienthello() as
  it wasn't particular SSLv2 related

* removed the low level socket calls from socksend_tls_clienthello()
  and called socksend_clienthello() instead

* renamed socksend_tls_clienthello() to prepare_tls_clienthello()
  as it is not a low level function anymore
This commit is contained in:
Dirk 2018-10-05 15:41:16 +02:00
parent d6f1064b9c
commit 76c7299124

View File

@ -3104,38 +3104,6 @@ sub_cipherlists() {
} }
# sockets inspired by http://blog.chris007.de/?p=238
# ARG1: hexbyte with a leading comma (!!), separated by commas
# ARG2: sleep
socksend2() {
local data
# the following works under BSD and Linux, which is quite tricky. So don't mess with it unless you're really sure what you do
if "$HAS_SED_E"; then
data=$(sed -e 's/# .*$//g' -e 's/ //g' <<< "$1" | sed -E 's/^[[:space:]]+//; s/[[:space:]]+$//; /^$/d' | sed 's/,/\\/g' | tr -d '\n')
else
data=$(sed -e 's/# .*$//g' -e 's/ //g' <<< "$1" | sed -r 's/^[[:space:]]+//; s/[[:space:]]+$//; /^$/d' | sed 's/,/\\/g' | tr -d '\n')
fi
[[ $DEBUG -ge 4 ]] && echo && echo "\"$data\""
$PRINTF -- "$data" >&5 2>/dev/null &
sleep $2
}
socksend() {
local data line
# read line per line and strip comments (bash internal func can't handle multiline statements
data="$(while read line; do
printf "${line%%\#*}"
done <<< "$1" )"
data="${data// /}" # strip ' '
data="${data//,/\\}" # s&r , by \
[[ $DEBUG -ge 4 ]] && echo && echo "\"$data\""
$PRINTF -- "$data" >&5 2>/dev/null &
sleep $2
}
openssl2rfc() { openssl2rfc() {
local rfcname="" local rfcname=""
local -i i local -i i
@ -9282,7 +9250,6 @@ EOF
return 1 return 1
} }
close_socket(){ close_socket(){
exec 5<&- exec 5<&-
exec 5>&- exec 5>&-
@ -9290,20 +9257,17 @@ close_socket(){
} }
# first: helper function for protocol checks # Format string properly for socket
# arg1: formatted string here in the code # ARG1: any commented sequence of two bytes hex, separated by commas. It can contain comments, new lines, tabs and white spaces
# NW_STR holds the global with the string prepared for printf, like '\x16\x03\x03\'
code2network() { code2network() {
NW_STR=$(sed -e 's/,/\\\x/g' <<< "$1" | sed -e 's/# .*$//g' -e 's/ //g' -e '/^$/d' | tr -d '\n' | tr -d '\t') NW_STR=$(sed -e 's/,/\\\x/g' <<< "$1" | sed -e 's/# .*$//g' -e 's/ //g' -e '/^$/d' | tr -d '\n' | tr -d '\t')
} }
len2twobytes() { # sockets inspired by http://blog.chris007.de/?p=238
local len_arg1=${#1} # ARG1: hexbytes separated by commas, with a leading comma
[[ $len_arg1 -le 2 ]] && LEN_STR=$(printf "00, %02s \n" "$1") # ARG2: seconds to sleep
[[ $len_arg1 -eq 3 ]] && LEN_STR=$(printf "0%s, %02s \n" "${1:0:1}" "${1:1:2}") socksend_clienthello() {
[[ $len_arg1 -eq 4 ]] && LEN_STR=$(printf "%02s, %02s \n" "${1:0:2}" "${1:2:2}")
}
socksend_sslv2_clienthello() {
local data="" local data=""
code2network "$1" code2network "$1"
@ -9313,7 +9277,26 @@ socksend_sslv2_clienthello() {
sleep $USLEEP_SND sleep $USLEEP_SND
} }
# ARG1: hexbytes -- preceeded by x -- separated by commas, with a leading comma
# ARG2: seconds to sleep
socksend() {
local data line
# read line per line and strip comments (bash internal func can't handle multiline statements
data="$(while read line; do
printf "${line%%\#*}"
done <<< "$1" )"
data="${data// /}" # strip ' '
data="${data//,/\\}" # s&r , by \
[[ $DEBUG -ge 4 ]] && echo && echo "\"$data\""
$PRINTF -- "$data" >&5 2>/dev/null &
sleep $2
}
# for SSLv2 to TLS 1.2: # for SSLv2 to TLS 1.2:
# ARG1: blocksize for reading
sockread_serverhello() { sockread_serverhello() {
[[ -z "$2" ]] && maxsleep=$MAX_WAITSOCK || maxsleep=$2 [[ -z "$2" ]] && maxsleep=$MAX_WAITSOCK || maxsleep=$2
SOCK_REPLY_FILE=$(mktemp $TEMPDIR/ddreply.XXXXXX) || return 7 SOCK_REPLY_FILE=$(mktemp $TEMPDIR/ddreply.XXXXXX) || return 7
@ -9323,10 +9306,19 @@ sockread_serverhello() {
} }
#trying a faster version #trying a faster version
# ARG1: blocksize for reading
sockread_fast() { sockread_fast() {
dd bs=$1 count=1 <&5 2>/dev/null | hexdump -v -e '16/1 "%02X"' dd bs=$1 count=1 <&5 2>/dev/null | hexdump -v -e '16/1 "%02X"'
} }
len2twobytes() {
local len_arg1=${#1}
[[ $len_arg1 -le 2 ]] && LEN_STR=$(printf "00, %02s \n" "$1")
[[ $len_arg1 -eq 3 ]] && LEN_STR=$(printf "0%s, %02s \n" "${1:0:1}" "${1:1:2}")
[[ $len_arg1 -eq 4 ]] && LEN_STR=$(printf "%02s, %02s \n" "${1:0:2}" "${1:2:2}")
}
get_pub_key_size() { get_pub_key_size() {
local pubkey pubkeybits local pubkey pubkeybits
local -i i len1 len local -i i len1 len
@ -11631,7 +11623,7 @@ sslv2_sockets() {
fd_socket 5 || return 6 fd_socket 5 || return 6
debugme echo -n "sending client hello... " debugme echo -n "sending client hello... "
socksend_sslv2_clienthello "$client_hello" socksend_clienthello "$client_hello"
sockread_serverhello 32768 sockread_serverhello 32768
if "$parse_complete"; then if "$parse_complete"; then
@ -11741,9 +11733,10 @@ generate_key_share_extension() {
# "ephemeralkey" - extract the server's ephemeral key (if any) # "ephemeralkey" - extract the server's ephemeral key (if any)
# ARG4: (optional) additional request extensions # ARG4: (optional) additional request extensions
# ARG5: (optional): "true" if ClientHello should advertise compression methods other than "NULL" # ARG5: (optional): "true" if ClientHello should advertise compression methods other than "NULL"
# ARG6: (optional): "false" if socksend_tls_clienthello() should not open a new socket # ARG6: (optional): "false" if prepare_tls_clienthello() should not open a new socket
# ARG7: (optional): "true" if this is a second ClientHello that follows receipt of a HelloRetryRequest # ARG7: (optional): "true" if this is a second ClientHello that follows receipt of a HelloRetryRequest
socksend_tls_clienthello() { #
prepare_tls_clienthello() {
local tls_low_byte="$1" tls_legacy_version="$1" local tls_low_byte="$1" tls_legacy_version="$1"
local process_full="$3" local process_full="$3"
local new_socket=true is_second_clienthello=false local new_socket=true is_second_clienthello=false
@ -12116,11 +12109,8 @@ socksend_tls_clienthello() {
fd_socket 5 || return 6 fd_socket 5 || return 6
fi fi
code2network "$TLS_CLIENT_HELLO$all_extensions" debugme echo -n "sending client hello... "
data="$NW_STR" socksend_clienthello "$TLS_CLIENT_HELLO$all_extensions" $USLEEP_SND
[[ "$DEBUG" -ge 4 ]] && echo && echo "\"$data\""
$PRINTF -- "$data" >&5 2>/dev/null &
sleep $USLEEP_SND
if [[ "$tls_low_byte" -gt 0x03 ]]; then if [[ "$tls_low_byte" -gt 0x03 ]]; then
TLS_CLIENT_HELLO="$(tolower "$NW_STR")" TLS_CLIENT_HELLO="$(tolower "$NW_STR")"
@ -12345,9 +12335,9 @@ resend_if_hello_retry_request() {
debugme echo -en "\nsending second client hello... " debugme echo -en "\nsending second client hello... "
# Starting with TLSv1.3 draft 24, the second ClientHello should specify a record layer version of 0x0303 # Starting with TLSv1.3 draft 24, the second ClientHello should specify a record layer version of 0x0303
if [[ "$server_version" == "0304" ]] || [[ 0x$server_version -ge 0x7f18 ]]; then if [[ "$server_version" == "0304" ]] || [[ 0x$server_version -ge 0x7f18 ]]; then
socksend_tls_clienthello "$tls_low_byte" "$cipher_list_2send" "$process_full" "$new_extra_extns" "" "false" "true" prepare_tls_clienthello "$tls_low_byte" "$cipher_list_2send" "$process_full" "$new_extra_extns" "" "false" "true"
else else
socksend_tls_clienthello "$tls_low_byte" "$cipher_list_2send" "$process_full" "$new_extra_extns" "" "false" prepare_tls_clienthello "$tls_low_byte" "$cipher_list_2send" "$process_full" "$new_extra_extns" "" "false"
fi fi
if [[ $? -ne 0 ]]; then if [[ $? -ne 0 ]]; then
debugme echo "stuck on sending: $ret" debugme echo "stuck on sending: $ret"
@ -12397,7 +12387,7 @@ tls_sockets() {
cipher_list_2send="$NW_STR" cipher_list_2send="$NW_STR"
debugme echo -en "\nsending client hello... " debugme echo -en "\nsending client hello... "
socksend_tls_clienthello "$tls_low_byte" "$cipher_list_2send" "$process_full" "$4" "$offer_compression" prepare_tls_clienthello "$tls_low_byte" "$cipher_list_2send" "$process_full" "$4" "$offer_compression"
ret=$? # 6 means opening socket didn't succeed, e.g. timeout ret=$? # 6 means opening socket didn't succeed, e.g. timeout
# if sending didn't succeed we don't bother # if sending didn't succeed we don't bother
@ -14825,7 +14815,7 @@ run_grease() {
# of the extension.) # of the extension.)
# The "extra extensions" parameter needs to include the padding and # The "extra extensions" parameter needs to include the padding and
# heartbeat extensions, since otherwise socksend_tls_clienthello() # heartbeat extensions, since otherwise prepare_tls_clienthello()
# will add these extensions to the end of the ClientHello. # will add these extensions to the end of the ClientHello.
debugme echo -e "\nSending ClientHello with empty last extension." debugme echo -e "\nSending ClientHello with empty last extension."
tls_sockets "$proto" "$cipher_list" "" " tls_sockets "$proto" "$cipher_list" "" "
@ -14869,7 +14859,7 @@ run_grease() {
# extension, however, may not be an option, since the server may reject the # extension, however, may not be an option, since the server may reject the
# connection attempt for that reason. # connection attempt for that reason.
if "$normal_hello_ok" && [[ "$proto" != "00" ]] && [[ ${#SNI} -le 87 ]]; then if "$normal_hello_ok" && [[ "$proto" != "00" ]] && [[ ${#SNI} -le 87 ]]; then
# Normally socksend_tls_clienthello() will add a padding extension with a length # Normally prepare_tls_clienthello() will add a padding extension with a length
# that will make the ClientHello be 512 bytes in length. Providing an "extra # that will make the ClientHello be 512 bytes in length. Providing an "extra
# extensions" parameter with a short padding extension prevents that. # extensions" parameter with a short padding extension prevents that.
debugme echo -e "\nSending ClientHello with length between 256 and 511 bytes." debugme echo -e "\nSending ClientHello with length between 256 and 511 bytes."