4

PyOpenSSLを優先してSSL プロトコルをブロックするにはどうすればよいTLSですか? 私はCentOS 7これらのバージョンを使用しています:

pyOpenSSL-0.13.1-3.el7.x86_64
openssl-1.0.1e-34.el7_0.7.x86_64

私の設定ファイル(CherryPyアプリの場合)には次のものがあります:

'server.ssl_module': 'pyopenssl',
4

2 に答える 2

1

これは、今日の CherryPy にとって本当に良い質問です。今月は、 CherryPy ユーザー グループで、SSL の問題と、py2.6+sslおよび pyOpenSSL を介した CherryPy のラッパーの全体的な保守性についての議論を開始しました。そこで SSL の問題についてのトピックを計画しているので、グループに登録して後で詳細を入手できます。

今のところ、可能なことは次のとおりです。私は Debian Wheezy、Python 2.7.3-4+deb7u1、OpenSSL 1.0.1e-2+deb7u16 を持っていました。リポジトリから CherryPy (3.6 では SSL が壊れています) と pyOpenSSL 0.14 をインストールしました。Qualys SSL labs testでいくつかのポイントを獲得するために、両方の CherryPy SSL アダプターをオーバーライドしようとしました。これは非常に役に立ちます。デプロイをテストすることを強くお勧めします (フロントエンドが CherryPy であるかどうかに関係なく)。

その結果、sslベースのアダプターにはまだ脆弱性があり、py2 < 2.7.9 (大規模な SSL 更新) および py3 < 3.3 で回避する方法がわかりません。CherryPysslアダプターはこれらの変更よりずっと前に作成されたため、古い方法と新しい方法 (主にSSL コンテキスト) の両方をサポートするために書き直す必要があります。一方、サブクラス化された pyOpenSSL を適用すると、次の点を除いてほとんど問題ありません。

  • Secure Client-Initiated Renegotiation を有効にしました。OpenSSL に依存している可能性があります。
  • Forward Secrecyは役に立たなかったSSL.OP_SINGLE_DH_USEかもしれませんが、そうではありませんでした。OpenSSL のバージョンにも依存する場合があります。

これがコードです。

#!/usr/bin/env python
# -*- coding: utf-8 -*-


import os
import sys
import ssl

import cherrypy
from cherrypy.wsgiserver.ssl_builtin import BuiltinSSLAdapter
from cherrypy.wsgiserver.ssl_pyopenssl import pyOpenSSLAdapter

from cherrypy import wsgiserver
if sys.version_info < (3, 0):
  from cherrypy.wsgiserver.wsgiserver2 import ssl_adapters  
else:
  from cherrypy.wsgiserver.wsgiserver3 import ssl_adapters

try:
  from OpenSSL import SSL
except ImportError:
  pass


ciphers = (
  'ECDH+AESGCM:DH+AESGCM:ECDH+AES256:DH+AES256:ECDH+AES128:DH+AES:ECDH+HIGH:'
  'DH+HIGH:ECDH+3DES:DH+3DES:RSA+AESGCM:RSA+AES:RSA+HIGH:RSA+3DES:!aNULL:'
  '!eNULL:!MD5:!DSS:!RC4:!SSLv2'
)

bundle = os.path.join(os.path.dirname(cherrypy.__file__), 'test', 'test.pem')

config = {
  'global' : {
    'server.socket_host' : '127.0.0.1',
    'server.socket_port' : 8443,
    'server.thread_pool' : 8,

    'server.ssl_module'      : 'custom-pyopenssl',
    'server.ssl_certificate' : bundle,
    'server.ssl_private_key' : bundle,
  }
}


class BuiltinSsl(BuiltinSSLAdapter):
  '''Vulnerable, on py2 < 2.7.9, py3 < 3.3:
    * POODLE (SSLv3), adding ``!SSLv3`` to cipher list makes it very incompatible
    * can't disable TLS compression (CRIME)
    * supports Secure Client-Initiated Renegotiation (DOS)
    * no Forward Secrecy
  Also session caching doesn't work. Some tweaks are posslbe, but don't really 
  change much. For example, it's possible to use ssl.PROTOCOL_TLSv1 instead of 
  ssl.PROTOCOL_SSLv23 with little worse compatiblity.
  '''

  def wrap(self, sock):
    """Wrap and return the given socket, plus WSGI environ entries."""
    try:
      s = ssl.wrap_socket(
        sock, 
        ciphers = ciphers, # the override is for this line
        do_handshake_on_connect = True,
        server_side = True, 
        certfile = self.certificate,
        keyfile = self.private_key,
        ssl_version = ssl.PROTOCOL_SSLv23
      )
    except ssl.SSLError:
      e = sys.exc_info()[1]
      if e.errno == ssl.SSL_ERROR_EOF:
        # This is almost certainly due to the cherrypy engine
        # 'pinging' the socket to assert it's connectable;
        # the 'ping' isn't SSL.
        return None, {}
      elif e.errno == ssl.SSL_ERROR_SSL:
        if e.args[1].endswith('http request'):
          # The client is speaking HTTP to an HTTPS server.
          raise wsgiserver.NoSSLError
        elif e.args[1].endswith('unknown protocol'):
          # The client is speaking some non-HTTP protocol.
          # Drop the conn.
          return None, {}
      raise

    return s, self.get_environ(s)

ssl_adapters['custom-ssl'] = BuiltinSsl


class Pyopenssl(pyOpenSSLAdapter):
  '''Mostly fine, except:
    * Secure Client-Initiated Renegotiation
    * no Forward Secrecy, SSL.OP_SINGLE_DH_USE could have helped but it didn't
  '''

  def get_context(self):
    """Return an SSL.Context from self attributes."""
    c = SSL.Context(SSL.SSLv23_METHOD)

    # override:
    c.set_options(SSL.OP_NO_COMPRESSION | SSL.OP_SINGLE_DH_USE | SSL.OP_NO_SSLv2 | SSL.OP_NO_SSLv3)
    c.set_cipher_list(ciphers)

    c.use_privatekey_file(self.private_key)
    if self.certificate_chain:
        c.load_verify_locations(self.certificate_chain)
    c.use_certificate_file(self.certificate)
    return c

ssl_adapters['custom-pyopenssl'] = Pyopenssl


class App:

  @cherrypy.expose
  def index(self):
    return '<em>Is this secure?</em>'


if __name__ == '__main__':
  cherrypy.quickstart(App(), '/', config)

アップデート

ここでは、CherryPy の SSL サポートの将来を決定する必要がある記事ディスカッションを示します。

于 2015-03-27T15:12:32.853 に答える