35
43
self.credentials.append([realm, host, username, password])
46
class RecordingServer(object):
47
"""A fake HTTP server.
49
It records the bytes sent to it, and replies with a 200.
52
def __init__(self, expect_body_tail=None):
55
:type expect_body_tail: str
56
:param expect_body_tail: a reply won't be sent until this string is
59
self._expect_body_tail = expect_body_tail
62
self.received_bytes = ''
65
self._sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
66
self._sock.bind(('127.0.0.1', 0))
67
self.host, self.port = self._sock.getsockname()
68
self._ready = threading.Event()
69
self._thread = threading.Thread(target=self._accept_read_and_reply)
70
self._thread.setDaemon(True)
74
def _accept_read_and_reply(self):
77
self._sock.settimeout(5)
79
conn, address = self._sock.accept()
80
# On win32, the accepted connection will be non-blocking to start
81
# with because we're using settimeout.
82
conn.setblocking(True)
83
while not self.received_bytes.endswith(self._expect_body_tail):
84
self.received_bytes += conn.recv(4096)
85
conn.sendall('HTTP/1.1 200 OK\r\n')
86
except socket.timeout:
87
# Make sure the client isn't stuck waiting for us to e.g. accept.
94
# We might have already closed it. We don't care.
38
100
class TestHttpUrls(TestCase):
39
102
def test_url_parsing(self):
41
104
url = extract_auth('http://example.com', f)
42
105
self.assertEquals('http://example.com', url)
43
106
self.assertEquals(0, len(f.credentials))
44
url = extract_auth('http://user:pass@www.bazaar-ng.org/bzr/bzr.dev', f)
45
self.assertEquals('http://www.bazaar-ng.org/bzr/bzr.dev', url)
107
url = extract_auth('http://user:pass@www.bazaar-vcs.org/bzr/bzr.dev', f)
108
self.assertEquals('http://www.bazaar-vcs.org/bzr/bzr.dev', url)
46
109
self.assertEquals(1, len(f.credentials))
47
self.assertEquals([None, 'www.bazaar-ng.org', 'user', 'pass'], f.credentials[0])
110
self.assertEquals([None, 'www.bazaar-vcs.org', 'user', 'pass'], f.credentials[0])
49
112
def test_abs_url(self):
50
113
"""Construction of absolute http URLs"""
51
t = HttpTransport_urllib('http://bazaar-ng.org/bzr/bzr.dev/')
114
t = HttpTransport_urllib('http://bazaar-vcs.org/bzr/bzr.dev/')
52
115
eq = self.assertEqualDiff
53
116
eq(t.abspath('.'),
54
'http://bazaar-ng.org/bzr/bzr.dev')
117
'http://bazaar-vcs.org/bzr/bzr.dev')
55
118
eq(t.abspath('foo/bar'),
56
'http://bazaar-ng.org/bzr/bzr.dev/foo/bar')
119
'http://bazaar-vcs.org/bzr/bzr.dev/foo/bar')
57
120
eq(t.abspath('.bzr'),
58
'http://bazaar-ng.org/bzr/bzr.dev/.bzr')
121
'http://bazaar-vcs.org/bzr/bzr.dev/.bzr')
59
122
eq(t.abspath('.bzr/1//2/./3'),
60
'http://bazaar-ng.org/bzr/bzr.dev/.bzr/1/2/3')
123
'http://bazaar-vcs.org/bzr/bzr.dev/.bzr/1/2/3')
62
125
def test_invalid_http_urls(self):
63
126
"""Trap invalid construction of urls"""
64
t = HttpTransport_urllib('http://bazaar-ng.org/bzr/bzr.dev/')
127
t = HttpTransport_urllib('http://bazaar-vcs.org/bzr/bzr.dev/')
65
128
self.assertRaises(ValueError,
68
self.assertRaises(ValueError,
72
132
def test_http_root_urls(self):
73
133
"""Construction of URLs from server root"""
122
183
self.assertTrue(server.logs[0].find(
123
184
'"GET /foo/bar HTTP/1.1" 200 - "-" "bzr/%s' % bzrlib.__version__) > -1)
186
def test_get_smart_medium(self):
187
# For HTTP, get_smart_medium should return the transport object.
188
server = self.get_readonly_server()
189
http_transport = self._transport(server.get_url())
190
medium = http_transport.get_smart_medium()
191
self.assertIs(medium, http_transport)
126
194
class TestHttpConnections_urllib(TestCaseWithWebserver, TestHttpMixins):
127
196
_transport = HttpTransport_urllib
130
199
TestCaseWithWebserver.setUp(self)
131
200
self._prep_tree()
202
def test_has_on_bogus_host(self):
204
# Get a random address, so that we can be sure there is no
205
# http handler there.
207
s.bind(('localhost', 0))
208
t = self._transport('http://%s:%s/' % s.getsockname())
209
self.assertRaises(urllib2.URLError, t.has, 'foo/bar')
135
212
class TestHttpConnections_pycurl(TestCaseWithWebserver, TestHttpMixins):
159
235
t = get_transport('http+urllib://bzr.google.com/')
160
236
self.assertIsInstance(t, Transport)
161
237
self.assertIsInstance(t, bzrlib.transport.http._urllib.HttpTransport_urllib)
240
class TestOffsets(TestCase):
241
"""Test offsets_to_ranges method"""
243
def test_offsets_to_ranges_simple(self):
244
to_range = HttpTransportBase.offsets_to_ranges
245
ranges = to_range([(10, 1)])
246
self.assertEqual([[10, 10]], ranges)
248
ranges = to_range([(0, 1), (1, 1)])
249
self.assertEqual([[0, 1]], ranges)
251
ranges = to_range([(1, 1), (0, 1)])
252
self.assertEqual([[0, 1]], ranges)
254
def test_offset_to_ranges_overlapped(self):
255
to_range = HttpTransportBase.offsets_to_ranges
257
ranges = to_range([(10, 1), (20, 2), (22, 5)])
258
self.assertEqual([[10, 10], [20, 26]], ranges)
260
ranges = to_range([(10, 1), (11, 2), (22, 5)])
261
self.assertEqual([[10, 12], [22, 26]], ranges)
264
class TestPost(TestCase):
266
def _test_post_body_is_received(self, scheme):
267
server = RecordingServer(expect_body_tail='end-of-body')
269
self.addCleanup(server.tearDown)
270
url = '%s://%s:%s/' % (scheme, server.host, server.port)
272
http_transport = get_transport(url)
273
except UnsupportedProtocol:
274
raise TestSkipped('%s not available' % scheme)
275
code, response = http_transport._post('abc def end-of-body')
277
server.received_bytes.startswith('POST /.bzr/smart HTTP/1.'))
278
self.assertTrue('content-length: 19\r' in server.received_bytes.lower())
279
# The transport should not be assuming that the server can accept
280
# chunked encoding the first time it connects, because HTTP/1.1, so we
281
# check for the literal string.
283
server.received_bytes.endswith('\r\n\r\nabc def end-of-body'))
285
def test_post_body_is_received_urllib(self):
286
self._test_post_body_is_received('http+urllib')
288
def test_post_body_is_received_pycurl(self):
289
self._test_post_body_is_received('http+pycurl')
292
class TestRangeHeader(TestCase):
293
"""Test range_header method"""
295
def check_header(self, value, ranges=[], tail=0):
296
range_header = HttpTransportBase.range_header
297
self.assertEqual(value, range_header(ranges, tail))
299
def test_range_header_single(self):
300
self.check_header('0-9', ranges=[[0,9]])
301
self.check_header('100-109', ranges=[[100,109]])
303
def test_range_header_tail(self):
304
self.check_header('-10', tail=10)
305
self.check_header('-50', tail=50)
307
def test_range_header_multi(self):
308
self.check_header('0-9,100-200,300-5000',
309
ranges=[(0,9), (100, 200), (300,5000)])
311
def test_range_header_mixed(self):
312
self.check_header('0-9,300-5000,-50',
313
ranges=[(0,9), (300,5000)],
317
class TestRecordingServer(TestCase):
319
def test_create(self):
320
server = RecordingServer(expect_body_tail=None)
321
self.assertEqual('', server.received_bytes)
322
self.assertEqual(None, server.host)
323
self.assertEqual(None, server.port)
325
def test_setUp_and_tearDown(self):
326
server = RecordingServer(expect_body_tail=None)
329
self.assertNotEqual(None, server.host)
330
self.assertNotEqual(None, server.port)
333
self.assertEqual(None, server.host)
334
self.assertEqual(None, server.port)
336
def test_send_receive_bytes(self):
337
server = RecordingServer(expect_body_tail='c')
339
self.addCleanup(server.tearDown)
340
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
341
sock.connect((server.host, server.port))
343
self.assertEqual('HTTP/1.1 200 OK\r\n',
344
osutils.recv_all(sock, 4096))
345
self.assertEqual('abc', server.received_bytes)