768
768
KeyboardInterrupt, server._serve_one_request, fake_protocol)
769
769
server_sock.close()
770
770
self.assertEqual('', client_sock.recv(1))
772
def build_protocol_pipe_like(self, bytes):
773
to_server = StringIO(bytes)
774
from_server = StringIO()
775
server = medium.SmartServerPipeStreamMedium(
776
to_server, from_server, None)
777
return server._build_protocol()
779
def build_protocol_socket(self, bytes):
780
server_sock, client_sock = self.portable_socket_pair()
781
server = medium.SmartServerSocketStreamMedium(
783
client_sock.sendall(bytes)
785
return server._build_protocol()
787
def assertProtocolOne(self, server_protocol):
788
# Use assertIs because assertIsInstance will wrongly pass
789
# SmartServerRequestProtocolTwo (because it subclasses
790
# SmartServerRequestProtocolOne).
792
type(server_protocol), protocol.SmartServerRequestProtocolOne)
794
def assertProtocolTwo(self, server_protocol):
795
self.assertIsInstance(
796
server_protocol, protocol.SmartServerRequestProtocolTwo)
798
def test_pipe_like_build_protocol_empty_bytes(self):
799
# Any empty request (i.e. no bytes) is detected as protocol version one.
800
server_protocol = self.build_protocol_pipe_like('')
801
self.assertProtocolOne(server_protocol)
803
def test_socket_like_build_protocol_empty_bytes(self):
804
# Any empty request (i.e. no bytes) is detected as protocol version one.
805
server_protocol = self.build_protocol_socket('')
806
self.assertProtocolOne(server_protocol)
808
def test_pipe_like_build_protocol_non_two(self):
809
# A request that doesn't start with "bzr request 2\n" is version one.
810
server_protocol = self.build_protocol_pipe_like('abc\n')
811
self.assertProtocolOne(server_protocol)
813
def test_socket_build_protocol_non_two(self):
814
# A request that doesn't start with "bzr request 2\n" is version one.
815
server_protocol = self.build_protocol_socket('abc\n')
816
self.assertProtocolOne(server_protocol)
818
def test_pipe_like_build_protocol_two(self):
819
# A request that starts with "bzr request 2\n" is version two.
820
server_protocol = self.build_protocol_pipe_like('bzr request 2\n')
821
self.assertProtocolTwo(server_protocol)
823
def test_socket_build_protocol_two(self):
824
# A request that starts with "bzr request 2\n" is version two.
825
server_protocol = self.build_protocol_socket('bzr request 2\n')
826
self.assertProtocolTwo(server_protocol)
773
829
class TestSmartTCPServer(tests.TestCase):
1442
1511
errors.ReadingCompleted, smart_protocol.read_body_bytes)
1514
class TestSmartProtocolTwo(TestSmartProtocol):
1515
"""Tests for the smart protocol version two.
1517
This test case is mostly the same as TestSmartProtocolOne.
1520
client_protocol_class = protocol.SmartClientRequestProtocolTwo
1521
server_protocol_class = protocol.SmartServerRequestProtocolTwo
1523
def test_construct_version_two_server_protocol(self):
1524
smart_protocol = protocol.SmartServerRequestProtocolTwo(None, None)
1525
self.assertEqual('', smart_protocol.excess_buffer)
1526
self.assertEqual('', smart_protocol.in_buffer)
1527
self.assertFalse(smart_protocol.has_dispatched)
1528
self.assertEqual(1, smart_protocol.next_read_size())
1530
def test_construct_version_two_client_protocol(self):
1531
# we can construct a client protocol from a client medium request
1533
client_medium = medium.SmartSimplePipesClientMedium(None, output)
1534
request = client_medium.get_request()
1535
client_protocol = protocol.SmartClientRequestProtocolTwo(request)
1537
def test_server_offset_serialisation(self):
1538
"""The Smart protocol serialises offsets as a comma and \n string.
1540
We check a number of boundary cases are as expected: empty, one offset,
1541
one with the order of reads not increasing (an out of order read), and
1542
one that should coalesce.
1544
self.assertOffsetSerialisation([], '', self.client_protocol)
1545
self.assertOffsetSerialisation([(1,2)], '1,2', self.client_protocol)
1546
self.assertOffsetSerialisation([(10,40), (0,5)], '10,40\n0,5',
1547
self.client_protocol)
1548
self.assertOffsetSerialisation([(1,2), (3,4), (100, 200)],
1549
'1,2\n3,4\n100,200', self.client_protocol)
1551
def test_accept_bytes_of_bad_request_to_protocol(self):
1552
out_stream = StringIO()
1553
smart_protocol = protocol.SmartServerRequestProtocolTwo(
1554
None, out_stream.write)
1555
smart_protocol.accept_bytes('abc')
1556
self.assertEqual('abc', smart_protocol.in_buffer)
1557
smart_protocol.accept_bytes('\n')
1559
protocol.RESPONSE_VERSION_TWO +
1560
"error\x01Generic bzr smart protocol error: bad request 'abc'\n",
1561
out_stream.getvalue())
1562
self.assertTrue(smart_protocol.has_dispatched)
1563
self.assertEqual(0, smart_protocol.next_read_size())
1565
def test_accept_body_bytes_to_protocol(self):
1566
protocol = self.build_protocol_waiting_for_body()
1567
self.assertEqual(6, protocol.next_read_size())
1568
protocol.accept_bytes('7\nabc')
1569
self.assertEqual(9, protocol.next_read_size())
1570
protocol.accept_bytes('defgd')
1571
protocol.accept_bytes('one\n')
1572
self.assertEqual(0, protocol.next_read_size())
1573
self.assertTrue(self.end_received)
1575
def test_accept_request_and_body_all_at_once(self):
1576
self._captureVar('BZR_NO_SMART_VFS', None)
1577
mem_transport = memory.MemoryTransport()
1578
mem_transport.put_bytes('foo', 'abcdefghij')
1579
out_stream = StringIO()
1580
smart_protocol = protocol.SmartServerRequestProtocolTwo(mem_transport,
1582
smart_protocol.accept_bytes('readv\x01foo\n3\n3,3done\n')
1583
self.assertEqual(0, smart_protocol.next_read_size())
1584
self.assertEqual(protocol.RESPONSE_VERSION_TWO + 'readv\n3\ndefdone\n',
1585
out_stream.getvalue())
1586
self.assertEqual('', smart_protocol.excess_buffer)
1587
self.assertEqual('', smart_protocol.in_buffer)
1589
def test_accept_excess_bytes_are_preserved(self):
1590
out_stream = StringIO()
1591
smart_protocol = protocol.SmartServerRequestProtocolTwo(
1592
None, out_stream.write)
1593
smart_protocol.accept_bytes('hello\nhello\n')
1594
self.assertEqual(protocol.RESPONSE_VERSION_TWO + "ok\x012\n",
1595
out_stream.getvalue())
1596
self.assertEqual("hello\n", smart_protocol.excess_buffer)
1597
self.assertEqual("", smart_protocol.in_buffer)
1599
def test_accept_excess_bytes_after_body(self):
1600
# The excess bytes look like the start of another request.
1601
server_protocol = self.build_protocol_waiting_for_body()
1602
server_protocol.accept_bytes(
1603
'7\nabcdefgdone\n' + protocol.RESPONSE_VERSION_TWO)
1604
self.assertTrue(self.end_received)
1605
self.assertEqual(protocol.RESPONSE_VERSION_TWO,
1606
server_protocol.excess_buffer)
1607
self.assertEqual("", server_protocol.in_buffer)
1608
server_protocol.accept_bytes('Y')
1609
self.assertEqual(protocol.RESPONSE_VERSION_TWO + "Y",
1610
server_protocol.excess_buffer)
1611
self.assertEqual("", server_protocol.in_buffer)
1613
def test_accept_excess_bytes_after_dispatch(self):
1614
out_stream = StringIO()
1615
smart_protocol = protocol.SmartServerRequestProtocolTwo(
1616
None, out_stream.write)
1617
smart_protocol.accept_bytes('hello\n')
1618
self.assertEqual(protocol.RESPONSE_VERSION_TWO + "ok\x012\n",
1619
out_stream.getvalue())
1620
smart_protocol.accept_bytes(protocol.REQUEST_VERSION_TWO + 'hel')
1621
self.assertEqual(protocol.REQUEST_VERSION_TWO + "hel",
1622
smart_protocol.excess_buffer)
1623
smart_protocol.accept_bytes('lo\n')
1624
self.assertEqual(protocol.REQUEST_VERSION_TWO + "hello\n",
1625
smart_protocol.excess_buffer)
1626
self.assertEqual("", smart_protocol.in_buffer)
1628
def test__send_response_sets_finished_reading(self):
1629
smart_protocol = protocol.SmartServerRequestProtocolTwo(
1630
None, lambda x: None)
1631
self.assertEqual(1, smart_protocol.next_read_size())
1632
smart_protocol._send_response(('x',))
1633
self.assertEqual(0, smart_protocol.next_read_size())
1635
def test_query_version(self):
1636
"""query_version on a SmartClientProtocolTwo should return a number.
1638
The protocol provides the query_version because the domain level clients
1639
may all need to be able to probe for capabilities.
1641
# What we really want to test here is that SmartClientProtocolTwo calls
1642
# accept_bytes(tuple_based_encoding_of_hello) and reads and parses the
1643
# response of tuple-encoded (ok, 1). Also, seperately we should test
1644
# the error if the response is a non-understood version.
1645
input = StringIO(protocol.RESPONSE_VERSION_TWO + 'ok\x012\n')
1647
client_medium = medium.SmartSimplePipesClientMedium(input, output)
1648
request = client_medium.get_request()
1649
smart_protocol = protocol.SmartClientRequestProtocolTwo(request)
1650
self.assertEqual(2, smart_protocol.query_version())
1652
def test_client_call_empty_response(self):
1653
# protocol.call() can get back an empty tuple as a response. This occurs
1654
# when the parsed line is an empty line, and results in a tuple with
1655
# one element - an empty string.
1656
self.assertServerToClientEncoding(
1657
protocol.RESPONSE_VERSION_TWO + '\n', ('', ), [(), ('', )])
1659
def test_client_call_three_element_response(self):
1660
# protocol.call() can get back tuples of other lengths. A three element
1661
# tuple should be unpacked as three strings.
1662
self.assertServerToClientEncoding(
1663
protocol.RESPONSE_VERSION_TWO + 'a\x01b\x0134\n', ('a', 'b', '34'),
1666
def test_client_call_with_body_bytes_uploads(self):
1667
# protocol.call_with_body_bytes should length-prefix the bytes onto the
1669
expected_bytes = protocol.REQUEST_VERSION_TWO + "foo\n7\nabcdefgdone\n"
1670
input = StringIO("\n")
1672
client_medium = medium.SmartSimplePipesClientMedium(input, output)
1673
request = client_medium.get_request()
1674
smart_protocol = protocol.SmartClientRequestProtocolTwo(request)
1675
smart_protocol.call_with_body_bytes(('foo', ), "abcdefg")
1676
self.assertEqual(expected_bytes, output.getvalue())
1678
def test_client_call_with_body_readv_array(self):
1679
# protocol.call_with_upload should encode the readv array and then
1680
# length-prefix the bytes onto the wire.
1681
expected_bytes = protocol.REQUEST_VERSION_TWO+"foo\n7\n1,2\n5,6done\n"
1682
input = StringIO("\n")
1684
client_medium = medium.SmartSimplePipesClientMedium(input, output)
1685
request = client_medium.get_request()
1686
smart_protocol = protocol.SmartClientRequestProtocolTwo(request)
1687
smart_protocol.call_with_body_readv_array(('foo', ), [(1,2),(5,6)])
1688
self.assertEqual(expected_bytes, output.getvalue())
1690
def test_client_read_body_bytes_all(self):
1691
# read_body_bytes should decode the body bytes from the wire into
1693
expected_bytes = "1234567"
1694
server_bytes = protocol.RESPONSE_VERSION_TWO + "ok\n7\n1234567done\n"
1695
input = StringIO(server_bytes)
1697
client_medium = medium.SmartSimplePipesClientMedium(input, output)
1698
request = client_medium.get_request()
1699
smart_protocol = protocol.SmartClientRequestProtocolTwo(request)
1700
smart_protocol.call('foo')
1701
smart_protocol.read_response_tuple(True)
1702
self.assertEqual(expected_bytes, smart_protocol.read_body_bytes())
1704
def test_client_read_body_bytes_incremental(self):
1705
# test reading a few bytes at a time from the body
1706
# XXX: possibly we should test dribbling the bytes into the stringio
1707
# to make the state machine work harder: however, as we use the
1708
# LengthPrefixedBodyDecoder that is already well tested - we can skip
1710
expected_bytes = "1234567"
1711
server_bytes = protocol.RESPONSE_VERSION_TWO + "ok\n7\n1234567done\n"
1712
input = StringIO(server_bytes)
1714
client_medium = medium.SmartSimplePipesClientMedium(input, output)
1715
request = client_medium.get_request()
1716
smart_protocol = protocol.SmartClientRequestProtocolTwo(request)
1717
smart_protocol.call('foo')
1718
smart_protocol.read_response_tuple(True)
1719
self.assertEqual(expected_bytes[0:2], smart_protocol.read_body_bytes(2))
1720
self.assertEqual(expected_bytes[2:4], smart_protocol.read_body_bytes(2))
1721
self.assertEqual(expected_bytes[4:6], smart_protocol.read_body_bytes(2))
1722
self.assertEqual(expected_bytes[6], smart_protocol.read_body_bytes())
1724
def test_client_cancel_read_body_does_not_eat_body_bytes(self):
1725
# cancelling the expected body needs to finish the request, but not
1726
# read any more bytes.
1727
expected_bytes = "1234567"
1728
server_bytes = protocol.RESPONSE_VERSION_TWO + "ok\n7\n1234567done\n"
1729
input = StringIO(server_bytes)
1731
client_medium = medium.SmartSimplePipesClientMedium(input, output)
1732
request = client_medium.get_request()
1733
smart_protocol = protocol.SmartClientRequestProtocolTwo(request)
1734
smart_protocol.call('foo')
1735
smart_protocol.read_response_tuple(True)
1736
smart_protocol.cancel_read_body()
1737
self.assertEqual(len(protocol.RESPONSE_VERSION_TWO + 'ok\n'),
1740
errors.ReadingCompleted, smart_protocol.read_body_bytes)
1445
1743
class TestSmartClientUnicode(tests.TestCase):
1446
1744
"""_SmartClient tests for unicode arguments.