~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/tests/test_commit.py

  • Committer: Martin Pool
  • Date: 2005-07-22 22:37:53 UTC
  • Revision ID: mbp@sourcefrog.net-20050722223753-7dced4e32d3ce21d
- add the start of a test for inventory file-id matching

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005 by Canonical Ltd
2
 
 
3
 
# This program is free software; you can redistribute it and/or modify
4
 
# it under the terms of the GNU General Public License as published by
5
 
# the Free Software Foundation; either version 2 of the License, or
6
 
# (at your option) any later version.
7
 
 
8
 
# This program is distributed in the hope that it will be useful,
9
 
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
 
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
 
# GNU General Public License for more details.
12
 
 
13
 
# You should have received a copy of the GNU General Public License
14
 
# along with this program; if not, write to the Free Software
15
 
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
 
 
17
 
 
18
 
import os
19
 
 
20
 
import bzrlib
21
 
from bzrlib.tests import TestCaseWithTransport
22
 
from bzrlib.branch import Branch
23
 
from bzrlib.workingtree import WorkingTree
24
 
from bzrlib.commit import Commit
25
 
from bzrlib.config import BranchConfig
26
 
from bzrlib.errors import PointlessCommit, BzrError, SigningFailed
27
 
 
28
 
 
29
 
# TODO: Test commit with some added, and added-but-missing files
30
 
 
31
 
class MustSignConfig(BranchConfig):
32
 
 
33
 
    def signature_needed(self):
34
 
        return True
35
 
 
36
 
    def gpg_signing_command(self):
37
 
        return ['cat', '-']
38
 
 
39
 
 
40
 
class BranchWithHooks(BranchConfig):
41
 
 
42
 
    def post_commit(self):
43
 
        return "bzrlib.ahook bzrlib.ahook"
44
 
 
45
 
 
46
 
class TestCommit(TestCaseWithTransport):
47
 
 
48
 
    def test_simple_commit(self):
49
 
        """Commit and check two versions of a single file."""
50
 
        wt = self.make_branch_and_tree('.')
51
 
        b = wt.branch
52
 
        file('hello', 'w').write('hello world')
53
 
        wt.add('hello')
54
 
        wt.commit(message='add hello')
55
 
        file_id = wt.path2id('hello')
56
 
 
57
 
        file('hello', 'w').write('version 2')
58
 
        wt.commit(message='commit 2')
59
 
 
60
 
        eq = self.assertEquals
61
 
        eq(b.revno(), 2)
62
 
        rh = b.revision_history()
63
 
        rev = b.repository.get_revision(rh[0])
64
 
        eq(rev.message, 'add hello')
65
 
 
66
 
        tree1 = b.repository.revision_tree(rh[0])
67
 
        text = tree1.get_file_text(file_id)
68
 
        eq(text, 'hello world')
69
 
 
70
 
        tree2 = b.repository.revision_tree(rh[1])
71
 
        eq(tree2.get_file_text(file_id), 'version 2')
72
 
 
73
 
    def test_delete_commit(self):
74
 
        """Test a commit with a deleted file"""
75
 
        wt = self.make_branch_and_tree('.')
76
 
        b = wt.branch
77
 
        file('hello', 'w').write('hello world')
78
 
        wt.add(['hello'], ['hello-id'])
79
 
        wt.commit(message='add hello')
80
 
 
81
 
        os.remove('hello')
82
 
        wt.commit('removed hello', rev_id='rev2')
83
 
 
84
 
        tree = b.repository.revision_tree('rev2')
85
 
        self.assertFalse(tree.has_id('hello-id'))
86
 
 
87
 
    def test_pointless_commit(self):
88
 
        """Commit refuses unless there are changes or it's forced."""
89
 
        wt = self.make_branch_and_tree('.')
90
 
        b = wt.branch
91
 
        file('hello', 'w').write('hello')
92
 
        wt.add(['hello'])
93
 
        wt.commit(message='add hello')
94
 
        self.assertEquals(b.revno(), 1)
95
 
        self.assertRaises(PointlessCommit,
96
 
                          wt.commit,
97
 
                          message='fails',
98
 
                          allow_pointless=False)
99
 
        self.assertEquals(b.revno(), 1)
100
 
        
101
 
    def test_commit_empty(self):
102
 
        """Commiting an empty tree works."""
103
 
        wt = self.make_branch_and_tree('.')
104
 
        b = wt.branch
105
 
        wt.commit(message='empty tree', allow_pointless=True)
106
 
        self.assertRaises(PointlessCommit,
107
 
                          wt.commit,
108
 
                          message='empty tree',
109
 
                          allow_pointless=False)
110
 
        wt.commit(message='empty tree', allow_pointless=True)
111
 
        self.assertEquals(b.revno(), 2)
112
 
 
113
 
    def test_selective_delete(self):
114
 
        """Selective commit in tree with deletions"""
115
 
        wt = self.make_branch_and_tree('.')
116
 
        b = wt.branch
117
 
        file('hello', 'w').write('hello')
118
 
        file('buongia', 'w').write('buongia')
119
 
        wt.add(['hello', 'buongia'],
120
 
              ['hello-id', 'buongia-id'])
121
 
        wt.commit(message='add files',
122
 
                 rev_id='test@rev-1')
123
 
        
124
 
        os.remove('hello')
125
 
        file('buongia', 'w').write('new text')
126
 
        wt.commit(message='update text',
127
 
                 specific_files=['buongia'],
128
 
                 allow_pointless=False,
129
 
                 rev_id='test@rev-2')
130
 
 
131
 
        wt.commit(message='remove hello',
132
 
                 specific_files=['hello'],
133
 
                 allow_pointless=False,
134
 
                 rev_id='test@rev-3')
135
 
 
136
 
        eq = self.assertEquals
137
 
        eq(b.revno(), 3)
138
 
 
139
 
        tree2 = b.repository.revision_tree('test@rev-2')
140
 
        self.assertTrue(tree2.has_filename('hello'))
141
 
        self.assertEquals(tree2.get_file_text('hello-id'), 'hello')
142
 
        self.assertEquals(tree2.get_file_text('buongia-id'), 'new text')
143
 
        
144
 
        tree3 = b.repository.revision_tree('test@rev-3')
145
 
        self.assertFalse(tree3.has_filename('hello'))
146
 
        self.assertEquals(tree3.get_file_text('buongia-id'), 'new text')
147
 
 
148
 
    def test_commit_rename(self):
149
 
        """Test commit of a revision where a file is renamed."""
150
 
        tree = self.make_branch_and_tree('.')
151
 
        b = tree.branch
152
 
        self.build_tree(['hello'], line_endings='binary')
153
 
        tree.add(['hello'], ['hello-id'])
154
 
        tree.commit(message='one', rev_id='test@rev-1', allow_pointless=False)
155
 
 
156
 
        tree.rename_one('hello', 'fruity')
157
 
        tree.commit(message='renamed', rev_id='test@rev-2', allow_pointless=False)
158
 
 
159
 
        eq = self.assertEquals
160
 
        tree1 = b.repository.revision_tree('test@rev-1')
161
 
        eq(tree1.id2path('hello-id'), 'hello')
162
 
        eq(tree1.get_file_text('hello-id'), 'contents of hello\n')
163
 
        self.assertFalse(tree1.has_filename('fruity'))
164
 
        self.check_inventory_shape(tree1.inventory, ['hello'])
165
 
        ie = tree1.inventory['hello-id']
166
 
        eq(ie.revision, 'test@rev-1')
167
 
 
168
 
        tree2 = b.repository.revision_tree('test@rev-2')
169
 
        eq(tree2.id2path('hello-id'), 'fruity')
170
 
        eq(tree2.get_file_text('hello-id'), 'contents of hello\n')
171
 
        self.check_inventory_shape(tree2.inventory, ['fruity'])
172
 
        ie = tree2.inventory['hello-id']
173
 
        eq(ie.revision, 'test@rev-2')
174
 
 
175
 
    def test_reused_rev_id(self):
176
 
        """Test that a revision id cannot be reused in a branch"""
177
 
        wt = self.make_branch_and_tree('.')
178
 
        b = wt.branch
179
 
        wt.commit('initial', rev_id='test@rev-1', allow_pointless=True)
180
 
        self.assertRaises(Exception,
181
 
                          b.working_tree().commit,
182
 
                          message='reused id',
183
 
                          rev_id='test@rev-1',
184
 
                          allow_pointless=True)
185
 
 
186
 
    def test_commit_move(self):
187
 
        """Test commit of revisions with moved files and directories"""
188
 
        eq = self.assertEquals
189
 
        wt = self.make_branch_and_tree('.')
190
 
        b = wt.branch
191
 
        r1 = 'test@rev-1'
192
 
        self.build_tree(['hello', 'a/', 'b/'])
193
 
        wt.add(['hello', 'a', 'b'], ['hello-id', 'a-id', 'b-id'])
194
 
        wt.commit('initial', rev_id=r1, allow_pointless=False)
195
 
        wt.move(['hello'], 'a')
196
 
        r2 = 'test@rev-2'
197
 
        wt.commit('two', rev_id=r2, allow_pointless=False)
198
 
        self.check_inventory_shape(b.working_tree().read_working_inventory(),
199
 
                                   ['a', 'a/hello', 'b'])
200
 
 
201
 
        wt.move(['b'], 'a')
202
 
        r3 = 'test@rev-3'
203
 
        wt.commit('three', rev_id=r3, allow_pointless=False)
204
 
        self.check_inventory_shape(wt.read_working_inventory(),
205
 
                                   ['a', 'a/hello', 'a/b'])
206
 
        self.check_inventory_shape(b.repository.get_revision_inventory(r3),
207
 
                                   ['a', 'a/hello', 'a/b'])
208
 
 
209
 
        wt.move(['a/hello'], 'a/b')
210
 
        r4 = 'test@rev-4'
211
 
        wt.commit('four', rev_id=r4, allow_pointless=False)
212
 
        self.check_inventory_shape(wt.read_working_inventory(),
213
 
                                   ['a', 'a/b/hello', 'a/b'])
214
 
 
215
 
        inv = b.repository.get_revision_inventory(r4)
216
 
        eq(inv['hello-id'].revision, r4)
217
 
        eq(inv['a-id'].revision, r1)
218
 
        eq(inv['b-id'].revision, r3)
219
 
        
220
 
    def test_removed_commit(self):
221
 
        """Commit with a removed file"""
222
 
        wt = self.make_branch_and_tree('.')
223
 
        b = wt.branch
224
 
        file('hello', 'w').write('hello world')
225
 
        wt.add(['hello'], ['hello-id'])
226
 
        wt.commit(message='add hello')
227
 
        wt.remove('hello')
228
 
        wt.commit('removed hello', rev_id='rev2')
229
 
 
230
 
        tree = b.repository.revision_tree('rev2')
231
 
        self.assertFalse(tree.has_id('hello-id'))
232
 
 
233
 
    def test_committed_ancestry(self):
234
 
        """Test commit appends revisions to ancestry."""
235
 
        wt = self.make_branch_and_tree('.')
236
 
        b = wt.branch
237
 
        rev_ids = []
238
 
        for i in range(4):
239
 
            file('hello', 'w').write((str(i) * 4) + '\n')
240
 
            if i == 0:
241
 
                wt.add(['hello'], ['hello-id'])
242
 
            rev_id = 'test@rev-%d' % (i+1)
243
 
            rev_ids.append(rev_id)
244
 
            wt.commit(message='rev %d' % (i+1),
245
 
                     rev_id=rev_id)
246
 
        eq = self.assertEquals
247
 
        eq(b.revision_history(), rev_ids)
248
 
        for i in range(4):
249
 
            anc = b.repository.get_ancestry(rev_ids[i])
250
 
            eq(anc, [None] + rev_ids[:i+1])
251
 
 
252
 
    def test_commit_new_subdir_child_selective(self):
253
 
        wt = self.make_branch_and_tree('.')
254
 
        b = wt.branch
255
 
        self.build_tree(['dir/', 'dir/file1', 'dir/file2'])
256
 
        wt.add(['dir', 'dir/file1', 'dir/file2'],
257
 
              ['dirid', 'file1id', 'file2id'])
258
 
        wt.commit('dir/file1', specific_files=['dir/file1'], rev_id='1')
259
 
        inv = b.repository.get_inventory('1')
260
 
        self.assertEqual('1', inv['dirid'].revision)
261
 
        self.assertEqual('1', inv['file1id'].revision)
262
 
        # FIXME: This should raise a KeyError I think, rbc20051006
263
 
        self.assertRaises(BzrError, inv.__getitem__, 'file2id')
264
 
 
265
 
    def test_strict_commit(self):
266
 
        """Try and commit with unknown files and strict = True, should fail."""
267
 
        from bzrlib.errors import StrictCommitFailed
268
 
        wt = self.make_branch_and_tree('.')
269
 
        b = wt.branch
270
 
        file('hello', 'w').write('hello world')
271
 
        wt.add('hello')
272
 
        file('goodbye', 'w').write('goodbye cruel world!')
273
 
        self.assertRaises(StrictCommitFailed, b.working_tree().commit,
274
 
            message='add hello but not goodbye', strict=True)
275
 
 
276
 
    def test_strict_commit_without_unknowns(self):
277
 
        """Try and commit with no unknown files and strict = True,
278
 
        should work."""
279
 
        from bzrlib.errors import StrictCommitFailed
280
 
        wt = self.make_branch_and_tree('.')
281
 
        b = wt.branch
282
 
        file('hello', 'w').write('hello world')
283
 
        wt.add('hello')
284
 
        wt.commit(message='add hello', strict=True)
285
 
 
286
 
    def test_nonstrict_commit(self):
287
 
        """Try and commit with unknown files and strict = False, should work."""
288
 
        wt = self.make_branch_and_tree('.')
289
 
        b = wt.branch
290
 
        file('hello', 'w').write('hello world')
291
 
        wt.add('hello')
292
 
        file('goodbye', 'w').write('goodbye cruel world!')
293
 
        wt.commit(message='add hello but not goodbye', strict=False)
294
 
 
295
 
    def test_nonstrict_commit_without_unknowns(self):
296
 
        """Try and commit with no unknown files and strict = False,
297
 
        should work."""
298
 
        wt = self.make_branch_and_tree('.')
299
 
        b = wt.branch
300
 
        file('hello', 'w').write('hello world')
301
 
        wt.add('hello')
302
 
        wt.commit(message='add hello', strict=False)
303
 
 
304
 
    def test_signed_commit(self):
305
 
        import bzrlib.gpg
306
 
        import bzrlib.commit as commit
307
 
        oldstrategy = bzrlib.gpg.GPGStrategy
308
 
        wt = self.make_branch_and_tree('.')
309
 
        branch = wt.branch
310
 
        wt.commit("base", allow_pointless=True, rev_id='A')
311
 
        self.failIf(branch.repository.revision_store.has_id('A', 'sig'))
312
 
        try:
313
 
            from bzrlib.testament import Testament
314
 
            # monkey patch gpg signing mechanism
315
 
            bzrlib.gpg.GPGStrategy = bzrlib.gpg.LoopbackGPGStrategy
316
 
            commit.Commit(config=MustSignConfig(branch)).commit(message="base",
317
 
                                                      allow_pointless=True,
318
 
                                                      rev_id='B',
319
 
                                                      working_tree=wt)
320
 
            self.assertEqual(Testament.from_revision(branch.repository,
321
 
                             'B').as_short_text(),
322
 
                             branch.repository.revision_store.get('B', 
323
 
                                                               'sig').read())
324
 
        finally:
325
 
            bzrlib.gpg.GPGStrategy = oldstrategy
326
 
 
327
 
    def test_commit_failed_signature(self):
328
 
        import bzrlib.gpg
329
 
        import bzrlib.commit as commit
330
 
        oldstrategy = bzrlib.gpg.GPGStrategy
331
 
        wt = self.make_branch_and_tree('.')
332
 
        branch = wt.branch
333
 
        wt.commit("base", allow_pointless=True, rev_id='A')
334
 
        self.failIf(branch.repository.revision_store.has_id('A', 'sig'))
335
 
        try:
336
 
            from bzrlib.testament import Testament
337
 
            # monkey patch gpg signing mechanism
338
 
            bzrlib.gpg.GPGStrategy = bzrlib.gpg.DisabledGPGStrategy
339
 
            config = MustSignConfig(branch)
340
 
            self.assertRaises(SigningFailed,
341
 
                              commit.Commit(config=config).commit,
342
 
                              branch, "base",
343
 
                              allow_pointless=True,
344
 
                              rev_id='B')
345
 
            branch = Branch.open(self.get_url('.'))
346
 
            self.assertEqual(branch.revision_history(), ['A'])
347
 
            self.failIf(branch.repository.revision_store.has_id('B'))
348
 
        finally:
349
 
            bzrlib.gpg.GPGStrategy = oldstrategy
350
 
 
351
 
    def test_commit_invokes_hooks(self):
352
 
        import bzrlib.commit as commit
353
 
        wt = self.make_branch_and_tree('.')
354
 
        branch = wt.branch
355
 
        calls = []
356
 
        def called(branch, rev_id):
357
 
            calls.append('called')
358
 
        bzrlib.ahook = called
359
 
        try:
360
 
            config = BranchWithHooks(branch)
361
 
            commit.Commit(config=config).commit(
362
 
                            message = "base",
363
 
                            allow_pointless=True,
364
 
                            rev_id='A', working_tree = wt)
365
 
            self.assertEqual(['called', 'called'], calls)
366
 
        finally:
367
 
            del bzrlib.ahook