Kaydet (Commit) 47fff870 authored tarafından Senthil Kumaran's avatar Senthil Kumaran

Merged revisions 76908 via svnmerge from

svn+ssh://pythondev@svn.python.org/python/trunk

........
  r76908 | senthil.kumaran | 2009-12-20 11:35:13 +0530 (Sun, 20 Dec 2009) | 4 lines

  Fix for issue 7291 - urllib2 cannot handle https with proxy requiring auth
  Refactored HTTPHandler tests and added testcase for proxy authorization.
........
üst 0639d5ae
...@@ -648,9 +648,18 @@ class HTTPConnection: ...@@ -648,9 +648,18 @@ class HTTPConnection:
if strict is not None: if strict is not None:
self.strict = strict self.strict = strict
def set_tunnel(self, host, port=None): def set_tunnel(self, host, port=None, headers=None):
""" Sets up the host and the port for the HTTP CONNECT Tunnelling.
The headers argument should be a mapping of extra HTTP headers
to send with the CONNECT request.
"""
self._tunnel_host = host self._tunnel_host = host
self._tunnel_port = port self._tunnel_port = port
if headers:
self._tunnel_headers = headers
else:
self._tunnel_headers.clear()
def _set_hostport(self, host, port): def _set_hostport(self, host, port):
if port is None: if port is None:
...@@ -674,12 +683,18 @@ class HTTPConnection: ...@@ -674,12 +683,18 @@ class HTTPConnection:
def _tunnel(self): def _tunnel(self):
self._set_hostport(self._tunnel_host, self._tunnel_port) self._set_hostport(self._tunnel_host, self._tunnel_port)
connect_str = "CONNECT %s:%d HTTP/1.0\r\n\r\n" %(self.host, self.port) connect_str = "CONNECT %s:%d HTTP/1.0\r\n" %(self.host, self.port)
connect_bytes = connect_str.encode("ascii") connect_bytes = connect_str.encode("ascii")
self.send(connect_bytes) self.send(connect_bytes)
for header, value in self._tunnel_headers.iteritems():
header_str = "%s: %s\r\n" % (header, value)
header_bytes = header_str.encode("ascii")
self.send(header_bytes)
response = self.response_class(self.sock, strict = self.strict, response = self.response_class(self.sock, strict = self.strict,
method= self._method) method = self._method)
(version, code, message) = response._read_status() (version, code, message) = response._read_status()
if code != 200: if code != 200:
self.close() self.close()
raise socket.error("Tunnel connection failed: %d %s" % (code, raise socket.error("Tunnel connection failed: %d %s" % (code,
......
...@@ -259,6 +259,61 @@ class FakeMethod: ...@@ -259,6 +259,61 @@ class FakeMethod:
def __call__(self, *args): def __call__(self, *args):
return self.handle(self.meth_name, self.action, *args) return self.handle(self.meth_name, self.action, *args)
class MockHTTPResponse(io.IOBase):
def __init__(self, fp, msg, status, reason):
self.fp = fp
self.msg = msg
self.status = status
self.reason = reason
self.code = 200
def read(self):
return ''
def info(self):
return {}
def geturl(self):
return self.url
class MockHTTPClass:
def __init__(self):
self.level = 0
self.req_headers = []
self.data = None
self.raise_on_endheaders = False
self._tunnel_headers = {}
def __call__(self, host, timeout=socket._GLOBAL_DEFAULT_TIMEOUT):
self.host = host
self.timeout = timeout
return self
def set_debuglevel(self, level):
self.level = level
def set_tunnel(self, host, port=None, headers=None):
self._tunnel_host = host
self._tunnel_port = port
if headers:
self._tunnel_headers = headers
else:
self._tunnel_headers.clear()
def request(self, method, url, body=None, headers={}):
self.method = method
self.selector = url
self.req_headers += headers.items()
self.req_headers.sort()
if body:
self.data = body
if self.raise_on_endheaders:
import socket
raise socket.error()
def getresponse(self):
return MockHTTPResponse(MockFile(), {}, 200, "OK")
class MockHandler: class MockHandler:
# useful for testing handler machinery # useful for testing handler machinery
# see add_ordered_mock_handlers() docstring # see add_ordered_mock_handlers() docstring
...@@ -366,6 +421,13 @@ class MockHTTPHandler(urllib.request.BaseHandler): ...@@ -366,6 +421,13 @@ class MockHTTPHandler(urllib.request.BaseHandler):
msg = email.message_from_string("\r\n\r\n") msg = email.message_from_string("\r\n\r\n")
return MockResponse(200, "OK", msg, "", req.get_full_url()) return MockResponse(200, "OK", msg, "", req.get_full_url())
class MockHTTPSHandler(urllib.request.AbstractHTTPHandler):
# Useful for testing the Proxy-Authorization request by verifying the
# properties of httpcon
httpconn = MockHTTPClass()
def https_open(self, req):
return self.do_open(self.httpconn, req)
class MockPasswordManager: class MockPasswordManager:
def add_password(self, realm, uri, user, password): def add_password(self, realm, uri, user, password):
self.realm = realm self.realm = realm
...@@ -677,43 +739,6 @@ class HandlerTests(unittest.TestCase): ...@@ -677,43 +739,6 @@ class HandlerTests(unittest.TestCase):
self.assertEqual(req.type, "ftp") self.assertEqual(req.type, "ftp")
def test_http(self): def test_http(self):
class MockHTTPResponse(io.IOBase):
def __init__(self, fp, msg, status, reason):
self.fp = fp
self.msg = msg
self.status = status
self.reason = reason
self.code = 200
def read(self):
return ''
def info(self):
return {}
def geturl(self):
return self.url
class MockHTTPClass:
def __init__(self):
self.level = 0
self.req_headers = []
self.data = None
self.raise_on_endheaders = False
def __call__(self, host, timeout=socket._GLOBAL_DEFAULT_TIMEOUT):
self.host = host
self.timeout = timeout
return self
def set_debuglevel(self, level):
self.level = level
def request(self, method, url, body=None, headers={}):
self.method = method
self.selector = url
self.req_headers += headers.items()
self.req_headers.sort()
if body:
self.data = body
if self.raise_on_endheaders:
import socket
raise socket.error()
def getresponse(self):
return MockHTTPResponse(MockFile(), {}, 200, "OK")
h = urllib.request.AbstractHTTPHandler() h = urllib.request.AbstractHTTPHandler()
o = h.parent = MockOpener() o = h.parent = MockOpener()
...@@ -979,6 +1004,28 @@ class HandlerTests(unittest.TestCase): ...@@ -979,6 +1004,28 @@ class HandlerTests(unittest.TestCase):
self.assertEqual([(handlers[0], "https_open")], self.assertEqual([(handlers[0], "https_open")],
[tup[0:2] for tup in o.calls]) [tup[0:2] for tup in o.calls])
def test_proxy_https_proxy_authorization(self):
o = OpenerDirector()
ph = urllib.request.ProxyHandler(dict(https='proxy.example.com:3128'))
o.add_handler(ph)
https_handler = MockHTTPSHandler()
o.add_handler(https_handler)
req = Request("https://www.example.com/")
req.add_header("Proxy-Authorization","FooBar")
req.add_header("User-Agent","Grail")
self.assertEqual(req.get_host(), "www.example.com")
self.assertIsNone(req._tunnel_host)
r = o.open(req)
# Verify Proxy-Authorization gets tunneled to request.
# httpsconn req_headers do not have the Proxy-Authorization header but
# the req will have.
self.assertFalse(("Proxy-Authorization","FooBar") in
https_handler.httpconn.req_headers)
self.assertTrue(("User-Agent","Grail") in
https_handler.httpconn.req_headers)
self.assertIsNotNone(req._tunnel_host)
self.assertEqual(req.get_host(), "proxy.example.com:3128")
self.assertEqual(req.get_header("Proxy-authorization"),"FooBar")
def test_basic_auth(self, quote_char='"'): def test_basic_auth(self, quote_char='"'):
opener = OpenerDirector() opener = OpenerDirector()
......
...@@ -31,8 +31,8 @@ install_opener -- Installs a new opener as the default opener. ...@@ -31,8 +31,8 @@ install_opener -- Installs a new opener as the default opener.
objects of interest: objects of interest:
OpenerDirector -- Sets up the User-Agent as the Python-urllib and manages the OpenerDirector -- Sets up the User Agent as the Python-urllib client and manages
Handler classes while dealing with both requests and responses. the Handler classes, while dealing with requests and responses.
Request -- An object that encapsulates the state of a request. The Request -- An object that encapsulates the state of a request. The
state can be as simple as the URL. It can also include extra HTTP state can be as simple as the URL. It can also include extra HTTP
...@@ -1059,7 +1059,14 @@ class AbstractHTTPHandler(BaseHandler): ...@@ -1059,7 +1059,14 @@ class AbstractHTTPHandler(BaseHandler):
headers = dict((name.title(), val) for name, val in headers.items()) headers = dict((name.title(), val) for name, val in headers.items())
if req._tunnel_host: if req._tunnel_host:
h.set_tunnel(req._tunnel_host) tunnel_headers = {}
proxy_auth_hdr = "Proxy-Authorization"
if proxy_auth_hdr in headers:
tunnel_headers[proxy_auth_hdr] = headers[proxy_auth_hdr]
# Proxy-Authorization should not be sent to origin
# server.
del headers[proxy_auth_hdr]
h.set_tunnel(req._tunnel_host, headers=tunnel_headers)
try: try:
h.request(req.get_method(), req.selector, req.data, headers) h.request(req.get_method(), req.selector, req.data, headers)
......
...@@ -157,6 +157,9 @@ C-API ...@@ -157,6 +157,9 @@ C-API
Library Library
------- -------
- Issue #7231: urllib2 cannot handle https with proxy requiring auth. Patch by
Tatsuhiro Tsujikawa.
- Issue #4757: `zlib.compress` and other methods in the zlib module now - Issue #4757: `zlib.compress` and other methods in the zlib module now
raise a TypeError when given an `str` object (rather than a `bytes`-like raise a TypeError when given an `str` object (rather than a `bytes`-like
object). Patch by Victor Stinner and Florent Xicluna. object). Patch by Victor Stinner and Florent Xicluna.
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment