477
449
self.assertEqual(bzrdir.BzrDirMetaFormat1, type(result))
478
450
self.assertEqual(None, result._repository_format)
479
451
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)
452
client.finished_test()
556
455
class TestBzrDirOpenBranch(TestRemote):
847
741
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
744
class OldSmartClient(object):
906
745
"""A fake smart client for test_old_version that just returns a version one
907
746
response to the 'hello' (query version) command.
1090
919
transport = transport.clone('quack')
1091
920
branch = self.make_remote_branch(transport, client)
1092
921
result = branch.tags.get_tag_dict()
1093
self.assertFinished(client)
922
client.finished_test()
1094
923
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
926
class TestBranchLastRevisionInfo(RemoteBranchTestCase):
1147
928
def test_empty_branch(self):
2229
1963
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
1966
class TestRepositoryIsShared(TestRemoteRepository):
2315
1968
def test_is_shared(self):
2430
2083
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)
2086
class TestRepositoryInsertStream(TestRemoteRepository):
2088
def test_unlocked_repo(self):
2089
transport_path = 'quack'
2090
repo, client = self.setup_fake_client_and_repository(transport_path)
2091
client.add_expected_call(
2092
'Repository.insert_stream', ('quack/', ''),
2094
client.add_expected_call(
2095
'Repository.insert_stream', ('quack/', ''),
2097
sink = repo._get_sink()
2098
fmt = repository.RepositoryFormat.get_default_format()
2099
resume_tokens, missing_keys = sink.insert_stream([], fmt, [])
2100
self.assertEqual([], resume_tokens)
2101
self.assertEqual(set(), missing_keys)
2102
client.finished_test()
2104
def test_locked_repo_with_no_lock_token(self):
2105
transport_path = 'quack'
2106
repo, client = self.setup_fake_client_and_repository(transport_path)
2107
client.add_expected_call(
2108
'Repository.lock_write', ('quack/', ''),
2109
'success', ('ok', ''))
2110
client.add_expected_call(
2111
'Repository.insert_stream', ('quack/', ''),
2113
client.add_expected_call(
2114
'Repository.insert_stream', ('quack/', ''),
2117
sink = repo._get_sink()
2118
fmt = repository.RepositoryFormat.get_default_format()
2119
resume_tokens, missing_keys = sink.insert_stream([], fmt, [])
2120
self.assertEqual([], resume_tokens)
2121
self.assertEqual(set(), missing_keys)
2122
client.finished_test()
2124
def test_locked_repo_with_lock_token(self):
2125
transport_path = 'quack'
2126
repo, client = self.setup_fake_client_and_repository(transport_path)
2127
client.add_expected_call(
2128
'Repository.lock_write', ('quack/', ''),
2129
'success', ('ok', 'a token'))
2130
client.add_expected_call(
2131
'Repository.insert_stream_locked', ('quack/', '', 'a token'),
2133
client.add_expected_call(
2134
'Repository.insert_stream_locked', ('quack/', '', 'a token'),
2137
sink = repo._get_sink()
2138
fmt = repository.RepositoryFormat.get_default_format()
2139
resume_tokens, missing_keys = sink.insert_stream([], fmt, [])
2140
self.assertEqual([], resume_tokens)
2141
self.assertEqual(set(), missing_keys)
2142
client.finished_test()
2653
2145
class TestRepositoryTarball(TestRemoteRepository):