|
|
925e6b |
|
|
|
925e6b |
# HG changeset patch
|
|
|
925e6b |
# User Benjamin Peterson <benjamin@python.org>
|
|
|
925e6b |
# Date 1416798122 21600
|
|
|
925e6b |
# Node ID fb83916c3ea12899569e88a7505469a90ab1f674
|
|
|
925e6b |
# Parent c84f36a5f556867c2ec50453dc879a500032d377
|
|
|
925e6b |
pep 476: verify certificates by default (#22417)
|
|
|
925e6b |
|
|
|
925e6b |
diff --git a/Doc/library/httplib.rst b/Doc/library/httplib.rst
|
|
|
925e6b |
--- a/Doc/library/httplib.rst
|
|
|
925e6b |
+++ b/Doc/library/httplib.rst
|
|
|
925e6b |
@@ -90,9 +90,6 @@ The module provides the following classe
|
|
|
925e6b |
server's certificate. If you want to change that behaviour, you can
|
|
|
925e6b |
explicitly set *check_hostname* to False.
|
|
|
925e6b |
|
|
|
925e6b |
- .. warning::
|
|
|
925e6b |
- This does not do any verification of the server's certificate.
|
|
|
925e6b |
-
|
|
|
925e6b |
.. versionadded:: 2.0
|
|
|
925e6b |
|
|
|
925e6b |
.. versionchanged:: 2.6
|
|
|
925e6b |
@@ -104,6 +101,11 @@ The module provides the following classe
|
|
|
925e6b |
.. versionchanged:: 2.7.9
|
|
|
925e6b |
*context* and *check_hostname* was added.
|
|
|
925e6b |
|
|
|
925e6b |
+ This class now performs all the necessary certificate and hostname checks
|
|
|
925e6b |
+ by default. To revert to the previous, unverified, behavior
|
|
|
925e6b |
+ :func:`ssl._create_unverified_context` can be passed to the *context*
|
|
|
925e6b |
+ parameter.
|
|
|
925e6b |
+
|
|
|
925e6b |
|
|
|
925e6b |
.. class:: HTTPResponse(sock, debuglevel=0, strict=0)
|
|
|
925e6b |
|
|
|
925e6b |
diff --git a/Lib/httplib.py b/Lib/httplib.py
|
|
|
925e6b |
--- a/Lib/httplib.py
|
|
|
925e6b |
+++ b/Lib/httplib.py
|
|
|
925e6b |
@@ -1193,7 +1193,7 @@ else:
|
|
|
925e6b |
self.key_file = key_file
|
|
|
925e6b |
self.cert_file = cert_file
|
|
|
925e6b |
if context is None:
|
|
|
925e6b |
- context = ssl.create_default_context()
|
|
|
925e6b |
+ context = ssl._create_default_https_context()
|
|
|
925e6b |
will_verify = context.verify_mode != ssl.CERT_NONE
|
|
|
925e6b |
if check_hostname is None:
|
|
|
925e6b |
check_hostname = will_verify
|
|
|
925e6b |
diff --git a/Lib/ssl.py b/Lib/ssl.py
|
|
|
925e6b |
--- a/Lib/ssl.py
|
|
|
925e6b |
+++ b/Lib/ssl.py
|
|
|
925e6b |
@@ -427,8 +427,7 @@ def create_default_context(purpose=Purpo
|
|
|
925e6b |
context.load_default_certs(purpose)
|
|
|
925e6b |
return context
|
|
|
925e6b |
|
|
|
925e6b |
-
|
|
|
925e6b |
-def _create_stdlib_context(protocol=PROTOCOL_SSLv23, cert_reqs=None,
|
|
|
925e6b |
+def _create_unverified_context(protocol=PROTOCOL_SSLv23, cert_reqs=None,
|
|
|
925e6b |
check_hostname=False, purpose=Purpose.SERVER_AUTH,
|
|
|
925e6b |
certfile=None, keyfile=None,
|
|
|
925e6b |
cafile=None, capath=None, cadata=None):
|
|
|
925e6b |
@@ -469,6 +468,14 @@ def _create_stdlib_context(protocol=PROT
|
|
|
925e6b |
|
|
|
925e6b |
return context
|
|
|
925e6b |
|
|
|
925e6b |
+# Used by http.client if no context is explicitly passed.
|
|
|
925e6b |
+_create_default_https_context = create_default_context
|
|
|
925e6b |
+
|
|
|
925e6b |
+
|
|
|
925e6b |
+# Backwards compatibility alias, even though it's not a public name.
|
|
|
925e6b |
+_create_stdlib_context = _create_unverified_context
|
|
|
925e6b |
+
|
|
|
925e6b |
+
|
|
|
925e6b |
class SSLSocket(socket):
|
|
|
925e6b |
"""This class implements a subtype of socket.socket that wraps
|
|
|
925e6b |
the underlying OS socket in an SSL context when necessary, and
|
|
|
925e6b |
diff --git a/Lib/test/test_httplib.py b/Lib/test/test_httplib.py
|
|
|
925e6b |
--- a/Lib/test/test_httplib.py
|
|
|
925e6b |
+++ b/Lib/test/test_httplib.py
|
|
|
925e6b |
@@ -1,10 +1,9 @@
|
|
|
925e6b |
import httplib
|
|
|
925e6b |
import array
|
|
|
925e6b |
-import httplib
|
|
|
925e6b |
-import os
|
|
|
925e6b |
import StringIO
|
|
|
925e6b |
import socket
|
|
|
925e6b |
import errno
|
|
|
925e6b |
+import os
|
|
|
925e6b |
|
|
|
925e6b |
import unittest
|
|
|
925e6b |
TestCase = unittest.TestCase
|
|
|
925e6b |
diff --git a/Lib/test/test_urllib2_localnet.py b/Lib/test/test_urllib2_localnet.py
|
|
|
925e6b |
--- a/Lib/test/test_urllib2_localnet.py
|
|
|
925e6b |
+++ b/Lib/test/test_urllib2_localnet.py
|
|
|
925e6b |
@@ -5,6 +5,7 @@ import urllib2
|
|
|
925e6b |
import BaseHTTPServer
|
|
|
925e6b |
import unittest
|
|
|
925e6b |
import hashlib
|
|
|
925e6b |
+import ssl
|
|
|
925e6b |
|
|
|
925e6b |
from test import test_support
|
|
|
925e6b |
|
|
|
925e6b |
@@ -562,15 +563,37 @@ class TestUrlopen(BaseTestCase):
|
|
|
925e6b |
cafile=CERT_localhost)
|
|
|
925e6b |
self.assertEqual(data, b"we care a bit")
|
|
|
925e6b |
# Bad cert
|
|
|
925e6b |
- with self.assertRaises(urllib2.URLError) as cm:
|
|
|
925e6b |
+ with self.assertRaises(urllib2.URLError):
|
|
|
925e6b |
self.urlopen("https://localhost:%s/bizarre" % handler.port,
|
|
|
925e6b |
cafile=CERT_fakehostname)
|
|
|
925e6b |
# Good cert, but mismatching hostname
|
|
|
925e6b |
handler = self.start_https_server(certfile=CERT_fakehostname)
|
|
|
925e6b |
- with self.assertRaises(ssl.CertificateError) as cm:
|
|
|
925e6b |
+ with self.assertRaises(ssl.CertificateError):
|
|
|
925e6b |
self.urlopen("https://localhost:%s/bizarre" % handler.port,
|
|
|
925e6b |
cafile=CERT_fakehostname)
|
|
|
925e6b |
|
|
|
925e6b |
+ def test_https_with_cadefault(self):
|
|
|
925e6b |
+ handler = self.start_https_server(certfile=CERT_localhost)
|
|
|
925e6b |
+ # Self-signed cert should fail verification with system certificate store
|
|
|
925e6b |
+ with self.assertRaises(urllib2.URLError):
|
|
|
925e6b |
+ self.urlopen("https://localhost:%s/bizarre" % handler.port,
|
|
|
925e6b |
+ cadefault=True)
|
|
|
925e6b |
+
|
|
|
925e6b |
+ def test_https_sni(self):
|
|
|
925e6b |
+ if ssl is None:
|
|
|
925e6b |
+ self.skipTest("ssl module required")
|
|
|
925e6b |
+ if not ssl.HAS_SNI:
|
|
|
925e6b |
+ self.skipTest("SNI support required in OpenSSL")
|
|
|
925e6b |
+ sni_name = [None]
|
|
|
925e6b |
+ def cb_sni(ssl_sock, server_name, initial_context):
|
|
|
925e6b |
+ sni_name[0] = server_name
|
|
|
925e6b |
+ context = ssl.SSLContext(ssl.PROTOCOL_TLSv1)
|
|
|
925e6b |
+ context.set_servername_callback(cb_sni)
|
|
|
925e6b |
+ handler = self.start_https_server(context=context, certfile=CERT_localhost)
|
|
|
925e6b |
+ context = ssl.create_default_context(cafile=CERT_localhost)
|
|
|
925e6b |
+ self.urlopen("https://localhost:%s" % handler.port, context=context)
|
|
|
925e6b |
+ self.assertEqual(sni_name[0], "localhost")
|
|
|
925e6b |
+
|
|
|
925e6b |
def test_sending_headers(self):
|
|
|
925e6b |
handler = self.start_server([(200, [], "we don't care")])
|
|
|
925e6b |
|
|
|
925e6b |
diff -up Python-2.7.5/Doc/library/xmlrpclib.rst.ver Python-2.7.5/Doc/library/xmlrpclib.rst
|
|
|
925e6b |
--- Python-2.7.5/Doc/library/xmlrpclib.rst.ver 2015-03-30 13:59:29.243493601 +0200
|
|
|
925e6b |
+++ Python-2.7.5/Doc/library/xmlrpclib.rst 2015-03-30 14:03:40.509532180 +0200
|
|
|
925e6b |
@@ -34,6 +34,10 @@ between conformable Python objects and X
|
|
|
925e6b |
constructed data. If you need to parse untrusted or unauthenticated data see
|
|
|
925e6b |
:ref:`xml-vulnerabilities`.
|
|
|
925e6b |
|
|
|
925e6b |
+.. versionchanged:: 2.7.9
|
|
|
925e6b |
+
|
|
|
925e6b |
+ For https URIs, :mod:`xmlrpclib` now performs all the necessary certificate
|
|
|
925e6b |
+ and hostname checks by default
|
|
|
925e6b |
|
|
|
925e6b |
.. class:: ServerProxy(uri[, transport[, encoding[, verbose[, allow_none[, use_datetime]]]]])
|
|
|
925e6b |
|