477
426
self.assertEqual(bzrdir.BzrDirMetaFormat1, type(result))
478
427
self.assertEqual(None, result._repository_format)
479
428
self.assertEqual(None, result._branch_format)
480
self.assertFinished(client)
483
class TestBzrDirOpen(TestRemote):
485
def make_fake_client_and_transport(self, path='quack'):
486
transport = MemoryTransport()
487
transport.mkdir(path)
488
transport = transport.clone(path)
489
client = FakeClient(transport.base)
490
return client, transport
492
def test_absent(self):
493
client, transport = self.make_fake_client_and_transport()
494
client.add_expected_call(
495
'BzrDir.open_2.1', ('quack/',), 'success', ('no',))
496
self.assertRaises(errors.NotBranchError, RemoteBzrDir, transport,
497
remote.RemoteBzrDirFormat(), _client=client, _force_probe=True)
498
self.assertFinished(client)
500
def test_present_without_workingtree(self):
501
client, transport = self.make_fake_client_and_transport()
502
client.add_expected_call(
503
'BzrDir.open_2.1', ('quack/',), 'success', ('yes', 'no'))
504
bd = RemoteBzrDir(transport, remote.RemoteBzrDirFormat(),
505
_client=client, _force_probe=True)
506
self.assertIsInstance(bd, RemoteBzrDir)
507
self.assertFalse(bd.has_workingtree())
508
self.assertRaises(errors.NoWorkingTree, bd.open_workingtree)
509
self.assertFinished(client)
511
def test_present_with_workingtree(self):
512
client, transport = self.make_fake_client_and_transport()
513
client.add_expected_call(
514
'BzrDir.open_2.1', ('quack/',), 'success', ('yes', 'yes'))
515
bd = RemoteBzrDir(transport, remote.RemoteBzrDirFormat(),
516
_client=client, _force_probe=True)
517
self.assertIsInstance(bd, RemoteBzrDir)
518
self.assertTrue(bd.has_workingtree())
519
self.assertRaises(errors.NotLocalUrl, bd.open_workingtree)
520
self.assertFinished(client)
522
def test_backwards_compat(self):
523
client, transport = self.make_fake_client_and_transport()
524
client.add_expected_call(
525
'BzrDir.open_2.1', ('quack/',), 'unknown', ('BzrDir.open_2.1',))
526
client.add_expected_call(
527
'BzrDir.open', ('quack/',), 'success', ('yes',))
528
bd = RemoteBzrDir(transport, remote.RemoteBzrDirFormat(),
529
_client=client, _force_probe=True)
530
self.assertIsInstance(bd, RemoteBzrDir)
531
self.assertFinished(client)
533
def test_backwards_compat_hpss_v2(self):
534
client, transport = self.make_fake_client_and_transport()
535
# Monkey-patch fake client to simulate real-world behaviour with v2
536
# server: upon first RPC call detect the protocol version, and because
537
# the version is 2 also do _remember_remote_is_before((1, 6)) before
538
# continuing with the RPC.
539
orig_check_call = client._check_call
540
def check_call(method, args):
541
client._medium._protocol_version = 2
542
client._medium._remember_remote_is_before((1, 6))
543
client._check_call = orig_check_call
544
client._check_call(method, args)
545
client._check_call = check_call
546
client.add_expected_call(
547
'BzrDir.open_2.1', ('quack/',), 'unknown', ('BzrDir.open_2.1',))
548
client.add_expected_call(
549
'BzrDir.open', ('quack/',), 'success', ('yes',))
550
bd = RemoteBzrDir(transport, remote.RemoteBzrDirFormat(),
551
_client=client, _force_probe=True)
552
self.assertIsInstance(bd, RemoteBzrDir)
553
self.assertFinished(client)
429
client.finished_test()
556
432
class TestBzrDirOpenBranch(TestRemote):
847
718
self.assertEqual(network_name, repo._format.network_name())
850
class TestBzrDirFormatInitializeEx(TestRemote):
852
def test_success(self):
853
"""Simple test for typical successful call."""
854
fmt = bzrdir.RemoteBzrDirFormat()
855
default_format_name = BzrDirFormat.get_default_format().network_name()
856
transport = self.get_transport()
857
client = FakeClient(transport.base)
858
client.add_expected_call(
859
'BzrDirFormat.initialize_ex_1.16',
860
(default_format_name, 'path', 'False', 'False', 'False', '',
861
'', '', '', 'False'),
863
('.', 'no', 'no', 'yes', 'repo fmt', 'repo bzrdir fmt',
864
'bzrdir fmt', 'False', '', '', 'repo lock token'))
865
# XXX: It would be better to call fmt.initialize_on_transport_ex, but
866
# it's currently hard to test that without supplying a real remote
867
# transport connected to a real server.
868
result = fmt._initialize_on_transport_ex_rpc(client, 'path',
869
transport, False, False, False, None, None, None, None, False)
870
self.assertFinished(client)
872
def test_error(self):
873
"""Error responses are translated, e.g. 'PermissionDenied' raises the
874
corresponding error from the client.
876
fmt = bzrdir.RemoteBzrDirFormat()
877
default_format_name = BzrDirFormat.get_default_format().network_name()
878
transport = self.get_transport()
879
client = FakeClient(transport.base)
880
client.add_expected_call(
881
'BzrDirFormat.initialize_ex_1.16',
882
(default_format_name, 'path', 'False', 'False', 'False', '',
883
'', '', '', 'False'),
885
('PermissionDenied', 'path', 'extra info'))
886
# XXX: It would be better to call fmt.initialize_on_transport_ex, but
887
# it's currently hard to test that without supplying a real remote
888
# transport connected to a real server.
889
err = self.assertRaises(errors.PermissionDenied,
890
fmt._initialize_on_transport_ex_rpc, client, 'path', transport,
891
False, False, False, None, None, None, None, False)
892
self.assertEqual('path', err.path)
893
self.assertEqual(': extra info', err.extra)
894
self.assertFinished(client)
896
def test_error_from_real_server(self):
897
"""Integration test for error translation."""
898
transport = self.make_smart_server('foo')
899
transport = transport.clone('no-such-path')
900
fmt = bzrdir.RemoteBzrDirFormat()
901
err = self.assertRaises(errors.NoSuchFile,
902
fmt.initialize_on_transport_ex, transport, create_prefix=False)
905
721
class OldSmartClient(object):
906
722
"""A fake smart client for test_old_version that just returns a version one
907
723
response to the 'hello' (query version) command.
1090
841
transport = transport.clone('quack')
1091
842
branch = self.make_remote_branch(transport, client)
1092
843
result = branch.tags.get_tag_dict()
1093
self.assertFinished(client)
844
client.finished_test()
1094
845
self.assertEqual({}, result)
1097
class TestBranchSetTagsBytes(RemoteBranchTestCase):
1099
def test_trivial(self):
1100
transport = MemoryTransport()
1101
client = FakeClient(transport.base)
1102
client.add_expected_call(
1103
'Branch.get_stacked_on_url', ('quack/',),
1104
'error', ('NotStacked',))
1105
client.add_expected_call(
1106
'Branch.set_tags_bytes', ('quack/', 'branch token', 'repo token'),
1108
transport.mkdir('quack')
1109
transport = transport.clone('quack')
1110
branch = self.make_remote_branch(transport, client)
1111
self.lock_remote_branch(branch)
1112
branch._set_tags_bytes('tags bytes')
1113
self.assertFinished(client)
1114
self.assertEqual('tags bytes', client._calls[-1][-1])
1116
def test_backwards_compatible(self):
1117
transport = MemoryTransport()
1118
client = FakeClient(transport.base)
1119
client.add_expected_call(
1120
'Branch.get_stacked_on_url', ('quack/',),
1121
'error', ('NotStacked',))
1122
client.add_expected_call(
1123
'Branch.set_tags_bytes', ('quack/', 'branch token', 'repo token'),
1124
'unknown', ('Branch.set_tags_bytes',))
1125
transport.mkdir('quack')
1126
transport = transport.clone('quack')
1127
branch = self.make_remote_branch(transport, client)
1128
self.lock_remote_branch(branch)
1129
class StubRealBranch(object):
1132
def _set_tags_bytes(self, bytes):
1133
self.calls.append(('set_tags_bytes', bytes))
1134
real_branch = StubRealBranch()
1135
branch._real_branch = real_branch
1136
branch._set_tags_bytes('tags bytes')
1137
# Call a second time, to exercise the 'remote version already inferred'
1139
branch._set_tags_bytes('tags bytes')
1140
self.assertFinished(client)
1142
[('set_tags_bytes', 'tags bytes')] * 2, real_branch.calls)
1145
848
class TestBranchLastRevisionInfo(RemoteBranchTestCase):
1147
850
def test_empty_branch(self):
1596
1310
self.assertEqual('rejection message', err.msg)
1599
class TestBranchGetSetConfig(RemoteBranchTestCase):
1313
class TestBranchControlGetBranchConf(tests.TestCaseWithMemoryTransport):
1314
"""Getting the branch configuration should use an abstract method not vfs.
1601
1317
def test_get_branch_conf(self):
1602
# in an empty branch we decode the response properly
1603
client = FakeClient()
1604
client.add_expected_call(
1605
'Branch.get_stacked_on_url', ('memory:///',),
1606
'error', ('NotStacked',),)
1607
client.add_success_response_with_body('# config file body', 'ok')
1608
transport = MemoryTransport()
1609
branch = self.make_remote_branch(transport, client)
1610
config = branch.get_config()
1611
config.has_explicit_nickname()
1613
[('call', 'Branch.get_stacked_on_url', ('memory:///',)),
1614
('call_expecting_body', 'Branch.get_config_file', ('memory:///',))],
1617
def test_get_multi_line_branch_conf(self):
1618
# Make sure that multiple-line branch.conf files are supported
1620
# https://bugs.launchpad.net/bzr/+bug/354075
1621
client = FakeClient()
1622
client.add_expected_call(
1623
'Branch.get_stacked_on_url', ('memory:///',),
1624
'error', ('NotStacked',),)
1625
client.add_success_response_with_body('a = 1\nb = 2\nc = 3\n', 'ok')
1626
transport = MemoryTransport()
1627
branch = self.make_remote_branch(transport, client)
1628
config = branch.get_config()
1629
self.assertEqual(u'2', config.get_user_option('b'))
1631
def test_set_option(self):
1632
client = FakeClient()
1633
client.add_expected_call(
1634
'Branch.get_stacked_on_url', ('memory:///',),
1635
'error', ('NotStacked',),)
1636
client.add_expected_call(
1637
'Branch.lock_write', ('memory:///', '', ''),
1638
'success', ('ok', 'branch token', 'repo token'))
1639
client.add_expected_call(
1640
'Branch.set_config_option', ('memory:///', 'branch token',
1641
'repo token', 'foo', 'bar', ''),
1643
client.add_expected_call(
1644
'Branch.unlock', ('memory:///', 'branch token', 'repo token'),
1646
transport = MemoryTransport()
1647
branch = self.make_remote_branch(transport, client)
1649
config = branch._get_config()
1650
config.set_option('foo', 'bar')
1652
self.assertFinished(client)
1654
def test_set_option_with_dict(self):
1655
client = FakeClient()
1656
client.add_expected_call(
1657
'Branch.get_stacked_on_url', ('memory:///',),
1658
'error', ('NotStacked',),)
1659
client.add_expected_call(
1660
'Branch.lock_write', ('memory:///', '', ''),
1661
'success', ('ok', 'branch token', 'repo token'))
1662
encoded_dict_value = 'd5:ascii1:a11:unicode \xe2\x8c\x9a3:\xe2\x80\xbde'
1663
client.add_expected_call(
1664
'Branch.set_config_option_dict', ('memory:///', 'branch token',
1665
'repo token', encoded_dict_value, 'foo', ''),
1667
client.add_expected_call(
1668
'Branch.unlock', ('memory:///', 'branch token', 'repo token'),
1670
transport = MemoryTransport()
1671
branch = self.make_remote_branch(transport, client)
1673
config = branch._get_config()
1675
{'ascii': 'a', u'unicode \N{WATCH}': u'\N{INTERROBANG}'},
1678
self.assertFinished(client)
1680
def test_backwards_compat_set_option(self):
1681
self.setup_smart_server_with_call_log()
1682
branch = self.make_branch('.')
1683
verb = 'Branch.set_config_option'
1684
self.disable_verb(verb)
1686
self.addCleanup(branch.unlock)
1687
self.reset_smart_call_log()
1688
branch._get_config().set_option('value', 'name')
1689
self.assertLength(10, self.hpss_calls)
1690
self.assertEqual('value', branch._get_config().get_option('name'))
1692
def test_backwards_compat_set_option_with_dict(self):
1693
self.setup_smart_server_with_call_log()
1694
branch = self.make_branch('.')
1695
verb = 'Branch.set_config_option_dict'
1696
self.disable_verb(verb)
1698
self.addCleanup(branch.unlock)
1699
self.reset_smart_call_log()
1700
config = branch._get_config()
1701
value_dict = {'ascii': 'a', u'unicode \N{WATCH}': u'\N{INTERROBANG}'}
1702
config.set_option(value_dict, 'name')
1703
self.assertLength(10, self.hpss_calls)
1704
self.assertEqual(value_dict, branch._get_config().get_option('name'))
1318
raise tests.KnownFailure('branch.conf is not retrieved by get_config_file')
1319
## # We should see that branch.get_config() does a single rpc to get the
1320
## # remote configuration file, abstracting away where that is stored on
1321
## # the server. However at the moment it always falls back to using the
1322
## # vfs, and this would need some changes in config.py.
1324
## # in an empty branch we decode the response properly
1325
## client = FakeClient([(('ok', ), '# config file body')], self.get_url())
1326
## # we need to make a real branch because the remote_branch.control_files
1327
## # will trigger _ensure_real.
1328
## branch = self.make_branch('quack')
1329
## transport = branch.bzrdir.root_transport
1330
## # we do not want bzrdir to make any remote calls
1331
## bzrdir = RemoteBzrDir(transport, _client=False)
1332
## branch = RemoteBranch(bzrdir, None, _client=client)
1333
## config = branch.get_config()
1334
## self.assertEqual(
1335
## [('call_expecting_body', 'Branch.get_config_file', ('quack/',))],
1707
1339
class TestBranchLockWrite(RemoteBranchTestCase):
2047
1616
errors.UnexpectedSmartServerResponse,
2048
1617
repo.get_parent_map, ['a-revision-id'])
2050
def test_get_parent_map_negative_caches_missing_keys(self):
2051
self.setup_smart_server_with_call_log()
2052
repo = self.make_repository('foo')
2053
self.assertIsInstance(repo, RemoteRepository)
2055
self.addCleanup(repo.unlock)
2056
self.reset_smart_call_log()
2057
graph = repo.get_graph()
2058
self.assertEqual({},
2059
graph.get_parent_map(['some-missing', 'other-missing']))
2060
self.assertLength(1, self.hpss_calls)
2061
# No call if we repeat this
2062
self.reset_smart_call_log()
2063
graph = repo.get_graph()
2064
self.assertEqual({},
2065
graph.get_parent_map(['some-missing', 'other-missing']))
2066
self.assertLength(0, self.hpss_calls)
2067
# Asking for more unknown keys makes a request.
2068
self.reset_smart_call_log()
2069
graph = repo.get_graph()
2070
self.assertEqual({},
2071
graph.get_parent_map(['some-missing', 'other-missing',
2073
self.assertLength(1, self.hpss_calls)
2075
def disableExtraResults(self):
2076
self.overrideAttr(SmartServerRepositoryGetParentMap,
2077
'no_extra_results', True)
2079
def test_null_cached_missing_and_stop_key(self):
2080
self.setup_smart_server_with_call_log()
2081
# Make a branch with a single revision.
2082
builder = self.make_branch_builder('foo')
2083
builder.start_series()
2084
builder.build_snapshot('first', None, [
2085
('add', ('', 'root-id', 'directory', ''))])
2086
builder.finish_series()
2087
branch = builder.get_branch()
2088
repo = branch.repository
2089
self.assertIsInstance(repo, RemoteRepository)
2090
# Stop the server from sending extra results.
2091
self.disableExtraResults()
2093
self.addCleanup(repo.unlock)
2094
self.reset_smart_call_log()
2095
graph = repo.get_graph()
2096
# Query for 'first' and 'null:'. Because 'null:' is a parent of
2097
# 'first' it will be a candidate for the stop_keys of subsequent
2098
# requests, and because 'null:' was queried but not returned it will be
2099
# cached as missing.
2100
self.assertEqual({'first': ('null:',)},
2101
graph.get_parent_map(['first', 'null:']))
2102
# Now query for another key. This request will pass along a recipe of
2103
# start and stop keys describing the already cached results, and this
2104
# recipe's revision count must be correct (or else it will trigger an
2105
# error from the server).
2106
self.assertEqual({}, graph.get_parent_map(['another-key']))
2107
# This assertion guards against disableExtraResults silently failing to
2108
# work, thus invalidating the test.
2109
self.assertLength(2, self.hpss_calls)
2111
def test_get_parent_map_gets_ghosts_from_result(self):
2112
# asking for a revision should negatively cache close ghosts in its
2114
self.setup_smart_server_with_call_log()
2115
tree = self.make_branch_and_memory_tree('foo')
2118
builder = treebuilder.TreeBuilder()
2119
builder.start_tree(tree)
2121
builder.finish_tree()
2122
tree.set_parent_ids(['non-existant'], allow_leftmost_as_ghost=True)
2123
rev_id = tree.commit('')
2127
self.addCleanup(tree.unlock)
2128
repo = tree.branch.repository
2129
self.assertIsInstance(repo, RemoteRepository)
2131
repo.get_parent_map([rev_id])
2132
self.reset_smart_call_log()
2133
# Now asking for rev_id's ghost parent should not make calls
2134
self.assertEqual({}, repo.get_parent_map(['non-existant']))
2135
self.assertLength(0, self.hpss_calls)
2138
1620
class TestGetParentMapAllowsNew(tests.TestCaseWithTransport):
2140
1622
def test_allows_new_revisions(self):
2141
1623
"""get_parent_map's results can be updated by commit."""
2142
smart_server = test_server.SmartTCPServer_for_testing()
2143
self.start_server(smart_server)
1624
smart_server = server.SmartTCPServer_for_testing()
1625
smart_server.setUp()
1626
self.addCleanup(smart_server.tearDown)
2144
1627
self.make_branch('branch')
2145
1628
branch = Branch.open(smart_server.get_url() + '/branch')
2146
1629
tree = branch.create_checkout('tree', lightweight=True)
2225
1705
repo, client = self.setup_fake_client_and_repository(transport_path)
2226
1706
client.add_error_response('AnUnexpectedError')
2227
1707
e = self.assertRaises(errors.UnknownErrorFromSmartServer,
2228
repo._get_revision_graph, revid)
1708
self.applyDeprecated, one_four, repo.get_revision_graph, revid)
2229
1709
self.assertEqual(('AnUnexpectedError',), e.error_tuple)
2232
class TestRepositoryGetRevIdForRevno(TestRemoteRepository):
2235
repo, client = self.setup_fake_client_and_repository('quack')
2236
client.add_expected_call(
2237
'Repository.get_rev_id_for_revno', ('quack/', 5, (42, 'rev-foo')),
2238
'success', ('ok', 'rev-five'))
2239
result = repo.get_rev_id_for_revno(5, (42, 'rev-foo'))
2240
self.assertEqual((True, 'rev-five'), result)
2241
self.assertFinished(client)
2243
def test_history_incomplete(self):
2244
repo, client = self.setup_fake_client_and_repository('quack')
2245
client.add_expected_call(
2246
'Repository.get_rev_id_for_revno', ('quack/', 5, (42, 'rev-foo')),
2247
'success', ('history-incomplete', 10, 'rev-ten'))
2248
result = repo.get_rev_id_for_revno(5, (42, 'rev-foo'))
2249
self.assertEqual((False, (10, 'rev-ten')), result)
2250
self.assertFinished(client)
2252
def test_history_incomplete_with_fallback(self):
2253
"""A 'history-incomplete' response causes the fallback repository to be
2254
queried too, if one is set.
2256
# Make a repo with a fallback repo, both using a FakeClient.
2257
format = remote.response_tuple_to_repo_format(
2258
('yes', 'no', 'yes', self.get_repo_format().network_name()))
2259
repo, client = self.setup_fake_client_and_repository('quack')
2260
repo._format = format
2261
fallback_repo, ignored = self.setup_fake_client_and_repository(
2263
fallback_repo._client = client
2264
fallback_repo._format = format
2265
repo.add_fallback_repository(fallback_repo)
2266
# First the client should ask the primary repo
2267
client.add_expected_call(
2268
'Repository.get_rev_id_for_revno', ('quack/', 1, (42, 'rev-foo')),
2269
'success', ('history-incomplete', 2, 'rev-two'))
2270
# Then it should ask the fallback, using revno/revid from the
2271
# history-incomplete response as the known revno/revid.
2272
client.add_expected_call(
2273
'Repository.get_rev_id_for_revno',('fallback/', 1, (2, 'rev-two')),
2274
'success', ('ok', 'rev-one'))
2275
result = repo.get_rev_id_for_revno(1, (42, 'rev-foo'))
2276
self.assertEqual((True, 'rev-one'), result)
2277
self.assertFinished(client)
2279
def test_nosuchrevision(self):
2280
# 'nosuchrevision' is returned when the known-revid is not found in the
2281
# remote repo. The client translates that response to NoSuchRevision.
2282
repo, client = self.setup_fake_client_and_repository('quack')
2283
client.add_expected_call(
2284
'Repository.get_rev_id_for_revno', ('quack/', 5, (42, 'rev-foo')),
2285
'error', ('nosuchrevision', 'rev-foo'))
2287
errors.NoSuchRevision,
2288
repo.get_rev_id_for_revno, 5, (42, 'rev-foo'))
2289
self.assertFinished(client)
2291
def test_branch_fallback_locking(self):
2292
"""RemoteBranch.get_rev_id takes a read lock, and tries to call the
2293
get_rev_id_for_revno verb. If the verb is unknown the VFS fallback
2294
will be invoked, which will fail if the repo is unlocked.
2296
self.setup_smart_server_with_call_log()
2297
tree = self.make_branch_and_memory_tree('.')
2300
rev1 = tree.commit('First')
2301
rev2 = tree.commit('Second')
2303
branch = tree.branch
2304
self.assertFalse(branch.is_locked())
2305
self.reset_smart_call_log()
2306
verb = 'Repository.get_rev_id_for_revno'
2307
self.disable_verb(verb)
2308
self.assertEqual(rev1, branch.get_rev_id(1))
2309
self.assertLength(1, [call for call in self.hpss_calls if
2310
call.call.method == verb])
2313
1712
class TestRepositoryIsShared(TestRemoteRepository):
2315
1714
def test_is_shared(self):
2430
1829
self.assertEqual([], client._calls)
2433
class TestRepositoryInsertStreamBase(TestRemoteRepository):
2434
"""Base class for Repository.insert_stream and .insert_stream_1.19
2438
def checkInsertEmptyStream(self, repo, client):
2439
"""Insert an empty stream, checking the result.
2441
This checks that there are no resume_tokens or missing_keys, and that
2442
the client is finished.
2444
sink = repo._get_sink()
2445
fmt = repository.RepositoryFormat.get_default_format()
2446
resume_tokens, missing_keys = sink.insert_stream([], fmt, [])
2447
self.assertEqual([], resume_tokens)
2448
self.assertEqual(set(), missing_keys)
2449
self.assertFinished(client)
2452
class TestRepositoryInsertStream(TestRepositoryInsertStreamBase):
2453
"""Tests for using Repository.insert_stream verb when the _1.19 variant is
2456
This test case is very similar to TestRepositoryInsertStream_1_19.
2460
TestRemoteRepository.setUp(self)
2461
self.disable_verb('Repository.insert_stream_1.19')
2463
def test_unlocked_repo(self):
2464
transport_path = 'quack'
2465
repo, client = self.setup_fake_client_and_repository(transport_path)
2466
client.add_expected_call(
2467
'Repository.insert_stream_1.19', ('quack/', ''),
2468
'unknown', ('Repository.insert_stream_1.19',))
2469
client.add_expected_call(
2470
'Repository.insert_stream', ('quack/', ''),
2472
client.add_expected_call(
2473
'Repository.insert_stream', ('quack/', ''),
2475
self.checkInsertEmptyStream(repo, client)
2477
def test_locked_repo_with_no_lock_token(self):
2478
transport_path = 'quack'
2479
repo, client = self.setup_fake_client_and_repository(transport_path)
2480
client.add_expected_call(
2481
'Repository.lock_write', ('quack/', ''),
2482
'success', ('ok', ''))
2483
client.add_expected_call(
2484
'Repository.insert_stream_1.19', ('quack/', ''),
2485
'unknown', ('Repository.insert_stream_1.19',))
2486
client.add_expected_call(
2487
'Repository.insert_stream', ('quack/', ''),
2489
client.add_expected_call(
2490
'Repository.insert_stream', ('quack/', ''),
2493
self.checkInsertEmptyStream(repo, client)
2495
def test_locked_repo_with_lock_token(self):
2496
transport_path = 'quack'
2497
repo, client = self.setup_fake_client_and_repository(transport_path)
2498
client.add_expected_call(
2499
'Repository.lock_write', ('quack/', ''),
2500
'success', ('ok', 'a token'))
2501
client.add_expected_call(
2502
'Repository.insert_stream_1.19', ('quack/', '', 'a token'),
2503
'unknown', ('Repository.insert_stream_1.19',))
2504
client.add_expected_call(
2505
'Repository.insert_stream_locked', ('quack/', '', 'a token'),
2507
client.add_expected_call(
2508
'Repository.insert_stream_locked', ('quack/', '', 'a token'),
2511
self.checkInsertEmptyStream(repo, client)
2513
def test_stream_with_inventory_deltas(self):
2514
"""'inventory-deltas' substreams cannot be sent to the
2515
Repository.insert_stream verb, because not all servers that implement
2516
that verb will accept them. So when one is encountered the RemoteSink
2517
immediately stops using that verb and falls back to VFS insert_stream.
2519
transport_path = 'quack'
2520
repo, client = self.setup_fake_client_and_repository(transport_path)
2521
client.add_expected_call(
2522
'Repository.insert_stream_1.19', ('quack/', ''),
2523
'unknown', ('Repository.insert_stream_1.19',))
2524
client.add_expected_call(
2525
'Repository.insert_stream', ('quack/', ''),
2527
client.add_expected_call(
2528
'Repository.insert_stream', ('quack/', ''),
2530
# Create a fake real repository for insert_stream to fall back on, so
2531
# that we can directly see the records the RemoteSink passes to the
2536
def insert_stream(self, stream, src_format, resume_tokens):
2537
for substream_kind, substream in stream:
2538
self.records.append(
2539
(substream_kind, [record.key for record in substream]))
2540
return ['fake tokens'], ['fake missing keys']
2541
fake_real_sink = FakeRealSink()
2542
class FakeRealRepository:
2543
def _get_sink(self):
2544
return fake_real_sink
2545
def is_in_write_group(self):
2547
def refresh_data(self):
2549
repo._real_repository = FakeRealRepository()
2550
sink = repo._get_sink()
2551
fmt = repository.RepositoryFormat.get_default_format()
2552
stream = self.make_stream_with_inv_deltas(fmt)
2553
resume_tokens, missing_keys = sink.insert_stream(stream, fmt, [])
2554
# Every record from the first inventory delta should have been sent to
2556
expected_records = [
2557
('inventory-deltas', [('rev2',), ('rev3',)]),
2558
('texts', [('some-rev', 'some-file')])]
2559
self.assertEqual(expected_records, fake_real_sink.records)
2560
# The return values from the real sink's insert_stream are propagated
2561
# back to the original caller.
2562
self.assertEqual(['fake tokens'], resume_tokens)
2563
self.assertEqual(['fake missing keys'], missing_keys)
2564
self.assertFinished(client)
2566
def make_stream_with_inv_deltas(self, fmt):
2567
"""Make a simple stream with an inventory delta followed by more
2568
records and more substreams to test that all records and substreams
2569
from that point on are used.
2571
This sends, in order:
2572
* inventories substream: rev1, rev2, rev3. rev2 and rev3 are
2574
* texts substream: (some-rev, some-file)
2576
# Define a stream using generators so that it isn't rewindable.
2577
inv = inventory.Inventory(revision_id='rev1')
2578
inv.root.revision = 'rev1'
2579
def stream_with_inv_delta():
2580
yield ('inventories', inventories_substream())
2581
yield ('inventory-deltas', inventory_delta_substream())
2583
versionedfile.FulltextContentFactory(
2584
('some-rev', 'some-file'), (), None, 'content')])
2585
def inventories_substream():
2586
# An empty inventory fulltext. This will be streamed normally.
2587
text = fmt._serializer.write_inventory_to_string(inv)
2588
yield versionedfile.FulltextContentFactory(
2589
('rev1',), (), None, text)
2590
def inventory_delta_substream():
2591
# An inventory delta. This can't be streamed via this verb, so it
2592
# will trigger a fallback to VFS insert_stream.
2593
entry = inv.make_entry(
2594
'directory', 'newdir', inv.root.file_id, 'newdir-id')
2595
entry.revision = 'ghost'
2596
delta = [(None, 'newdir', 'newdir-id', entry)]
2597
serializer = inventory_delta.InventoryDeltaSerializer(
2598
versioned_root=True, tree_references=False)
2599
lines = serializer.delta_to_lines('rev1', 'rev2', delta)
2600
yield versionedfile.ChunkedContentFactory(
2601
('rev2',), (('rev1',)), None, lines)
2603
lines = serializer.delta_to_lines('rev1', 'rev3', delta)
2604
yield versionedfile.ChunkedContentFactory(
2605
('rev3',), (('rev1',)), None, lines)
2606
return stream_with_inv_delta()
2609
class TestRepositoryInsertStream_1_19(TestRepositoryInsertStreamBase):
2611
def test_unlocked_repo(self):
2612
transport_path = 'quack'
2613
repo, client = self.setup_fake_client_and_repository(transport_path)
2614
client.add_expected_call(
2615
'Repository.insert_stream_1.19', ('quack/', ''),
2617
client.add_expected_call(
2618
'Repository.insert_stream_1.19', ('quack/', ''),
2620
self.checkInsertEmptyStream(repo, client)
2622
def test_locked_repo_with_no_lock_token(self):
2623
transport_path = 'quack'
2624
repo, client = self.setup_fake_client_and_repository(transport_path)
2625
client.add_expected_call(
2626
'Repository.lock_write', ('quack/', ''),
2627
'success', ('ok', ''))
2628
client.add_expected_call(
2629
'Repository.insert_stream_1.19', ('quack/', ''),
2631
client.add_expected_call(
2632
'Repository.insert_stream_1.19', ('quack/', ''),
2635
self.checkInsertEmptyStream(repo, client)
2637
def test_locked_repo_with_lock_token(self):
2638
transport_path = 'quack'
2639
repo, client = self.setup_fake_client_and_repository(transport_path)
2640
client.add_expected_call(
2641
'Repository.lock_write', ('quack/', ''),
2642
'success', ('ok', 'a token'))
2643
client.add_expected_call(
2644
'Repository.insert_stream_1.19', ('quack/', '', 'a token'),
2646
client.add_expected_call(
2647
'Repository.insert_stream_1.19', ('quack/', '', 'a token'),
2650
self.checkInsertEmptyStream(repo, client)
1832
class TestRepositoryInsertStream(TestRemoteRepository):
1834
def test_unlocked_repo(self):
1835
transport_path = 'quack'
1836
repo, client = self.setup_fake_client_and_repository(transport_path)
1837
client.add_expected_call(
1838
'Repository.insert_stream', ('quack/', ''),
1840
client.add_expected_call(
1841
'Repository.insert_stream', ('quack/', ''),
1843
sink = repo._get_sink()
1844
fmt = repository.RepositoryFormat.get_default_format()
1845
resume_tokens, missing_keys = sink.insert_stream([], fmt, [])
1846
self.assertEqual([], resume_tokens)
1847
self.assertEqual(set(), missing_keys)
1848
client.finished_test()
1850
def test_locked_repo_with_no_lock_token(self):
1851
transport_path = 'quack'
1852
repo, client = self.setup_fake_client_and_repository(transport_path)
1853
client.add_expected_call(
1854
'Repository.lock_write', ('quack/', ''),
1855
'success', ('ok', ''))
1856
client.add_expected_call(
1857
'Repository.insert_stream', ('quack/', ''),
1859
client.add_expected_call(
1860
'Repository.insert_stream', ('quack/', ''),
1863
sink = repo._get_sink()
1864
fmt = repository.RepositoryFormat.get_default_format()
1865
resume_tokens, missing_keys = sink.insert_stream([], fmt, [])
1866
self.assertEqual([], resume_tokens)
1867
self.assertEqual(set(), missing_keys)
1868
client.finished_test()
1870
def test_locked_repo_with_lock_token(self):
1871
transport_path = 'quack'
1872
repo, client = self.setup_fake_client_and_repository(transport_path)
1873
client.add_expected_call(
1874
'Repository.lock_write', ('quack/', ''),
1875
'success', ('ok', 'a token'))
1876
client.add_expected_call(
1877
'Repository.insert_stream_locked', ('quack/', '', 'a token'),
1879
client.add_expected_call(
1880
'Repository.insert_stream_locked', ('quack/', '', 'a token'),
1883
sink = repo._get_sink()
1884
fmt = repository.RepositoryFormat.get_default_format()
1885
resume_tokens, missing_keys = sink.insert_stream([], fmt, [])
1886
self.assertEqual([], resume_tokens)
1887
self.assertEqual(set(), missing_keys)
1888
client.finished_test()
2653
1891
class TestRepositoryTarball(TestRemoteRepository):