17
17
"""Tests that branch classes implement hook callouts correctly."""
19
from bzrlib.branch import Branch, ChangeBranchTipParams
20
from bzrlib.errors import HookFailed, TipChangeRejected
21
from bzrlib.remote import RemoteBranch
22
from bzrlib.revision import NULL_REVISION
23
from bzrlib.smart import server
24
from bzrlib.tests import TestCaseWithMemoryTransport
27
class ChangeBranchTipTestCase(TestCaseWithMemoryTransport):
20
branch as _mod_branch,
26
from bzrlib.tests import test_server
28
class ChangeBranchTipTestCase(tests.TestCaseWithMemoryTransport):
28
29
"""Base TestCase for testing pre/post_change_branch_tip hooks."""
30
31
def install_logging_hook(self, prefix):
54
55
if hook_calls is None:
55
56
hook_calls = self.hook_calls
56
if isinstance(branch, RemoteBranch):
57
if isinstance(branch, remote.RemoteBranch):
57
58
# For a remote branch, both the server and the client will raise
58
59
# this hook, and we see both in the test environment. The remote
59
60
# instance comes in between the clients - the client doe pre, the
85
86
def test_set_rh_empty_history(self):
86
87
branch = self.make_branch('source')
87
Branch.hooks.install_named_hook('set_rh', self.capture_set_rh_hook,
88
_mod_branch.Branch.hooks.install_named_hook(
89
'set_rh', self.capture_set_rh_hook, None)
89
90
branch.set_revision_history([])
90
91
expected_params = ('set_rh', branch, [], True)
91
92
self.assertHookCalls(expected_params, branch)
98
99
tree.commit('empty commit', rev_id='foo')
100
101
branch = tree.branch
101
Branch.hooks.install_named_hook('set_rh', self.capture_set_rh_hook,
102
_mod_branch.Branch.hooks.install_named_hook(
103
'set_rh', self.capture_set_rh_hook, None)
103
104
# some branches require that their history be set to a revision in the
105
106
branch.set_revision_history(['f\xc2\xb5'])
109
110
def test_set_rh_branch_is_locked(self):
110
111
branch = self.make_branch('source')
111
Branch.hooks.install_named_hook('set_rh', self.capture_set_rh_hook,
112
_mod_branch.Branch.hooks.install_named_hook(
113
'set_rh', self.capture_set_rh_hook, None)
113
114
branch.set_revision_history([])
114
115
expected_params = ('set_rh', branch, [], True)
115
116
self.assertHookCalls(expected_params, branch)
117
118
def test_set_rh_calls_all_hooks_no_errors(self):
118
119
branch = self.make_branch('source')
119
Branch.hooks.install_named_hook('set_rh', self.capture_set_rh_hook,
121
Branch.hooks.install_named_hook('set_rh', self.capture_set_rh_hook,
120
_mod_branch.Branch.hooks.install_named_hook(
121
'set_rh', self.capture_set_rh_hook, None)
122
_mod_branch.Branch.hooks.install_named_hook(
123
'set_rh', self.capture_set_rh_hook, None)
123
124
branch.set_revision_history([])
124
125
expected_calls = [('set_rh', branch, [], True),
125
126
('set_rh', branch, [], True),
127
if isinstance(branch, RemoteBranch):
128
if isinstance(branch, remote.RemoteBranch):
128
129
# For a remote branch, both the server and the client will raise
129
130
# set_rh, and the server will do so first because that is where
130
131
# the change takes place.
134
135
self.assertEqual(expected_calls, self.hook_calls)
137
class TestOpen(TestCaseWithMemoryTransport):
138
class TestOpen(tests.TestCaseWithMemoryTransport):
139
140
def capture_hook(self, branch):
140
141
self.hook_calls.append(branch)
142
143
def install_hook(self):
143
144
self.hook_calls = []
144
Branch.hooks.install_named_hook('open', self.capture_hook, None)
145
_mod_branch.Branch.hooks.install_named_hook(
146
'open', self.capture_hook, None)
146
148
def test_create(self):
147
149
self.install_hook()
148
150
b = self.make_branch('.')
149
if isinstance(b, RemoteBranch):
151
if isinstance(b, remote.RemoteBranch):
150
152
# RemoteBranch creation:
151
if (self.transport_readonly_server ==
152
server.ReadonlySmartTCPServer_for_testing_v2_only):
153
if (self.transport_readonly_server
154
== test_server.ReadonlySmartTCPServer_for_testing_v2_only):
154
156
self.assertEqual(3, len(self.hook_calls))
155
157
# creates the branch via the VFS (for older servers)
170
172
def test_open(self):
171
173
branch_url = self.make_branch('.').bzrdir.root_transport.base
172
174
self.install_hook()
173
b = Branch.open(branch_url)
174
if isinstance(b, RemoteBranch):
175
b = _mod_branch.Branch.open(branch_url)
176
if isinstance(b, remote.RemoteBranch):
175
177
self.assertEqual(3, len(self.hook_calls))
176
178
# open_branchV2 RPC
177
179
self.assertRealBranch(self.hook_calls[0])
185
187
def assertRealBranch(self, b):
186
188
# Branches opened on the server don't have comparable URLs, so we just
187
189
# assert that it is not a RemoteBranch.
188
self.assertIsInstance(b, Branch)
189
self.assertFalse(isinstance(b, RemoteBranch))
190
self.assertIsInstance(b, _mod_branch.Branch)
191
self.assertFalse(isinstance(b, remote.RemoteBranch))
192
194
class TestPreChangeBranchTip(ChangeBranchTipTestCase):
203
205
def assertBranchAtRevision1(params):
204
206
self.assertEquals(
205
207
(1, 'revid-one'), params.branch.last_revision_info())
206
Branch.hooks.install_named_hook(
208
_mod_branch.Branch.hooks.install_named_hook(
207
209
'pre_change_branch_tip', assertBranchAtRevision1, None)
208
branch.set_last_revision_info(0, NULL_REVISION)
210
branch.set_last_revision_info(0, revision.NULL_REVISION)
210
212
def test_hook_failure_prevents_change(self):
211
"""If a hook raises an exception, the change does not take effect.
213
Also, a HookFailed exception will be raised.
213
"""If a hook raises an exception, the change does not take effect."""
215
214
branch = self.make_branch_with_revision_ids(
216
215
'one-\xc2\xb5', 'two-\xc2\xb5')
217
216
class PearShapedError(Exception):
219
218
def hook_that_raises(params):
220
219
raise PearShapedError()
221
Branch.hooks.install_named_hook(
220
_mod_branch.Branch.hooks.install_named_hook(
222
221
'pre_change_branch_tip', hook_that_raises, None)
223
222
hook_failed_exc = self.assertRaises(
224
HookFailed, branch.set_last_revision_info, 0, NULL_REVISION)
225
self.assertIsInstance(hook_failed_exc.exc_value, PearShapedError)
224
branch.set_last_revision_info, 0, revision.NULL_REVISION)
226
225
# The revision info is unchanged.
227
226
self.assertEqual((2, 'two-\xc2\xb5'), branch.last_revision_info())
229
228
def test_empty_history(self):
230
229
branch = self.make_branch('source')
231
230
hook_calls = self.install_logging_hook('pre')
232
branch.set_last_revision_info(0, NULL_REVISION)
233
expected_params = ChangeBranchTipParams(
234
branch, 0, 0, NULL_REVISION, NULL_REVISION)
231
branch.set_last_revision_info(0, revision.NULL_REVISION)
232
expected_params = _mod_branch.ChangeBranchTipParams(
233
branch, 0, 0, revision.NULL_REVISION, revision.NULL_REVISION)
235
234
self.assertHookCalls(expected_params, branch, hook_calls, pre=True)
237
236
def test_nonempty_history(self):
242
241
'one-\xc2\xb5', 'two-\xc2\xb5')
243
242
hook_calls = self.install_logging_hook('pre')
244
243
branch.set_last_revision_info(1, 'one-\xc2\xb5')
245
expected_params = ChangeBranchTipParams(
244
expected_params = _mod_branch.ChangeBranchTipParams(
246
245
branch, 2, 1, 'two-\xc2\xb5', 'one-\xc2\xb5')
247
246
self.assertHookCalls(expected_params, branch, hook_calls, pre=True)
250
249
branch = self.make_branch('source')
251
250
def assertBranchIsLocked(params):
252
251
self.assertTrue(params.branch.is_locked())
253
Branch.hooks.install_named_hook(
252
_mod_branch.Branch.hooks.install_named_hook(
254
253
'pre_change_branch_tip', assertBranchIsLocked, None)
255
branch.set_last_revision_info(0, NULL_REVISION)
254
branch.set_last_revision_info(0, revision.NULL_REVISION)
257
256
def test_calls_all_hooks_no_errors(self):
258
257
"""If multiple hooks are registered, all are called (if none raise
262
261
hook_calls_1 = self.install_logging_hook('pre')
263
262
hook_calls_2 = self.install_logging_hook('pre')
264
263
self.assertIsNot(hook_calls_1, hook_calls_2)
265
branch.set_last_revision_info(0, NULL_REVISION)
264
branch.set_last_revision_info(0, revision.NULL_REVISION)
266
265
# Both hooks are called.
267
if isinstance(branch, RemoteBranch):
266
if isinstance(branch, remote.RemoteBranch):
279
278
branch = self.make_branch_with_revision_ids(
280
279
'one-\xc2\xb5', 'two-\xc2\xb5')
281
280
def hook_that_rejects(params):
282
raise TipChangeRejected('rejection message')
283
Branch.hooks.install_named_hook(
281
raise errors.TipChangeRejected('rejection message')
282
_mod_branch.Branch.hooks.install_named_hook(
284
283
'pre_change_branch_tip', hook_that_rejects, None)
285
284
self.assertRaises(
286
TipChangeRejected, branch.set_last_revision_info, 0, NULL_REVISION)
285
errors.TipChangeRejected,
286
branch.set_last_revision_info, 0, revision.NULL_REVISION)
287
287
# The revision info is unchanged.
288
288
self.assertEqual((2, 'two-\xc2\xb5'), branch.last_revision_info())
301
301
branch = self.make_branch_with_revision_ids('revid-one')
302
302
def assertBranchAtRevision1(params):
303
303
self.assertEquals(
304
(0, NULL_REVISION), params.branch.last_revision_info())
305
Branch.hooks.install_named_hook(
304
(0, revision.NULL_REVISION), params.branch.last_revision_info())
305
_mod_branch.Branch.hooks.install_named_hook(
306
306
'post_change_branch_tip', assertBranchAtRevision1, None)
307
branch.set_last_revision_info(0, NULL_REVISION)
307
branch.set_last_revision_info(0, revision.NULL_REVISION)
309
309
def test_empty_history(self):
310
310
branch = self.make_branch('source')
311
311
hook_calls = self.install_logging_hook('post')
312
branch.set_last_revision_info(0, NULL_REVISION)
313
expected_params = ChangeBranchTipParams(
314
branch, 0, 0, NULL_REVISION, NULL_REVISION)
312
branch.set_last_revision_info(0, revision.NULL_REVISION)
313
expected_params = _mod_branch.ChangeBranchTipParams(
314
branch, 0, 0, revision.NULL_REVISION, revision.NULL_REVISION)
315
315
self.assertHookCalls(expected_params, branch, hook_calls)
317
317
def test_nonempty_history(self):
322
322
'one-\xc2\xb5', 'two-\xc2\xb5')
323
323
hook_calls = self.install_logging_hook('post')
324
324
branch.set_last_revision_info(1, 'one-\xc2\xb5')
325
expected_params = ChangeBranchTipParams(
325
expected_params = _mod_branch.ChangeBranchTipParams(
326
326
branch, 2, 1, 'two-\xc2\xb5', 'one-\xc2\xb5')
327
327
self.assertHookCalls(expected_params, branch, hook_calls)
331
331
branch = self.make_branch('source')
332
332
def assertBranchIsLocked(params):
333
333
self.assertTrue(params.branch.is_locked())
334
Branch.hooks.install_named_hook(
334
_mod_branch.Branch.hooks.install_named_hook(
335
335
'post_change_branch_tip', assertBranchIsLocked, None)
336
branch.set_last_revision_info(0, NULL_REVISION)
336
branch.set_last_revision_info(0, revision.NULL_REVISION)
338
338
def test_calls_all_hooks_no_errors(self):
339
339
"""If multiple hooks are registered, all are called (if none raise
343
343
hook_calls_1 = self.install_logging_hook('post')
344
344
hook_calls_2 = self.install_logging_hook('post')
345
345
self.assertIsNot(hook_calls_1, hook_calls_2)
346
branch.set_last_revision_info(0, NULL_REVISION)
346
branch.set_last_revision_info(0, revision.NULL_REVISION)
347
347
# Both hooks are called.
348
if isinstance(branch, RemoteBranch):
348
if isinstance(branch, remote.RemoteBranch):
392
392
def test_set_last_revision_info(self):
393
393
branch = self.make_branch('')
394
branch.set_last_revision_info(0, NULL_REVISION)
394
branch.set_last_revision_info(0, revision.NULL_REVISION)
395
395
self.assertPreAndPostHooksWereInvoked(branch, True)
397
397
def test_generate_revision_history(self):
398
398
branch = self.make_branch('')
399
branch.generate_revision_history(NULL_REVISION)
399
branch.generate_revision_history(revision.NULL_REVISION)
400
400
# NB: for HPSS protocols < v3, the server does not invoke branch tip
401
401
# change events on generate_revision_history, as the change is done
402
402
# directly by the client over the VFS.