blob: 867fcfa41e9cb7ca5b6ffe89a67e5e14d4145da3 [file] [log] [blame]
Jeremy Hylton5d9c3032004-08-07 17:40:50 +00001#!/usr/bin/env python
2
3import unittest
4from test import test_support
Georg Brandl1b06a1d2006-05-03 05:15:10 +00005from test.test_urllib2 import sanepathname2url
Jeremy Hylton5d9c3032004-08-07 17:40:50 +00006
7import socket
8import urllib2
Jeremy Hylton5d9c3032004-08-07 17:40:50 +00009import os
Senthil Kumaran281b5512010-04-20 06:54:59 +000010import sys
11
12TIMEOUT = 60 # seconds
Jeremy Hylton5d9c3032004-08-07 17:40:50 +000013
Neal Norwitz769d0ee2008-01-25 06:37:23 +000014
Facundo Batista6a5a1772008-06-07 13:36:36 +000015def _retry_thrice(func, exc, *args, **kwargs):
Neal Norwitz769d0ee2008-01-25 06:37:23 +000016 for i in range(3):
17 try:
Facundo Batista6a5a1772008-06-07 13:36:36 +000018 return func(*args, **kwargs)
19 except exc, last_exc:
Neal Norwitz769d0ee2008-01-25 06:37:23 +000020 continue
21 except:
22 raise
23 raise last_exc
24
Facundo Batista6a5a1772008-06-07 13:36:36 +000025def _wrap_with_retry_thrice(func, exc):
26 def wrapped(*args, **kwargs):
27 return _retry_thrice(func, exc, *args, **kwargs)
28 return wrapped
29
30# Connecting to remote hosts is flaky. Make it more robust by retrying
31# the connection several times.
32_urlopen_with_retry = _wrap_with_retry_thrice(urllib2.urlopen, urllib2.URLError)
Neal Norwitz769d0ee2008-01-25 06:37:23 +000033
Georg Brandlfa42bd72006-04-30 07:06:11 +000034
35class AuthTests(unittest.TestCase):
36 """Tests urllib2 authentication features."""
37
38## Disabled at the moment since there is no page under python.org which
39## could be used to HTTP authentication.
40#
41# def test_basic_auth(self):
42# import httplib
43#
44# test_url = "http://www.python.org/test/test_urllib2/basic_auth"
45# test_hostport = "www.python.org"
46# test_realm = 'Test Realm'
47# test_user = 'test.test_urllib2net'
48# test_password = 'blah'
49#
50# # failure
51# try:
Neal Norwitz769d0ee2008-01-25 06:37:23 +000052# _urlopen_with_retry(test_url)
Georg Brandlfa42bd72006-04-30 07:06:11 +000053# except urllib2.HTTPError, exc:
54# self.assertEqual(exc.code, 401)
55# else:
56# self.fail("urlopen() should have failed with 401")
57#
58# # success
59# auth_handler = urllib2.HTTPBasicAuthHandler()
60# auth_handler.add_password(test_realm, test_hostport,
61# test_user, test_password)
62# opener = urllib2.build_opener(auth_handler)
63# f = opener.open('http://localhost/')
Neal Norwitz769d0ee2008-01-25 06:37:23 +000064# response = _urlopen_with_retry("http://www.python.org/")
Georg Brandlfa42bd72006-04-30 07:06:11 +000065#
66# # The 'userinfo' URL component is deprecated by RFC 3986 for security
67# # reasons, let's not implement it! (it's already implemented for proxy
68# # specification strings (that is, URLs or authorities specifying a
69# # proxy), so we must keep that)
70# self.assertRaises(httplib.InvalidURL,
71# urllib2.urlopen, "http://evil:thing@example.com")
72
73
Georg Brandldd7b0522007-01-21 10:35:10 +000074class CloseSocketTest(unittest.TestCase):
75
76 def test_close(self):
Georg Brandla4f46e12010-02-07 17:03:15 +000077 import httplib
Georg Brandldd7b0522007-01-21 10:35:10 +000078
79 # calling .close() on urllib2's response objects should close the
80 # underlying socket
81
82 # delve deep into response to fetch socket._socketobject
Neal Norwitz769d0ee2008-01-25 06:37:23 +000083 response = _urlopen_with_retry("http://www.python.org/")
Georg Brandldd7b0522007-01-21 10:35:10 +000084 abused_fileobject = response.fp
Benjamin Peterson5c8da862009-06-30 22:57:08 +000085 self.assertTrue(abused_fileobject.__class__ is socket._fileobject)
Georg Brandldd7b0522007-01-21 10:35:10 +000086 httpresponse = abused_fileobject._sock
Benjamin Peterson5c8da862009-06-30 22:57:08 +000087 self.assertTrue(httpresponse.__class__ is httplib.HTTPResponse)
Georg Brandldd7b0522007-01-21 10:35:10 +000088 fileobject = httpresponse.fp
Benjamin Peterson5c8da862009-06-30 22:57:08 +000089 self.assertTrue(fileobject.__class__ is socket._fileobject)
Georg Brandldd7b0522007-01-21 10:35:10 +000090
Benjamin Peterson5c8da862009-06-30 22:57:08 +000091 self.assertTrue(not fileobject.closed)
Georg Brandldd7b0522007-01-21 10:35:10 +000092 response.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +000093 self.assertTrue(fileobject.closed)
Georg Brandldd7b0522007-01-21 10:35:10 +000094
Georg Brandl1b06a1d2006-05-03 05:15:10 +000095class OtherNetworkTests(unittest.TestCase):
96 def setUp(self):
97 if 0: # for debugging
98 import logging
99 logger = logging.getLogger("test_urllib2net")
100 logger.addHandler(logging.StreamHandler())
101
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000102 # XXX The rest of these tests aren't very good -- they don't check much.
103 # They do sometimes catch some major disasters, though.
104
105 def test_ftp(self):
106 urls = [
Gregory P. Smithe9fef692007-09-09 23:36:46 +0000107 'ftp://ftp.kernel.org/pub/linux/kernel/README',
Mark Dickinson3e4caeb2009-02-21 20:27:01 +0000108 'ftp://ftp.kernel.org/pub/linux/kernel/non-existent-file',
Gregory P. Smithe9fef692007-09-09 23:36:46 +0000109 #'ftp://ftp.kernel.org/pub/leenox/kernel/test',
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000110 'ftp://gatekeeper.research.compaq.com/pub/DEC/SRC'
111 '/research-reports/00README-Legal-Rules-Regs',
112 ]
113 self._test_urls(urls, self._extra_handlers())
114
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000115 def test_file(self):
116 TESTFN = test_support.TESTFN
117 f = open(TESTFN, 'w')
118 try:
119 f.write('hi there\n')
120 f.close()
121 urls = [
122 'file:'+sanepathname2url(os.path.abspath(TESTFN)),
Gregory P. Smithe9fef692007-09-09 23:36:46 +0000123 ('file:///nonsensename/etc/passwd', None, urllib2.URLError),
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000124 ]
Facundo Batista6a5a1772008-06-07 13:36:36 +0000125 self._test_urls(urls, self._extra_handlers(), retry=True)
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000126 finally:
127 os.remove(TESTFN)
128
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000129 # XXX Following test depends on machine configurations that are internal
130 # to CNRI. Need to set up a public server with the right authentication
131 # configuration for test purposes.
132
133## def test_cnri(self):
134## if socket.gethostname() == 'bitdiddle':
135## localhost = 'bitdiddle.cnri.reston.va.us'
136## elif socket.gethostname() == 'bitdiddle.concentric.net':
137## localhost = 'localhost'
138## else:
139## localhost = None
140## if localhost is not None:
141## urls = [
142## 'file://%s/etc/passwd' % localhost,
143## 'http://%s/simple/' % localhost,
144## 'http://%s/digest/' % localhost,
145## 'http://%s/not/found.h' % localhost,
146## ]
147
148## bauth = HTTPBasicAuthHandler()
149## bauth.add_password('basic_test_realm', localhost, 'jhylton',
150## 'password')
151## dauth = HTTPDigestAuthHandler()
152## dauth.add_password('digest_test_realm', localhost, 'jhylton',
153## 'password')
154
155## self._test_urls(urls, self._extra_handlers()+[bauth, dauth])
156
Senthil Kumaranb4ec7ee2010-08-08 11:43:45 +0000157 def test_urlwithfrag(self):
158 urlwith_frag = "http://docs.python.org/glossary.html#glossary"
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000159 with test_support.transient_internet(urlwith_frag):
160 req = urllib2.Request(urlwith_frag)
161 res = urllib2.urlopen(req)
162 self.assertEqual(res.geturl(),
Senthil Kumaran49c44082011-04-13 07:31:45 +0800163 "http://docs.python.org/glossary.html#glossary")
Senthil Kumaranb4ec7ee2010-08-08 11:43:45 +0000164
Senthil Kumarand389cb52010-09-21 01:38:15 +0000165 def test_fileno(self):
166 req = urllib2.Request("http://www.python.org")
167 opener = urllib2.build_opener()
168 res = opener.open(req)
169 try:
170 res.fileno()
171 except AttributeError:
172 self.fail("HTTPResponse object should return a valid fileno")
173 finally:
174 res.close()
175
Senthil Kumaran176c73d2010-09-27 01:40:59 +0000176 def test_custom_headers(self):
177 url = "http://www.example.com"
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000178 with test_support.transient_internet(url):
179 opener = urllib2.build_opener()
180 request = urllib2.Request(url)
181 self.assertFalse(request.header_items())
182 opener.open(request)
183 self.assertTrue(request.header_items())
184 self.assertTrue(request.has_header('User-agent'))
185 request.add_header('User-Agent','Test-Agent')
186 opener.open(request)
187 self.assertEqual(request.get_header('User-agent'),'Test-Agent')
Senthil Kumaran176c73d2010-09-27 01:40:59 +0000188
Facundo Batista6a5a1772008-06-07 13:36:36 +0000189 def _test_urls(self, urls, handlers, retry=True):
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000190 import time
191 import logging
192 debug = logging.getLogger("test_urllib2").debug
193
Facundo Batista6a5a1772008-06-07 13:36:36 +0000194 urlopen = urllib2.build_opener(*handlers).open
195 if retry:
196 urlopen = _wrap_with_retry_thrice(urlopen, urllib2.URLError)
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000197
198 for url in urls:
199 if isinstance(url, tuple):
200 url, req, expected_err = url
201 else:
202 req = expected_err = None
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000203 with test_support.transient_internet(url):
204 debug(url)
Senthil Kumaran281b5512010-04-20 06:54:59 +0000205 try:
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000206 f = urlopen(url, req, TIMEOUT)
207 except EnvironmentError as err:
208 debug(err)
209 if expected_err:
210 msg = ("Didn't get expected error(s) %s for %s %s, got %s: %s" %
211 (expected_err, url, req, type(err), err))
212 self.assertIsInstance(err, expected_err, msg)
213 except urllib2.URLError as err:
214 if isinstance(err[0], socket.timeout):
215 print >>sys.stderr, "<timeout: %s>" % url
216 continue
217 else:
218 raise
219 else:
220 try:
221 with test_support.transient_internet(url):
222 buf = f.read()
223 debug("read %d bytes" % len(buf))
224 except socket.timeout:
225 print >>sys.stderr, "<timeout: %s>" % url
226 f.close()
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000227 debug("******** next url coming up...")
228 time.sleep(0.1)
229
230 def _extra_handlers(self):
231 handlers = []
232
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000233 cfh = urllib2.CacheFTPHandler()
234 cfh.setTimeout(1)
235 handlers.append(cfh)
236
237 return handlers
238
Gregory P. Smith0001c2e2008-03-28 08:00:44 +0000239
Facundo Batista10951d52007-06-06 17:15:23 +0000240class TimeoutTest(unittest.TestCase):
241 def test_http_basic(self):
Facundo Batista4f1b1ed2008-05-29 16:39:26 +0000242 self.assertTrue(socket.getdefaulttimeout() is None)
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000243 url = "http://www.python.org"
244 with test_support.transient_internet(url, timeout=None):
245 u = _urlopen_with_retry(url)
246 self.assertTrue(u.fp._sock.fp._sock.gettimeout() is None)
Facundo Batista10951d52007-06-06 17:15:23 +0000247
Facundo Batista4f1b1ed2008-05-29 16:39:26 +0000248 def test_http_default_timeout(self):
249 self.assertTrue(socket.getdefaulttimeout() is None)
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000250 url = "http://www.python.org"
251 with test_support.transient_internet(url):
252 socket.setdefaulttimeout(60)
253 try:
254 u = _urlopen_with_retry(url)
255 finally:
256 socket.setdefaulttimeout(None)
257 self.assertEqual(u.fp._sock.fp._sock.gettimeout(), 60)
Facundo Batista4f1b1ed2008-05-29 16:39:26 +0000258
259 def test_http_no_timeout(self):
260 self.assertTrue(socket.getdefaulttimeout() is None)
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000261 url = "http://www.python.org"
262 with test_support.transient_internet(url):
263 socket.setdefaulttimeout(60)
264 try:
265 u = _urlopen_with_retry(url, timeout=None)
266 finally:
267 socket.setdefaulttimeout(None)
268 self.assertTrue(u.fp._sock.fp._sock.gettimeout() is None)
Facundo Batista10951d52007-06-06 17:15:23 +0000269
Facundo Batista4f1b1ed2008-05-29 16:39:26 +0000270 def test_http_timeout(self):
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000271 url = "http://www.python.org"
272 with test_support.transient_internet(url):
273 u = _urlopen_with_retry(url, timeout=120)
274 self.assertEqual(u.fp._sock.fp._sock.gettimeout(), 120)
Facundo Batista10951d52007-06-06 17:15:23 +0000275
Martin v. Löwis7bc26b92010-04-08 17:40:54 +0000276 FTP_HOST = "ftp://ftp.mirror.nl/pub/gnu/"
Neal Norwitz769d0ee2008-01-25 06:37:23 +0000277
Facundo Batista10951d52007-06-06 17:15:23 +0000278 def test_ftp_basic(self):
Facundo Batista4f1b1ed2008-05-29 16:39:26 +0000279 self.assertTrue(socket.getdefaulttimeout() is None)
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000280 with test_support.transient_internet(self.FTP_HOST, timeout=None):
281 u = _urlopen_with_retry(self.FTP_HOST)
282 self.assertTrue(u.fp.fp._sock.gettimeout() is None)
Facundo Batista10951d52007-06-06 17:15:23 +0000283
Facundo Batista4f1b1ed2008-05-29 16:39:26 +0000284 def test_ftp_default_timeout(self):
285 self.assertTrue(socket.getdefaulttimeout() is None)
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000286 with test_support.transient_internet(self.FTP_HOST):
287 socket.setdefaulttimeout(60)
288 try:
289 u = _urlopen_with_retry(self.FTP_HOST)
290 finally:
291 socket.setdefaulttimeout(None)
292 self.assertEqual(u.fp.fp._sock.gettimeout(), 60)
Facundo Batista4f1b1ed2008-05-29 16:39:26 +0000293
294 def test_ftp_no_timeout(self):
295 self.assertTrue(socket.getdefaulttimeout() is None)
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000296 with test_support.transient_internet(self.FTP_HOST):
297 socket.setdefaulttimeout(60)
298 try:
299 u = _urlopen_with_retry(self.FTP_HOST, timeout=None)
300 finally:
301 socket.setdefaulttimeout(None)
302 self.assertTrue(u.fp.fp._sock.gettimeout() is None)
Facundo Batista10951d52007-06-06 17:15:23 +0000303
Facundo Batista4f1b1ed2008-05-29 16:39:26 +0000304 def test_ftp_timeout(self):
Antoine Pitrou9f3f9c52010-10-31 13:58:00 +0000305 with test_support.transient_internet(self.FTP_HOST):
306 u = _urlopen_with_retry(self.FTP_HOST, timeout=60)
307 self.assertEqual(u.fp.fp._sock.gettimeout(), 60)
Facundo Batista10951d52007-06-06 17:15:23 +0000308
Georg Brandl1b06a1d2006-05-03 05:15:10 +0000309
Jeremy Hylton5d9c3032004-08-07 17:40:50 +0000310def test_main():
311 test_support.requires("network")
Gregory P. Smith0001c2e2008-03-28 08:00:44 +0000312 test_support.run_unittest(AuthTests,
Georg Brandldd7b0522007-01-21 10:35:10 +0000313 OtherNetworkTests,
314 CloseSocketTest,
Facundo Batista10951d52007-06-06 17:15:23 +0000315 TimeoutTest,
Georg Brandldd7b0522007-01-21 10:35:10 +0000316 )
Jeremy Hylton5d9c3032004-08-07 17:40:50 +0000317
318if __name__ == "__main__":
319 test_main()