Python SSLError: ошибка на стороне клиента (EOF произошла с нарушением протокола), ошибка на стороне сервера (SSL3_GET_RECORD: неверный номер версии) - PullRequest
4 голосов
/ 23 ноября 2011

У меня возникли некоторые трудности при попытке создать сокет SSL в Python для использования прокси-сервера, который требует аутентификации. Я очень извиняюсь за длину, но я чувствовал, что было бы лучше включить как можно больше деталей.

Во-первых, код сервера выглядит следующим образом:

class ThreadedTCPServer(SocketServer.ThreadingMixIn, SocketServer.TCPServer):

    def __init__(self, server_address, RequestHandlerClass, client_manager, recv_queue):
        SocketServer.TCPServer.__init__(self, server_address, RequestHandlerClass, bind_and_activate=True)

         <snipped out extra code>

class ThreadedTCPRequestHandler(SocketServer.BaseRequestHandler):

    def setup(self):

        while True:
            try:
                print 'trying to wrap in ssl'
                self.request = ssl.wrap_socket(self.request,
                                   certfile=(os.getcwd() + '/ssl_certs/newcert.pem'),
                                   keyfile=(os.getcwd() + '/ssl_certs/webserver.nopass.key'),
                                   server_side=True,
                                   cert_reqs=ssl.CERT_NONE,
                                   ssl_version=ssl.PROTOCOL_TLSv1,
                                   do_handshake_on_connect=False,
                                   suppress_ragged_eofs=True)
                break

            except Exception, ex:
                print 'error trying to wrap in ssl %s' % ex

    def handle(self):
        # Display message that client has connected
        print '\r[*] Received connection from %s:%s\r' % (self.client_address[0], self.client_address[1])

        while self.stopped() == False:
            recv_msg = self.request.read(1024)

            if recv_msg == '':
                self.stop.set()
                server.recv_queue.put(recv_msg)
                break
            else:
                server.recv_queue.put(recv_msg)

        if self.stopped():
            print '[!] Received STOP signal from %s:%s; Exiting!' % (self.client_address[0], self.client_address[1])

Во-вторых, это код клиента, в котором я настроил информацию, необходимую для подключения через прокси-сервер, который требует аутентификации:

class proxyCommsHandler():

    def __init__(self, user, password, remote_host, remote_port, list_of_proxies):
        # information needed to connect
        self.user = 'username'
        self.passwd = 'password'
        self.remote_host = 'remote_host_ip'
        self.remote_port = 8008
        self.list_of_proxies = [['proxyserver.hostname.com', 8080]]

        # setup basic authentication to send to the proxy when we try to connect
        self.user_pass = base64.encodestring(self.user + ':' + self.passwd)
        self.proxy_authorization = 'Proxy-authorization: Basic ' + self.user_pass + '\r\n'
        self.proxy_connect = 'CONNECT %s:%s HTTP/1.1\r\n' % (self.remote_host, self.remote_port)
        self.user_agent = "User-Agent: Mozilla/5.0 (Windows NT 6.1; rv:7.0.1) Gecko/20100101 Firefox/7.0.1\r\n"
        self.proxy_pieces = self.proxy_connect + self.proxy_authorization + self.user_agent + '\r\n'

Теперь вот где я первоначально подключаюсь к прокси, где я получаю никаких ошибок (я получаю код состояния '200'):

self.proxy = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self.proxy.connect( (proxy_host, proxy_port) )
self.proxy.sendall(self.proxy_pieces)
self.response = proxy.recv(1024)

Здесь клиент терпит неудачу (я думаю). Я пытаюсь взять self.proxy и обернуть его в SSL, например:

sslsock = ssl.wrap_socket(self.proxy, server_side=False, do_handshake_on_connect=True,
                                      ssl_version=ssl.PROTOCOL_TLSv1)

Это ошибка, которую я вижу на клиенте:

Traceback (most recent call last):
    File "C:\Python27\pyrevshell.py", line 467, in <module>
        proxyCommsHandler(None, None, None, None, list_of_proxies).run()
    File "C:\Python27\pyrevshell.py", line 300, in run
        ssl_version=ssl.PROTOCOL_TLSv1)
    File "C:\Python27\lib\ssl.py", line 372, in wrap_socket
        ciphers=ciphers)
    File "C:\Python27\lib\ssl.py", line 134, in __init__
        self.do_handshake()
    File "C:\Python27\lib\ssl.py", line 296, in do_handshake
        self._sslobj.do_handshake()
    SSLError: [Errno 8] _ssl.c:503: EOF occurred in violation of protocol

Клиент подключается , как показано здесь:

trying to wrap in ssl
[*] Received connection from x.x.x.x:47144
[*] x.x.x.x:47144 added to the client list

Но затем сразу следует исключение:

----------------------------------------
Exception happened during processing of request from ('x.x.x.x', 47144)
Traceback (most recent call last):
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/SocketServer.py", line 582, in process_request_thread
    self.finish_request(request, client_address)
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/SocketServer.py", line 323, in finish_request
    self.RequestHandlerClass(request, client_address, self)
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/SocketServer.py", line 639, in __init__
      self.handle()
   File "shell_server.py", line 324, in handle
      recv_msg = self.request.read(1024)
   File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/ssl.py", line 138, in read
      return self._sslobj.read(len)
SSLError: [Errno 1] _ssl.c:1348: error:1408F10B:SSL routines:SSL3_GET_RECORD:wrong version number
----------------------------------------

Хотя я понимаю, что это звучит как очевидная проблема, основанная на исключениях, которые были брошены, интересные моменты таковы:

  • Сначала я успешно подключился через прокси, как показано выше
  • Я могу успешно подключиться через веб-браузер за за тем же прокси , и никаких исключений не выдается; Я могу передать данные в браузер
  • Я пробовал разные версии протокола SSL как на стороне сервера, так и на стороне клиента, как показано в таблице здесь в документации по Python ; это ошибки на стороне клиента каждый раз

Я использовал Wireshark на обоих концах соединения. При использовании обычного браузера и подключении к серверу я вижу весь процесс рукопожатия и согласования SSL, и все идет гладко.

Однако, когда я использую клиент, показанный выше, как только я соединяюсь с ним, я вижу, что клиент отправляет сообщение Client Hello, но затем мой сервер отправляет пакет RST, чтобы разорвать соединение (у меня нет t определяется, будет ли это до или после того, как будет сгенерировано исключение).

Снова, я прошу прощения за длину, но я остро нуждаюсь в совете эксперта.

1 Ответ

1 голос
/ 25 ноября 2011

Я понял проблему с моей проблемой.Я отправляю self.user_agent на удаленный хост при первом подключении через прокси-сервер, что мешает SSL рукопожатию.

Чтобы решить эту проблему, я поставил начальную self.request.recv() в def setup(self) функция до я вызываю ssl.wrap_socket на сокете.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...