31
31
from bzrlib import (
33
branch as _mod_branch,
42
from bzrlib.smart import (
43
branch as smart_branch,
45
repository as smart_repo,
46
packrepository as smart_packrepo,
51
from bzrlib.tests import test_server
52
from bzrlib.transport import (
40
from bzrlib.branch import Branch, BranchReferenceFormat
41
import bzrlib.smart.branch
42
import bzrlib.smart.bzrdir, bzrlib.smart.bzrdir as smart_dir
43
import bzrlib.smart.packrepository
44
import bzrlib.smart.repository
45
from bzrlib.smart.request import (
46
FailedSmartServerResponse,
49
SuccessfulSmartServerResponse,
51
from bzrlib.tests import (
54
from bzrlib.transport import chroot, get_transport
58
57
def load_tests(standard_tests, module, loader):
59
58
"""Multiply tests version and protocol consistency."""
60
59
# FindRepository tests.
60
bzrdir_mod = bzrlib.smart.bzrdir
62
62
("find_repository", {
63
"_request_class": smart_dir.SmartServerRequestFindRepositoryV1}),
63
"_request_class":bzrdir_mod.SmartServerRequestFindRepositoryV1}),
64
64
("find_repositoryV2", {
65
"_request_class": smart_dir.SmartServerRequestFindRepositoryV2}),
65
"_request_class":bzrdir_mod.SmartServerRequestFindRepositoryV2}),
66
66
("find_repositoryV3", {
67
"_request_class": smart_dir.SmartServerRequestFindRepositoryV3}),
67
"_request_class":bzrdir_mod.SmartServerRequestFindRepositoryV3}),
69
to_adapt, result = tests.split_suite_by_re(standard_tests,
69
to_adapt, result = split_suite_by_re(standard_tests,
70
70
"TestSmartServerRequestFindRepository")
71
v2_only, v1_and_2 = tests.split_suite_by_re(to_adapt,
71
v2_only, v1_and_2 = split_suite_by_re(to_adapt,
73
73
tests.multiply_tests(v1_and_2, scenarios, result)
74
74
# The first scenario is only applicable to v1 protocols, it is deleted
106
106
self.transport_server = self.make_transport_server
108
108
def make_transport_server(self):
109
return test_server.SmartTCPServer_for_testing('-' + self.id())
109
return smart.server.SmartTCPServer_for_testing('-' + self.id())
111
111
def get_smart_medium(self):
112
112
"""Get a smart medium to use in tests."""
113
113
return self.get_transport().get_smart_medium()
116
class TestByteStreamToStream(tests.TestCase):
118
def test_repeated_substreams_same_kind_are_one_stream(self):
119
# Make a stream - an iterable of bytestrings.
120
stream = [('text', [versionedfile.FulltextContentFactory(('k1',), None,
121
None, 'foo')]),('text', [
122
versionedfile.FulltextContentFactory(('k2',), None, None, 'bar')])]
123
fmt = bzrdir.format_registry.get('pack-0.92')().repository_format
124
bytes = smart_repo._stream_to_byte_stream(stream, fmt)
126
# Iterate the resulting iterable; checking that we get only one stream
128
fmt, stream = smart_repo._byte_stream_to_stream(bytes)
129
for kind, substream in stream:
130
streams.append((kind, list(substream)))
131
self.assertLength(1, streams)
132
self.assertLength(2, streams[0][1])
135
116
class TestSmartServerResponse(tests.TestCase):
137
118
def test__eq__(self):
138
self.assertEqual(smart_req.SmartServerResponse(('ok', )),
139
smart_req.SmartServerResponse(('ok', )))
140
self.assertEqual(smart_req.SmartServerResponse(('ok', ), 'body'),
141
smart_req.SmartServerResponse(('ok', ), 'body'))
142
self.assertNotEqual(smart_req.SmartServerResponse(('ok', )),
143
smart_req.SmartServerResponse(('notok', )))
144
self.assertNotEqual(smart_req.SmartServerResponse(('ok', ), 'body'),
145
smart_req.SmartServerResponse(('ok', )))
119
self.assertEqual(SmartServerResponse(('ok', )),
120
SmartServerResponse(('ok', )))
121
self.assertEqual(SmartServerResponse(('ok', ), 'body'),
122
SmartServerResponse(('ok', ), 'body'))
123
self.assertNotEqual(SmartServerResponse(('ok', )),
124
SmartServerResponse(('notok', )))
125
self.assertNotEqual(SmartServerResponse(('ok', ), 'body'),
126
SmartServerResponse(('ok', )))
146
127
self.assertNotEqual(None,
147
smart_req.SmartServerResponse(('ok', )))
128
SmartServerResponse(('ok', )))
149
130
def test__str__(self):
150
131
"""SmartServerResponses can be stringified."""
151
132
self.assertEqual(
152
133
"<SuccessfulSmartServerResponse args=('args',) body='body'>",
153
str(smart_req.SuccessfulSmartServerResponse(('args',), 'body')))
134
str(SuccessfulSmartServerResponse(('args',), 'body')))
154
135
self.assertEqual(
155
136
"<FailedSmartServerResponse args=('args',) body='body'>",
156
str(smart_req.FailedSmartServerResponse(('args',), 'body')))
137
str(FailedSmartServerResponse(('args',), 'body')))
159
140
class TestSmartServerRequest(tests.TestCaseWithMemoryTransport):
161
142
def test_translate_client_path(self):
162
143
transport = self.get_transport()
163
request = smart_req.SmartServerRequest(transport, 'foo/')
144
request = SmartServerRequest(transport, 'foo/')
164
145
self.assertEqual('./', request.translate_client_path('foo/'))
165
146
self.assertRaises(
166
147
errors.InvalidURLJoin, request.translate_client_path, 'foo/..')
282
if repo._format.supports_external_lookups:
286
if (smart_dir.SmartServerRequestFindRepositoryV3 ==
250
if (smart.bzrdir.SmartServerRequestFindRepositoryV3 ==
287
251
self._request_class):
288
return smart_req.SuccessfulSmartServerResponse(
289
('ok', '', rich_root, subtrees, external,
252
return SuccessfulSmartServerResponse(
253
('ok', '', rich_root, subtrees, 'no',
290
254
repo._format.network_name()))
291
elif (smart_dir.SmartServerRequestFindRepositoryV2 ==
255
elif (smart.bzrdir.SmartServerRequestFindRepositoryV2 ==
292
256
self._request_class):
293
257
# All tests so far are on formats, and for non-external
295
return smart_req.SuccessfulSmartServerResponse(
296
('ok', '', rich_root, subtrees, external))
259
return SuccessfulSmartServerResponse(
260
('ok', '', rich_root, subtrees, 'no'))
298
return smart_req.SuccessfulSmartServerResponse(
299
('ok', '', rich_root, subtrees))
262
return SuccessfulSmartServerResponse(('ok', '', rich_root, subtrees))
301
264
def test_shared_repository(self):
302
265
"""When there is a shared repository, we get 'ok', 'relpath-to-repo'."""
427
384
"""Initializing an extant directory should fail like the bzrdir api."""
428
385
backing = self.get_transport()
429
386
name = self.make_bzrdir('reference')._format.network_name()
430
request = smart_dir.SmartServerRequestBzrDirInitializeEx(backing)
387
request = smart.bzrdir.SmartServerRequestBzrDirInitializeEx(backing)
431
388
self.make_bzrdir('subdir')
432
389
self.assertRaises(errors.FileExists, request.execute, name, 'subdir',
433
390
'False', 'False', 'False', '', '', '', '', 'False')
436
class TestSmartServerRequestOpenBzrDir(tests.TestCaseWithMemoryTransport):
438
def test_no_directory(self):
439
backing = self.get_transport()
440
request = smart_dir.SmartServerRequestOpenBzrDir(backing)
441
self.assertEqual(smart_req.SmartServerResponse(('no', )),
442
request.execute('does-not-exist'))
444
def test_empty_directory(self):
445
backing = self.get_transport()
446
backing.mkdir('empty')
447
request = smart_dir.SmartServerRequestOpenBzrDir(backing)
448
self.assertEqual(smart_req.SmartServerResponse(('no', )),
449
request.execute('empty'))
451
def test_outside_root_client_path(self):
452
backing = self.get_transport()
453
request = smart_dir.SmartServerRequestOpenBzrDir(backing,
454
root_client_path='root')
455
self.assertEqual(smart_req.SmartServerResponse(('no', )),
456
request.execute('not-root'))
459
class TestSmartServerRequestOpenBzrDir_2_1(tests.TestCaseWithMemoryTransport):
461
def test_no_directory(self):
462
backing = self.get_transport()
463
request = smart_dir.SmartServerRequestOpenBzrDir_2_1(backing)
464
self.assertEqual(smart_req.SmartServerResponse(('no', )),
465
request.execute('does-not-exist'))
467
def test_empty_directory(self):
468
backing = self.get_transport()
469
backing.mkdir('empty')
470
request = smart_dir.SmartServerRequestOpenBzrDir_2_1(backing)
471
self.assertEqual(smart_req.SmartServerResponse(('no', )),
472
request.execute('empty'))
474
def test_present_without_workingtree(self):
475
backing = self.get_transport()
476
request = smart_dir.SmartServerRequestOpenBzrDir_2_1(backing)
477
self.make_bzrdir('.')
478
self.assertEqual(smart_req.SmartServerResponse(('yes', 'no')),
481
def test_outside_root_client_path(self):
482
backing = self.get_transport()
483
request = smart_dir.SmartServerRequestOpenBzrDir_2_1(backing,
484
root_client_path='root')
485
self.assertEqual(smart_req.SmartServerResponse(('no',)),
486
request.execute('not-root'))
489
class TestSmartServerRequestOpenBzrDir_2_1_disk(TestCaseWithChrootedTransport):
491
def test_present_with_workingtree(self):
492
self.vfs_transport_factory = test_server.LocalURLServer
493
backing = self.get_transport()
494
request = smart_dir.SmartServerRequestOpenBzrDir_2_1(backing)
495
bd = self.make_bzrdir('.')
496
bd.create_repository()
498
bd.create_workingtree()
499
self.assertEqual(smart_req.SmartServerResponse(('yes', 'yes')),
503
393
class TestSmartServerRequestOpenBranch(TestCaseWithChrootedTransport):
505
395
def test_no_branch(self):
506
396
"""When there is no branch, ('nobranch', ) is returned."""
507
397
backing = self.get_transport()
508
request = smart_dir.SmartServerRequestOpenBranch(backing)
398
request = smart.bzrdir.SmartServerRequestOpenBranch(backing)
509
399
self.make_bzrdir('.')
510
self.assertEqual(smart_req.SmartServerResponse(('nobranch', )),
400
self.assertEqual(SmartServerResponse(('nobranch', )),
511
401
request.execute(''))
513
403
def test_branch(self):
514
404
"""When there is a branch, 'ok' is returned."""
515
405
backing = self.get_transport()
516
request = smart_dir.SmartServerRequestOpenBranch(backing)
406
request = smart.bzrdir.SmartServerRequestOpenBranch(backing)
517
407
self.make_branch('.')
518
self.assertEqual(smart_req.SmartServerResponse(('ok', '')),
408
self.assertEqual(SmartServerResponse(('ok', '')),
519
409
request.execute(''))
521
411
def test_branch_reference(self):
522
412
"""When there is a branch reference, the reference URL is returned."""
523
self.vfs_transport_factory = test_server.LocalURLServer
524
413
backing = self.get_transport()
525
request = smart_dir.SmartServerRequestOpenBranch(backing)
414
request = smart.bzrdir.SmartServerRequestOpenBranch(backing)
526
415
branch = self.make_branch('branch')
527
416
checkout = branch.create_checkout('reference',lightweight=True)
528
reference_url = _mod_branch.BranchReferenceFormat().get_reference(
417
reference_url = BranchReferenceFormat().get_reference(checkout.bzrdir)
530
418
self.assertFileEqual(reference_url, 'reference/.bzr/branch/location')
531
self.assertEqual(smart_req.SmartServerResponse(('ok', reference_url)),
419
self.assertEqual(SmartServerResponse(('ok', reference_url)),
532
420
request.execute('reference'))
534
def test_notification_on_branch_from_repository(self):
535
"""When there is a repository, the error should return details."""
536
backing = self.get_transport()
537
request = smart_dir.SmartServerRequestOpenBranch(backing)
538
repo = self.make_repository('.')
539
self.assertEqual(smart_req.SmartServerResponse(('nobranch',)),
543
423
class TestSmartServerRequestOpenBranchV2(TestCaseWithChrootedTransport):
546
426
"""When there is no branch, ('nobranch', ) is returned."""
547
427
backing = self.get_transport()
548
428
self.make_bzrdir('.')
549
request = smart_dir.SmartServerRequestOpenBranchV2(backing)
550
self.assertEqual(smart_req.SmartServerResponse(('nobranch', )),
553
def test_branch(self):
554
"""When there is a branch, 'ok' is returned."""
555
backing = self.get_transport()
556
expected = self.make_branch('.')._format.network_name()
557
request = smart_dir.SmartServerRequestOpenBranchV2(backing)
558
self.assertEqual(smart_req.SuccessfulSmartServerResponse(
559
('branch', expected)),
562
def test_branch_reference(self):
563
"""When there is a branch reference, the reference URL is returned."""
564
self.vfs_transport_factory = test_server.LocalURLServer
565
backing = self.get_transport()
566
request = smart_dir.SmartServerRequestOpenBranchV2(backing)
567
branch = self.make_branch('branch')
568
checkout = branch.create_checkout('reference',lightweight=True)
569
reference_url = _mod_branch.BranchReferenceFormat().get_reference(
571
self.assertFileEqual(reference_url, 'reference/.bzr/branch/location')
572
self.assertEqual(smart_req.SuccessfulSmartServerResponse(
573
('ref', reference_url)),
574
request.execute('reference'))
576
def test_stacked_branch(self):
577
"""Opening a stacked branch does not open the stacked-on branch."""
578
trunk = self.make_branch('trunk')
579
feature = self.make_branch('feature')
580
feature.set_stacked_on_url(trunk.base)
582
_mod_branch.Branch.hooks.install_named_hook(
583
'open', opened_branches.append, None)
584
backing = self.get_transport()
585
request = smart_dir.SmartServerRequestOpenBranchV2(backing)
588
response = request.execute('feature')
590
request.teardown_jail()
591
expected_format = feature._format.network_name()
592
self.assertEqual(smart_req.SuccessfulSmartServerResponse(
593
('branch', expected_format)),
595
self.assertLength(1, opened_branches)
597
def test_notification_on_branch_from_repository(self):
598
"""When there is a repository, the error should return details."""
599
backing = self.get_transport()
600
request = smart_dir.SmartServerRequestOpenBranchV2(backing)
601
repo = self.make_repository('.')
602
self.assertEqual(smart_req.SmartServerResponse(('nobranch',)),
606
class TestSmartServerRequestOpenBranchV3(TestCaseWithChrootedTransport):
608
def test_no_branch(self):
609
"""When there is no branch, ('nobranch', ) is returned."""
610
backing = self.get_transport()
611
self.make_bzrdir('.')
612
request = smart_dir.SmartServerRequestOpenBranchV3(backing)
613
self.assertEqual(smart_req.SmartServerResponse(('nobranch',)),
616
def test_branch(self):
617
"""When there is a branch, 'ok' is returned."""
618
backing = self.get_transport()
619
expected = self.make_branch('.')._format.network_name()
620
request = smart_dir.SmartServerRequestOpenBranchV3(backing)
621
self.assertEqual(smart_req.SuccessfulSmartServerResponse(
622
('branch', expected)),
625
def test_branch_reference(self):
626
"""When there is a branch reference, the reference URL is returned."""
627
self.vfs_transport_factory = test_server.LocalURLServer
628
backing = self.get_transport()
629
request = smart_dir.SmartServerRequestOpenBranchV3(backing)
630
branch = self.make_branch('branch')
631
checkout = branch.create_checkout('reference',lightweight=True)
632
reference_url = _mod_branch.BranchReferenceFormat().get_reference(
634
self.assertFileEqual(reference_url, 'reference/.bzr/branch/location')
635
self.assertEqual(smart_req.SuccessfulSmartServerResponse(
636
('ref', reference_url)),
637
request.execute('reference'))
639
def test_stacked_branch(self):
640
"""Opening a stacked branch does not open the stacked-on branch."""
641
trunk = self.make_branch('trunk')
642
feature = self.make_branch('feature')
643
feature.set_stacked_on_url(trunk.base)
645
_mod_branch.Branch.hooks.install_named_hook(
646
'open', opened_branches.append, None)
647
backing = self.get_transport()
648
request = smart_dir.SmartServerRequestOpenBranchV3(backing)
651
response = request.execute('feature')
653
request.teardown_jail()
654
expected_format = feature._format.network_name()
655
self.assertEqual(smart_req.SuccessfulSmartServerResponse(
656
('branch', expected_format)),
658
self.assertLength(1, opened_branches)
660
def test_notification_on_branch_from_repository(self):
661
"""When there is a repository, the error should return details."""
662
backing = self.get_transport()
663
request = smart_dir.SmartServerRequestOpenBranchV3(backing)
664
repo = self.make_repository('.')
665
self.assertEqual(smart_req.SmartServerResponse(
666
('nobranch', 'location is a repository')),
429
request = smart.bzrdir.SmartServerRequestOpenBranchV2(backing)
430
self.assertEqual(SmartServerResponse(('nobranch', )),
433
def test_branch(self):
434
"""When there is a branch, 'ok' is returned."""
435
backing = self.get_transport()
436
expected = self.make_branch('.')._format.network_name()
437
request = smart.bzrdir.SmartServerRequestOpenBranchV2(backing)
438
self.assertEqual(SuccessfulSmartServerResponse(('branch', expected)),
441
def test_branch_reference(self):
442
"""When there is a branch reference, the reference URL is returned."""
443
backing = self.get_transport()
444
request = smart.bzrdir.SmartServerRequestOpenBranchV2(backing)
445
branch = self.make_branch('branch')
446
checkout = branch.create_checkout('reference',lightweight=True)
447
reference_url = BranchReferenceFormat().get_reference(checkout.bzrdir)
448
self.assertFileEqual(reference_url, 'reference/.bzr/branch/location')
449
self.assertEqual(SuccessfulSmartServerResponse(('ref', reference_url)),
450
request.execute('reference'))
452
def test_stacked_branch(self):
453
"""Opening a stacked branch does not open the stacked-on branch."""
454
trunk = self.make_branch('trunk')
455
feature = self.make_branch('feature', format='1.9')
456
feature.set_stacked_on_url(trunk.base)
458
Branch.hooks.install_named_hook('open', opened_branches.append, None)
459
backing = self.get_transport()
460
request = smart.bzrdir.SmartServerRequestOpenBranchV2(backing)
463
response = request.execute('feature')
465
request.teardown_jail()
466
expected_format = feature._format.network_name()
468
SuccessfulSmartServerResponse(('branch', expected_format)),
470
self.assertLength(1, opened_branches)
670
473
class TestSmartServerRequestRevisionHistory(tests.TestCaseWithMemoryTransport):
705
508
def test_branch_reference(self):
706
509
"""When there is a branch reference, NotBranchError is raised."""
707
510
backing = self.get_transport()
708
request = smart_branch.SmartServerBranchRequest(backing)
511
request = smart.branch.SmartServerBranchRequest(backing)
709
512
branch = self.make_branch('branch')
710
513
checkout = branch.create_checkout('reference',lightweight=True)
711
514
self.assertRaises(errors.NotBranchError,
712
515
request.execute, 'checkout')
715
class TestSmartServerBranchRequestLastRevisionInfo(
716
tests.TestCaseWithMemoryTransport):
518
class TestSmartServerBranchRequestLastRevisionInfo(tests.TestCaseWithMemoryTransport):
718
520
def test_empty(self):
719
521
"""For an empty branch, the result is ('ok', '0', 'null:')."""
720
522
backing = self.get_transport()
721
request = smart_branch.SmartServerBranchRequestLastRevisionInfo(backing)
523
request = smart.branch.SmartServerBranchRequestLastRevisionInfo(backing)
722
524
self.make_branch('.')
723
self.assertEqual(smart_req.SmartServerResponse(('ok', '0', 'null:')),
525
self.assertEqual(SmartServerResponse(('ok', '0', 'null:')),
724
526
request.execute(''))
726
528
def test_not_empty(self):
727
529
"""For a non-empty branch, the result is ('ok', 'revno', 'revid')."""
728
530
backing = self.get_transport()
729
request = smart_branch.SmartServerBranchRequestLastRevisionInfo(backing)
531
request = smart.branch.SmartServerBranchRequestLastRevisionInfo(backing)
730
532
tree = self.make_branch_and_memory_tree('.')
731
533
tree.lock_write()
778
579
def test_value_name(self):
779
580
branch = self.make_branch('.')
780
request = smart_branch.SmartServerBranchRequestSetConfigOption(
581
request = smart.branch.SmartServerBranchRequestSetConfigOption(
781
582
branch.bzrdir.root_transport)
782
583
branch_token, repo_token = self.get_lock_tokens(branch)
783
584
config = branch._get_config()
784
585
result = request.execute('', branch_token, repo_token, 'bar', 'foo',
786
self.assertEqual(smart_req.SuccessfulSmartServerResponse(()), result)
587
self.assertEqual(SuccessfulSmartServerResponse(()), result)
787
588
self.assertEqual('bar', config.get_option('foo'))
791
592
def test_value_name_section(self):
792
593
branch = self.make_branch('.')
793
request = smart_branch.SmartServerBranchRequestSetConfigOption(
594
request = smart.branch.SmartServerBranchRequestSetConfigOption(
794
595
branch.bzrdir.root_transport)
795
596
branch_token, repo_token = self.get_lock_tokens(branch)
796
597
config = branch._get_config()
797
598
result = request.execute('', branch_token, repo_token, 'bar', 'foo',
799
self.assertEqual(smart_req.SuccessfulSmartServerResponse(()), result)
600
self.assertEqual(SuccessfulSmartServerResponse(()), result)
800
601
self.assertEqual('bar', config.get_option('foo', 'gam'))
805
class TestSmartServerBranchRequestSetConfigOptionDict(TestLockedBranch):
808
TestLockedBranch.setUp(self)
809
# A dict with non-ascii keys and values to exercise unicode
811
self.encoded_value_dict = (
812
'd5:ascii1:a11:unicode \xe2\x8c\x9a3:\xe2\x80\xbde')
814
'ascii': 'a', u'unicode \N{WATCH}': u'\N{INTERROBANG}'}
816
def test_value_name(self):
817
branch = self.make_branch('.')
818
request = smart_branch.SmartServerBranchRequestSetConfigOptionDict(
819
branch.bzrdir.root_transport)
820
branch_token, repo_token = self.get_lock_tokens(branch)
821
config = branch._get_config()
822
result = request.execute('', branch_token, repo_token,
823
self.encoded_value_dict, 'foo', '')
824
self.assertEqual(smart_req.SuccessfulSmartServerResponse(()), result)
825
self.assertEqual(self.value_dict, config.get_option('foo'))
829
def test_value_name_section(self):
830
branch = self.make_branch('.')
831
request = smart_branch.SmartServerBranchRequestSetConfigOptionDict(
832
branch.bzrdir.root_transport)
833
branch_token, repo_token = self.get_lock_tokens(branch)
834
config = branch._get_config()
835
result = request.execute('', branch_token, repo_token,
836
self.encoded_value_dict, 'foo', 'gam')
837
self.assertEqual(smart_req.SuccessfulSmartServerResponse(()), result)
838
self.assertEqual(self.value_dict, config.get_option('foo', 'gam'))
843
606
class TestSmartServerBranchRequestSetTagsBytes(TestLockedBranch):
844
607
# Only called when the branch format and tags match [yay factory
845
608
# methods] so only need to test straight forward cases.
1097
859
def test_get_parent_none(self):
1098
860
base_branch = self.make_branch('base')
1099
request = smart_branch.SmartServerBranchGetParent(self.get_transport())
861
request = smart.branch.SmartServerBranchGetParent(self.get_transport())
1100
862
response = request.execute('base')
1101
863
self.assertEquals(
1102
smart_req.SuccessfulSmartServerResponse(('',)), response)
864
SuccessfulSmartServerResponse(('',)), response)
1104
866
def test_get_parent_something(self):
1105
867
base_branch = self.make_branch('base')
1106
868
base_branch.set_parent(self.get_url('foo'))
1107
request = smart_branch.SmartServerBranchGetParent(self.get_transport())
869
request = smart.branch.SmartServerBranchGetParent(self.get_transport())
1108
870
response = request.execute('base')
1109
871
self.assertEquals(
1110
smart_req.SuccessfulSmartServerResponse(("../foo",)),
872
SuccessfulSmartServerResponse(("../foo",)),
1114
class TestSmartServerBranchRequestSetParent(TestLockedBranch):
876
class TestSmartServerBranchRequestSetParent(tests.TestCaseWithMemoryTransport):
1116
878
def test_set_parent_none(self):
1117
879
branch = self.make_branch('base', format="1.9")
1118
880
branch.lock_write()
1119
881
branch._set_parent_location('foo')
1121
request = smart_branch.SmartServerBranchRequestSetParentLocation(
883
request = smart.branch.SmartServerBranchRequestSetParentLocation(
1122
884
self.get_transport())
1123
branch_token, repo_token = self.get_lock_tokens(branch)
885
branch_token = branch.lock_write()
886
repo_token = branch.repository.lock_write()
1125
888
response = request.execute('base', branch_token, repo_token, '')
890
branch.repository.unlock()
1128
self.assertEqual(smart_req.SuccessfulSmartServerResponse(()), response)
892
self.assertEqual(SuccessfulSmartServerResponse(()), response)
1129
893
self.assertEqual(None, branch.get_parent())
1131
895
def test_set_parent_something(self):
1132
896
branch = self.make_branch('base', format="1.9")
1133
request = smart_branch.SmartServerBranchRequestSetParentLocation(
897
request = smart.branch.SmartServerBranchRequestSetParentLocation(
1134
898
self.get_transport())
1135
branch_token, repo_token = self.get_lock_tokens(branch)
899
branch_token = branch.lock_write()
900
repo_token = branch.repository.lock_write()
1137
902
response = request.execute('base', branch_token, repo_token,
905
branch.repository.unlock()
1141
self.assertEqual(smart_req.SuccessfulSmartServerResponse(()), response)
907
self.assertEqual(SuccessfulSmartServerResponse(()), response)
1142
908
self.assertEqual('http://bar/', branch.get_parent())
1145
class TestSmartServerBranchRequestGetTagsBytes(
1146
tests.TestCaseWithMemoryTransport):
911
class TestSmartServerBranchRequestGetTagsBytes(tests.TestCaseWithMemoryTransport):
1147
912
# Only called when the branch format and tags match [yay factory
1148
913
# methods] so only need to test straight forward cases.
1150
915
def test_get_bytes(self):
1151
916
base_branch = self.make_branch('base')
1152
request = smart_branch.SmartServerBranchGetTagsBytes(
917
request = smart.branch.SmartServerBranchGetTagsBytes(
1153
918
self.get_transport())
1154
919
response = request.execute('base')
1155
920
self.assertEquals(
1156
smart_req.SuccessfulSmartServerResponse(('',)), response)
921
SuccessfulSmartServerResponse(('',)), response)
1159
924
class TestSmartServerBranchRequestGetStackedOnURL(tests.TestCaseWithMemoryTransport):
1163
928
stacked_branch = self.make_branch('stacked', format='1.6')
1164
929
# typically should be relative
1165
930
stacked_branch.set_stacked_on_url('../base')
1166
request = smart_branch.SmartServerBranchRequestGetStackedOnURL(
931
request = smart.branch.SmartServerBranchRequestGetStackedOnURL(
1167
932
self.get_transport())
1168
933
response = request.execute('stacked')
1169
934
self.assertEquals(
1170
smart_req.SmartServerResponse(('ok', '../base')),
935
SmartServerResponse(('ok', '../base')),
1174
class TestSmartServerBranchRequestLockWrite(TestLockedBranch):
939
class TestSmartServerBranchRequestLockWrite(tests.TestCaseWithMemoryTransport):
1176
941
def setUp(self):
1177
942
tests.TestCaseWithMemoryTransport.setUp(self)
1179
944
def test_lock_write_on_unlocked_branch(self):
1180
945
backing = self.get_transport()
1181
request = smart_branch.SmartServerBranchRequestLockWrite(backing)
946
request = smart.branch.SmartServerBranchRequestLockWrite(backing)
1182
947
branch = self.make_branch('.', format='knit')
1183
948
repository = branch.repository
1184
949
response = request.execute('')
1185
950
branch_nonce = branch.control_files._lock.peek().get('nonce')
1186
951
repository_nonce = repository.control_files._lock.peek().get('nonce')
1187
self.assertEqual(smart_req.SmartServerResponse(
1188
('ok', branch_nonce, repository_nonce)),
953
SmartServerResponse(('ok', branch_nonce, repository_nonce)),
1190
955
# The branch (and associated repository) is now locked. Verify that
1191
956
# with a new branch object.
1192
957
new_branch = repository.bzrdir.open_branch()
1193
958
self.assertRaises(errors.LockContention, new_branch.lock_write)
1195
request = smart_branch.SmartServerBranchRequestUnlock(backing)
960
request = smart.branch.SmartServerBranchRequestUnlock(backing)
1196
961
response = request.execute('', branch_nonce, repository_nonce)
1198
963
def test_lock_write_on_locked_branch(self):
1199
964
backing = self.get_transport()
1200
request = smart_branch.SmartServerBranchRequestLockWrite(backing)
965
request = smart.branch.SmartServerBranchRequestLockWrite(backing)
1201
966
branch = self.make_branch('.')
1202
branch_token = branch.lock_write().branch_token
967
branch_token = branch.lock_write()
1203
968
branch.leave_lock_in_place()
1205
970
response = request.execute('')
1206
971
self.assertEqual(
1207
smart_req.SmartServerResponse(('LockContention',)), response)
972
SmartServerResponse(('LockContention',)), response)
1209
974
branch.lock_write(branch_token)
1210
975
branch.dont_leave_lock_in_place()
1308
1078
def test_unlock_on_unlocked_branch_unlocked_repo(self):
1309
1079
backing = self.get_transport()
1310
request = smart_branch.SmartServerBranchRequestUnlock(backing)
1080
request = smart.branch.SmartServerBranchRequestUnlock(backing)
1311
1081
branch = self.make_branch('.', format='knit')
1312
1082
response = request.execute(
1313
1083
'', 'branch token', 'repo token')
1314
1084
self.assertEqual(
1315
smart_req.SmartServerResponse(('TokenMismatch',)), response)
1085
SmartServerResponse(('TokenMismatch',)), response)
1317
1087
def test_unlock_on_unlocked_branch_locked_repo(self):
1318
1088
backing = self.get_transport()
1319
request = smart_branch.SmartServerBranchRequestUnlock(backing)
1089
request = smart.branch.SmartServerBranchRequestUnlock(backing)
1320
1090
branch = self.make_branch('.', format='knit')
1321
1091
# Lock the repository.
1322
repo_token = branch.repository.lock_write().repository_token
1092
repo_token = branch.repository.lock_write()
1323
1093
branch.repository.leave_lock_in_place()
1324
1094
branch.repository.unlock()
1325
1095
# Issue branch lock_write request on the unlocked branch (with locked
1327
response = request.execute('', 'branch token', repo_token)
1097
response = request.execute(
1098
'', 'branch token', repo_token)
1328
1099
self.assertEqual(
1329
smart_req.SmartServerResponse(('TokenMismatch',)), response)
1100
SmartServerResponse(('TokenMismatch',)), response)
1331
1102
branch.repository.lock_write(repo_token)
1332
1103
branch.repository.dont_leave_lock_in_place()
1354
1125
def test_trivial_bzipped(self):
1355
1126
# This tests that the wire encoding is actually bzipped
1356
1127
backing = self.get_transport()
1357
request = smart_repo.SmartServerRepositoryGetParentMap(backing)
1128
request = smart.repository.SmartServerRepositoryGetParentMap(backing)
1358
1129
tree = self.make_branch_and_memory_tree('.')
1360
1131
self.assertEqual(None,
1361
1132
request.execute('', 'missing-id'))
1362
1133
# Note that it returns a body that is bzipped.
1363
1134
self.assertEqual(
1364
smart_req.SuccessfulSmartServerResponse(('ok', ), bz2.compress('')),
1135
SuccessfulSmartServerResponse(('ok', ), bz2.compress('')),
1365
1136
request.do_body('\n\n0\n'))
1367
1138
def test_trivial_include_missing(self):
1368
1139
backing = self.get_transport()
1369
request = smart_repo.SmartServerRepositoryGetParentMap(backing)
1140
request = smart.repository.SmartServerRepositoryGetParentMap(backing)
1370
1141
tree = self.make_branch_and_memory_tree('.')
1372
1143
self.assertEqual(None,
1373
1144
request.execute('', 'missing-id', 'include-missing:'))
1374
1145
self.assertEqual(
1375
smart_req.SuccessfulSmartServerResponse(('ok', ),
1146
SuccessfulSmartServerResponse(('ok', ),
1376
1147
bz2.compress('missing:missing-id')),
1377
1148
request.do_body('\n\n0\n'))
1380
class TestSmartServerRepositoryGetRevisionGraph(
1381
tests.TestCaseWithMemoryTransport):
1151
class TestSmartServerRepositoryGetRevisionGraph(tests.TestCaseWithMemoryTransport):
1383
1153
def test_none_argument(self):
1384
1154
backing = self.get_transport()
1385
request = smart_repo.SmartServerRepositoryGetRevisionGraph(backing)
1155
request = smart.repository.SmartServerRepositoryGetRevisionGraph(backing)
1386
1156
tree = self.make_branch_and_memory_tree('.')
1387
1157
tree.lock_write()
1443
1212
tree.commit('2nd commit', rev_id=rev2_id_utf8)
1446
self.assertEqual(smart_req.SmartServerResponse(('ok', rev1_id_utf8)),
1215
self.assertEqual(SmartServerResponse(('ok', rev1_id_utf8)),
1447
1216
request.execute('', 1, (2, rev2_id_utf8)))
1449
1218
def test_known_revid_missing(self):
1450
1219
backing = self.get_transport()
1451
request = smart_repo.SmartServerRepositoryGetRevIdForRevno(backing)
1220
request = smart.repository.SmartServerRepositoryGetRevIdForRevno(backing)
1452
1221
repo = self.make_repository('.')
1453
1222
self.assertEqual(
1454
smart_req.FailedSmartServerResponse(('nosuchrevision', 'ghost')),
1223
FailedSmartServerResponse(('nosuchrevision', 'ghost')),
1455
1224
request.execute('', 1, (2, 'ghost')))
1457
1226
def test_history_incomplete(self):
1458
1227
backing = self.get_transport()
1459
request = smart_repo.SmartServerRepositoryGetRevIdForRevno(backing)
1228
request = smart.repository.SmartServerRepositoryGetRevIdForRevno(backing)
1460
1229
parent = self.make_branch_and_memory_tree('parent', format='1.9')
1461
1230
parent.lock_write()
1462
1231
parent.add([''], ['TREE_ROOT'])
1621
1389
def test_lock_write_on_unlocked_repo(self):
1622
1390
backing = self.get_transport()
1623
request = smart_repo.SmartServerRepositoryLockWrite(backing)
1391
request = smart.repository.SmartServerRepositoryLockWrite(backing)
1624
1392
repository = self.make_repository('.', format='knit')
1625
1393
response = request.execute('')
1626
1394
nonce = repository.control_files._lock.peek().get('nonce')
1627
self.assertEqual(smart_req.SmartServerResponse(('ok', nonce)), response)
1395
self.assertEqual(SmartServerResponse(('ok', nonce)), response)
1628
1396
# The repository is now locked. Verify that with a new repository
1630
1398
new_repo = repository.bzrdir.open_repository()
1631
1399
self.assertRaises(errors.LockContention, new_repo.lock_write)
1633
request = smart_repo.SmartServerRepositoryUnlock(backing)
1401
request = smart.repository.SmartServerRepositoryUnlock(backing)
1634
1402
response = request.execute('', nonce)
1636
1404
def test_lock_write_on_locked_repo(self):
1637
1405
backing = self.get_transport()
1638
request = smart_repo.SmartServerRepositoryLockWrite(backing)
1406
request = smart.repository.SmartServerRepositoryLockWrite(backing)
1639
1407
repository = self.make_repository('.', format='knit')
1640
repo_token = repository.lock_write().repository_token
1408
repo_token = repository.lock_write()
1641
1409
repository.leave_lock_in_place()
1642
1410
repository.unlock()
1643
1411
response = request.execute('')
1644
1412
self.assertEqual(
1645
smart_req.SmartServerResponse(('LockContention',)), response)
1413
SmartServerResponse(('LockContention',)), response)
1647
1415
repository.lock_write(repo_token)
1648
1416
repository.dont_leave_lock_in_place()
1669
1437
def test_insert_stream_empty(self):
1670
1438
backing = self.get_transport()
1671
request = smart_repo.SmartServerRepositoryInsertStream(backing)
1439
request = smart.repository.SmartServerRepositoryInsertStream(backing)
1672
1440
repository = self.make_repository('.')
1673
1441
response = request.execute('', '')
1674
1442
self.assertEqual(None, response)
1675
1443
response = request.do_chunk(self.make_empty_byte_stream(repository))
1676
1444
self.assertEqual(None, response)
1677
1445
response = request.do_end()
1678
self.assertEqual(smart_req.SmartServerResponse(('ok', )), response)
1446
self.assertEqual(SmartServerResponse(('ok', )), response)
1681
1449
class TestSmartServerRepositoryInsertStreamLocked(TestInsertStreamBase):
1683
1451
def test_insert_stream_empty(self):
1684
1452
backing = self.get_transport()
1685
request = smart_repo.SmartServerRepositoryInsertStreamLocked(
1453
request = smart.repository.SmartServerRepositoryInsertStreamLocked(
1687
1455
repository = self.make_repository('.', format='knit')
1688
lock_token = repository.lock_write().repository_token
1456
lock_token = repository.lock_write()
1689
1457
response = request.execute('', '', lock_token)
1690
1458
self.assertEqual(None, response)
1691
1459
response = request.do_chunk(self.make_empty_byte_stream(repository))
1692
1460
self.assertEqual(None, response)
1693
1461
response = request.do_end()
1694
self.assertEqual(smart_req.SmartServerResponse(('ok', )), response)
1462
self.assertEqual(SmartServerResponse(('ok', )), response)
1695
1463
repository.unlock()
1697
1465
def test_insert_stream_with_wrong_lock_token(self):
1698
1466
backing = self.get_transport()
1699
request = smart_repo.SmartServerRepositoryInsertStreamLocked(
1467
request = smart.repository.SmartServerRepositoryInsertStreamLocked(
1701
1469
repository = self.make_repository('.', format='knit')
1702
lock_token = repository.lock_write().repository_token
1470
lock_token = repository.lock_write()
1703
1471
self.assertRaises(
1704
1472
errors.TokenMismatch, request.execute, '', '', 'wrong-token')
1705
1473
repository.unlock()
1729
1497
def test_unlock_on_unlocked_repo(self):
1730
1498
backing = self.get_transport()
1731
request = smart_repo.SmartServerRepositoryUnlock(backing)
1499
request = smart.repository.SmartServerRepositoryUnlock(backing)
1732
1500
repository = self.make_repository('.', format='knit')
1733
1501
response = request.execute('', 'some token')
1734
1502
self.assertEqual(
1735
smart_req.SmartServerResponse(('TokenMismatch',)), response)
1503
SmartServerResponse(('TokenMismatch',)), response)
1738
1506
class TestSmartServerIsReadonly(tests.TestCaseWithMemoryTransport):
1740
1508
def test_is_readonly_no(self):
1741
1509
backing = self.get_transport()
1742
request = smart_req.SmartServerIsReadonly(backing)
1510
request = smart.request.SmartServerIsReadonly(backing)
1743
1511
response = request.execute()
1744
1512
self.assertEqual(
1745
smart_req.SmartServerResponse(('no',)), response)
1513
SmartServerResponse(('no',)), response)
1747
1515
def test_is_readonly_yes(self):
1748
1516
backing = self.get_readonly_transport()
1749
request = smart_req.SmartServerIsReadonly(backing)
1517
request = smart.request.SmartServerIsReadonly(backing)
1750
1518
response = request.execute()
1751
1519
self.assertEqual(
1752
smart_req.SmartServerResponse(('yes',)), response)
1755
class TestSmartServerRepositorySetMakeWorkingTrees(
1756
tests.TestCaseWithMemoryTransport):
1520
SmartServerResponse(('yes',)), response)
1523
class TestSmartServerRepositorySetMakeWorkingTrees(tests.TestCaseWithMemoryTransport):
1758
1525
def test_set_false(self):
1759
1526
backing = self.get_transport()
1760
1527
repo = self.make_repository('.', shared=True)
1761
1528
repo.set_make_working_trees(True)
1762
request_class = smart_repo.SmartServerRepositorySetMakeWorkingTrees
1529
request_class = smart.repository.SmartServerRepositorySetMakeWorkingTrees
1763
1530
request = request_class(backing)
1764
self.assertEqual(smart_req.SuccessfulSmartServerResponse(('ok',)),
1531
self.assertEqual(SuccessfulSmartServerResponse(('ok',)),
1765
1532
request.execute('', 'False'))
1766
1533
repo = repo.bzrdir.open_repository()
1767
1534
self.assertFalse(repo.make_working_trees())
1849
1603
"""All registered request_handlers can be found."""
1850
1604
# If there's a typo in a register_lazy call, this loop will fail with
1851
1605
# an AttributeError.
1852
for key, item in smart_req.request_handlers.iteritems():
1606
for key, item in smart.request.request_handlers.iteritems():
1855
1609
def assertHandlerEqual(self, verb, handler):
1856
self.assertEqual(smart_req.request_handlers.get(verb), handler)
1610
self.assertEqual(smart.request.request_handlers.get(verb), handler)
1858
1612
def test_registered_methods(self):
1859
1613
"""Test that known methods are registered to the correct object."""
1860
1614
self.assertHandlerEqual('Branch.get_config_file',
1861
smart_branch.SmartServerBranchGetConfigFile)
1615
smart.branch.SmartServerBranchGetConfigFile)
1862
1616
self.assertHandlerEqual('Branch.get_parent',
1863
smart_branch.SmartServerBranchGetParent)
1617
smart.branch.SmartServerBranchGetParent)
1864
1618
self.assertHandlerEqual('Branch.get_tags_bytes',
1865
smart_branch.SmartServerBranchGetTagsBytes)
1619
smart.branch.SmartServerBranchGetTagsBytes)
1866
1620
self.assertHandlerEqual('Branch.lock_write',
1867
smart_branch.SmartServerBranchRequestLockWrite)
1621
smart.branch.SmartServerBranchRequestLockWrite)
1868
1622
self.assertHandlerEqual('Branch.last_revision_info',
1869
smart_branch.SmartServerBranchRequestLastRevisionInfo)
1623
smart.branch.SmartServerBranchRequestLastRevisionInfo)
1870
1624
self.assertHandlerEqual('Branch.revision_history',
1871
smart_branch.SmartServerRequestRevisionHistory)
1625
smart.branch.SmartServerRequestRevisionHistory)
1872
1626
self.assertHandlerEqual('Branch.set_config_option',
1873
smart_branch.SmartServerBranchRequestSetConfigOption)
1627
smart.branch.SmartServerBranchRequestSetConfigOption)
1874
1628
self.assertHandlerEqual('Branch.set_last_revision',
1875
smart_branch.SmartServerBranchRequestSetLastRevision)
1629
smart.branch.SmartServerBranchRequestSetLastRevision)
1876
1630
self.assertHandlerEqual('Branch.set_last_revision_info',
1877
smart_branch.SmartServerBranchRequestSetLastRevisionInfo)
1631
smart.branch.SmartServerBranchRequestSetLastRevisionInfo)
1878
1632
self.assertHandlerEqual('Branch.set_last_revision_ex',
1879
smart_branch.SmartServerBranchRequestSetLastRevisionEx)
1633
smart.branch.SmartServerBranchRequestSetLastRevisionEx)
1880
1634
self.assertHandlerEqual('Branch.set_parent_location',
1881
smart_branch.SmartServerBranchRequestSetParentLocation)
1635
smart.branch.SmartServerBranchRequestSetParentLocation)
1882
1636
self.assertHandlerEqual('Branch.unlock',
1883
smart_branch.SmartServerBranchRequestUnlock)
1637
smart.branch.SmartServerBranchRequestUnlock)
1884
1638
self.assertHandlerEqual('BzrDir.find_repository',
1885
smart_dir.SmartServerRequestFindRepositoryV1)
1639
smart.bzrdir.SmartServerRequestFindRepositoryV1)
1886
1640
self.assertHandlerEqual('BzrDir.find_repositoryV2',
1887
smart_dir.SmartServerRequestFindRepositoryV2)
1641
smart.bzrdir.SmartServerRequestFindRepositoryV2)
1888
1642
self.assertHandlerEqual('BzrDirFormat.initialize',
1889
smart_dir.SmartServerRequestInitializeBzrDir)
1643
smart.bzrdir.SmartServerRequestInitializeBzrDir)
1890
1644
self.assertHandlerEqual('BzrDirFormat.initialize_ex_1.16',
1891
smart_dir.SmartServerRequestBzrDirInitializeEx)
1645
smart.bzrdir.SmartServerRequestBzrDirInitializeEx)
1892
1646
self.assertHandlerEqual('BzrDir.cloning_metadir',
1893
smart_dir.SmartServerBzrDirRequestCloningMetaDir)
1647
smart.bzrdir.SmartServerBzrDirRequestCloningMetaDir)
1894
1648
self.assertHandlerEqual('BzrDir.get_config_file',
1895
smart_dir.SmartServerBzrDirRequestConfigFile)
1649
smart.bzrdir.SmartServerBzrDirRequestConfigFile)
1896
1650
self.assertHandlerEqual('BzrDir.open_branch',
1897
smart_dir.SmartServerRequestOpenBranch)
1651
smart.bzrdir.SmartServerRequestOpenBranch)
1898
1652
self.assertHandlerEqual('BzrDir.open_branchV2',
1899
smart_dir.SmartServerRequestOpenBranchV2)
1900
self.assertHandlerEqual('BzrDir.open_branchV3',
1901
smart_dir.SmartServerRequestOpenBranchV3)
1653
smart.bzrdir.SmartServerRequestOpenBranchV2)
1902
1654
self.assertHandlerEqual('PackRepository.autopack',
1903
smart_packrepo.SmartServerPackRepositoryAutopack)
1655
smart.packrepository.SmartServerPackRepositoryAutopack)
1904
1656
self.assertHandlerEqual('Repository.gather_stats',
1905
smart_repo.SmartServerRepositoryGatherStats)
1657
smart.repository.SmartServerRepositoryGatherStats)
1906
1658
self.assertHandlerEqual('Repository.get_parent_map',
1907
smart_repo.SmartServerRepositoryGetParentMap)
1659
smart.repository.SmartServerRepositoryGetParentMap)
1908
1660
self.assertHandlerEqual('Repository.get_rev_id_for_revno',
1909
smart_repo.SmartServerRepositoryGetRevIdForRevno)
1661
smart.repository.SmartServerRepositoryGetRevIdForRevno)
1910
1662
self.assertHandlerEqual('Repository.get_revision_graph',
1911
smart_repo.SmartServerRepositoryGetRevisionGraph)
1663
smart.repository.SmartServerRepositoryGetRevisionGraph)
1912
1664
self.assertHandlerEqual('Repository.get_stream',
1913
smart_repo.SmartServerRepositoryGetStream)
1665
smart.repository.SmartServerRepositoryGetStream)
1914
1666
self.assertHandlerEqual('Repository.has_revision',
1915
smart_repo.SmartServerRequestHasRevision)
1667
smart.repository.SmartServerRequestHasRevision)
1916
1668
self.assertHandlerEqual('Repository.insert_stream',
1917
smart_repo.SmartServerRepositoryInsertStream)
1669
smart.repository.SmartServerRepositoryInsertStream)
1918
1670
self.assertHandlerEqual('Repository.insert_stream_locked',
1919
smart_repo.SmartServerRepositoryInsertStreamLocked)
1671
smart.repository.SmartServerRepositoryInsertStreamLocked)
1920
1672
self.assertHandlerEqual('Repository.is_shared',
1921
smart_repo.SmartServerRepositoryIsShared)
1673
smart.repository.SmartServerRepositoryIsShared)
1922
1674
self.assertHandlerEqual('Repository.lock_write',
1923
smart_repo.SmartServerRepositoryLockWrite)
1675
smart.repository.SmartServerRepositoryLockWrite)
1924
1676
self.assertHandlerEqual('Repository.tarball',
1925
smart_repo.SmartServerRepositoryTarball)
1677
smart.repository.SmartServerRepositoryTarball)
1926
1678
self.assertHandlerEqual('Repository.unlock',
1927
smart_repo.SmartServerRepositoryUnlock)
1679
smart.repository.SmartServerRepositoryUnlock)
1928
1680
self.assertHandlerEqual('Transport.is_readonly',
1929
smart_req.SmartServerIsReadonly)
1681
smart.request.SmartServerIsReadonly)