1
# Copyright (C) 2005-2011 Canonical Ltd
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.
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.
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
"""Tests for version_info"""
19
from cStringIO import StringIO
30
from bzrlib.tests import TestCaseWithTransport
31
from bzrlib.rio import read_stanzas
33
from bzrlib.version_info_formats.format_custom import CustomVersionInfoBuilder
34
from bzrlib.version_info_formats.format_rio import RioVersionInfoBuilder
35
from bzrlib.version_info_formats.format_python import PythonVersionInfoBuilder
38
class VersionInfoTestCase(TestCaseWithTransport):
40
def create_branch(self):
41
wt = self.make_branch_and_tree('branch')
43
self.build_tree(['branch/a'])
45
wt.commit('a', rev_id='r1')
47
self.build_tree(['branch/b'])
49
wt.commit('b', rev_id='r2')
51
self.build_tree_contents([('branch/a', 'new contents\n')])
52
wt.commit(u'\xe52', rev_id='r3')
56
def create_tree_with_dotted_revno(self):
57
wt = self.make_branch_and_tree('branch')
58
self.build_tree(['branch/a'])
60
wt.commit('a', rev_id='r1')
62
other = wt.bzrdir.sprout('other').open_workingtree()
63
self.build_tree(['other/b.a'])
65
other.commit('b.a', rev_id='o2')
68
self.run_bzr('merge ../other')
69
wt.commit('merge', rev_id='merge')
71
wt.update(revision='o2')
76
class TestVersionInfoRio(VersionInfoTestCase):
78
def test_rio_null(self):
79
wt = self.make_branch_and_tree('branch')
82
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt)
85
self.assertContainsRe(val, 'build-date:')
86
self.assertContainsRe(val, 'revno: 0')
88
def test_rio_dotted_revno(self):
89
wt = self.create_tree_with_dotted_revno()
92
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt)
95
self.assertContainsRe(val, 'revno: 1.1.1')
97
def regen_text(self, wt, **kwargs):
99
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt, **kwargs)
100
builder.generate(sio)
104
def test_simple(self):
105
wt = self.create_branch()
107
val = self.regen_text(wt)
108
self.assertContainsRe(val, 'build-date:')
109
self.assertContainsRe(val, 'date:')
110
self.assertContainsRe(val, 'revno: 3')
111
self.assertContainsRe(val, 'revision-id: r3')
113
def test_clean(self):
114
wt = self.create_branch()
115
val = self.regen_text(wt, check_for_clean=True)
116
self.assertContainsRe(val, 'clean: True')
118
def test_no_clean(self):
119
wt = self.create_branch()
120
self.build_tree(['branch/c'])
121
val = self.regen_text(wt, check_for_clean=True)
122
self.assertContainsRe(val, 'clean: False')
124
def test_history(self):
125
wt = self.create_branch()
127
val = self.regen_text(wt, include_revision_history=True)
128
self.assertContainsRe(val, 'id: r1')
129
self.assertContainsRe(val, 'message: a')
130
self.assertContainsRe(val, 'id: r2')
131
self.assertContainsRe(val, 'message: b')
132
self.assertContainsRe(val, 'id: r3')
133
self.assertContainsRe(val, 'message: \xc3\xa52') # utf8 encoding '\xe5'
135
def regen(self, wt, **kwargs):
137
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt, **kwargs)
138
builder.generate(sio)
140
stanzas = list(read_stanzas(sio))
141
self.assertEqual(1, len(stanzas))
144
def test_rio_version_hook(self):
145
def update_stanza(rev, stanza):
146
stanza.add('bla', 'bloe')
147
RioVersionInfoBuilder.hooks.install_named_hook(
148
'revision', update_stanza, None)
149
wt = self.create_branch()
151
stanza = self.regen(wt)
152
self.assertEqual(['bloe'], stanza.get_all('bla'))
154
def get_one_stanza(self, stanza, key):
155
new_stanzas = list(read_stanzas(StringIO(stanza[key].encode('utf8'))))
156
self.assertEqual(1, len(new_stanzas))
157
return new_stanzas[0]
159
def test_build_date(self):
160
wt = self.create_branch()
161
stanza = self.regen(wt)
162
self.assertTrue('date' in stanza)
163
self.assertTrue('build-date' in stanza)
164
self.assertEqual(['3'], stanza.get_all('revno'))
165
self.assertEqual(['r3'], stanza.get_all('revision-id'))
167
def test_not_clean(self):
168
wt = self.create_branch()
169
self.build_tree(['branch/c'])
170
stanza = self.regen(wt, check_for_clean=True, include_file_revisions=True)
171
self.assertEqual(['False'], stanza.get_all('clean'))
173
def test_file_revisions(self):
174
wt = self.create_branch()
175
self.build_tree(['branch/c'])
176
stanza = self.regen(wt, check_for_clean=True, include_file_revisions=True)
177
# This assumes it's being run against a tree that does not update the
178
# root revision on every commit.
179
file_rev_stanza = self.get_one_stanza(stanza, 'file-revisions')
180
self.assertEqual(['', 'a', 'b', 'c'], file_rev_stanza.get_all('path'))
181
self.assertEqual(['r1', 'r3', 'r2', 'unversioned'],
182
file_rev_stanza.get_all('revision'))
184
def test_revision_history(self):
185
wt = self.create_branch()
186
stanza = self.regen(wt, include_revision_history=True)
187
revision_stanza = self.get_one_stanza(stanza, 'revisions')
188
self.assertEqual(['r1', 'r2', 'r3'], revision_stanza.get_all('id'))
189
self.assertEqual(['a', 'b', u'\xe52'], revision_stanza.get_all('message'))
190
self.assertEqual(3, len(revision_stanza.get_all('date')))
192
def test_file_revisions_with_rename(self):
193
# a was modified, so it should show up modified again
194
wt = self.create_branch()
195
self.build_tree(['branch/a', 'branch/c'])
197
wt.rename_one('b', 'd')
198
stanza = self.regen(wt, check_for_clean=True, include_file_revisions=True)
199
file_rev_stanza = self.get_one_stanza(stanza, 'file-revisions')
200
self.assertEqual(['', 'a', 'b', 'c', 'd'],
201
file_rev_stanza.get_all('path'))
202
self.assertEqual(['r1', 'modified', 'renamed to d', 'new',
204
file_rev_stanza.get_all('revision'))
206
def test_file_revisions_with_removal(self):
207
wt = self.create_branch()
208
self.build_tree(['branch/a', 'branch/c'])
210
wt.rename_one('b', 'd')
212
wt.commit('modified', rev_id='r4')
214
wt.remove(['c', 'd'])
215
os.remove('branch/d')
216
stanza = self.regen(wt, check_for_clean=True, include_file_revisions=True)
217
file_rev_stanza = self.get_one_stanza(stanza, 'file-revisions')
218
self.assertEqual(['', 'a', 'c', 'd'], file_rev_stanza.get_all('path'))
219
self.assertEqual(['r1', 'r4', 'unversioned', 'removed'],
220
file_rev_stanza.get_all('revision'))
222
def test_revision(self):
223
wt = self.create_branch()
224
self.build_tree(['branch/a', 'branch/c'])
226
wt.rename_one('b', 'd')
228
stanza = self.regen(wt, check_for_clean=True,
229
include_file_revisions=True, revision_id=wt.last_revision())
230
file_rev_stanza = self.get_one_stanza(stanza, 'file-revisions')
231
self.assertEqual(['', 'a', 'b'], file_rev_stanza.get_all('path'))
232
self.assertEqual(['r1', 'r3', 'r2'],
233
file_rev_stanza.get_all('revision'))
236
class PythonVersionInfoTests(VersionInfoTestCase):
238
def test_python_null(self):
239
wt = self.make_branch_and_tree('branch')
242
builder = PythonVersionInfoBuilder(wt.branch, working_tree=wt)
243
builder.generate(sio)
245
self.assertContainsRe(val, "'revision_id': None")
246
self.assertContainsRe(val, "'revno': '0'")
247
self.assertNotContainsString(val, '\n\n\n\n')
249
def test_python_dotted_revno(self):
250
wt = self.create_tree_with_dotted_revno()
253
builder = PythonVersionInfoBuilder(wt.branch, working_tree=wt)
254
builder.generate(sio)
256
self.assertContainsRe(val, "'revno': '1.1.1'")
258
def regen(self, wt, **kwargs):
259
"""Create a test module, import and return it"""
260
outf = open('test_version_information.py', 'wb')
262
builder = PythonVersionInfoBuilder(wt.branch, working_tree=wt,
264
builder.generate(outf)
267
module_info = imp.find_module('test_version_information',
269
tvi = imp.load_module('tvi', *module_info)
270
# Make sure the module isn't cached
271
sys.modules.pop('tvi', None)
272
sys.modules.pop('test_version_information', None)
273
# Delete the compiled versions, because we are generating
274
# a new file fast enough that python doesn't detect it
275
# needs to recompile, and using sleep() just makes the
277
if os.path.exists('test_version_information.pyc'):
278
os.remove('test_version_information.pyc')
279
if os.path.exists('test_version_information.pyo'):
280
os.remove('test_version_information.pyo')
283
def test_python_version(self):
284
wt = self.create_branch()
287
self.assertEqual('3', tvi.version_info['revno'])
288
self.assertEqual('r3', tvi.version_info['revision_id'])
289
self.assertTrue(tvi.version_info.has_key('date'))
290
self.assertEqual(None, tvi.version_info['clean'])
292
tvi = self.regen(wt, check_for_clean=True)
293
self.assertEqual(True, tvi.version_info['clean'])
295
self.build_tree(['branch/c'])
296
tvi = self.regen(wt, check_for_clean=True, include_file_revisions=True)
297
self.assertEqual(False, tvi.version_info['clean'])
298
self.assertEqual(['', 'a', 'b', 'c'],
299
sorted(tvi.file_revisions.keys()))
300
self.assertEqual('r3', tvi.file_revisions['a'])
301
self.assertEqual('r2', tvi.file_revisions['b'])
302
self.assertEqual('unversioned', tvi.file_revisions['c'])
303
os.remove('branch/c')
305
tvi = self.regen(wt, include_revision_history=True)
307
rev_info = [(rev, message) for rev, message, timestamp, timezone
309
self.assertEqual([('r1', 'a'), ('r2', 'b'), ('r3', u'\xe52')], rev_info)
311
# a was modified, so it should show up modified again
312
self.build_tree(['branch/a', 'branch/c'])
314
wt.rename_one('b', 'd')
315
tvi = self.regen(wt, check_for_clean=True, include_file_revisions=True)
316
self.assertEqual(['', 'a', 'b', 'c', 'd'],
317
sorted(tvi.file_revisions.keys()))
318
self.assertEqual('modified', tvi.file_revisions['a'])
319
self.assertEqual('renamed to d', tvi.file_revisions['b'])
320
self.assertEqual('new', tvi.file_revisions['c'])
321
self.assertEqual('renamed from b', tvi.file_revisions['d'])
323
wt.commit('modified', rev_id='r4')
324
wt.remove(['c', 'd'])
325
os.remove('branch/d')
326
tvi = self.regen(wt, check_for_clean=True, include_file_revisions=True)
327
self.assertEqual(['', 'a', 'c', 'd'],
328
sorted(tvi.file_revisions.keys()))
329
self.assertEqual('r4', tvi.file_revisions['a'])
330
self.assertEqual('unversioned', tvi.file_revisions['c'])
331
self.assertEqual('removed', tvi.file_revisions['d'])
334
class CustomVersionInfoTests(VersionInfoTestCase):
336
def test_custom_null(self):
338
wt = self.make_branch_and_tree('branch')
339
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
340
template='revno: {revno}')
341
builder.generate(sio)
342
self.assertEquals("revno: 0", sio.getvalue())
344
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
345
template='{revno} revid: {revision_id}')
346
# revision_id is not available yet
347
self.assertRaises(errors.MissingTemplateVariable,
348
builder.generate, sio)
350
def test_custom_dotted_revno(self):
352
wt = self.create_tree_with_dotted_revno()
353
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
354
template='{revno} revid: {revision_id}')
355
builder.generate(sio)
356
self.assertEquals("1.1.1 revid: o2", sio.getvalue())
358
def regen(self, wt, tpl, **kwargs):
360
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
361
template=tpl, **kwargs)
362
builder.generate(sio)
366
def test_build_date(self):
367
wt = self.create_branch()
369
val = self.regen(wt, 'build-date: "{build_date}"\ndate: "{date}"')
370
self.assertContainsRe(val, 'build-date: "[0-9-+: ]+"')
371
self.assertContainsRe(val, 'date: "[0-9-+: ]+"')
373
def test_revno(self):
374
wt = self.create_branch()
375
val = self.regen(wt, 'revno: {revno}')
376
self.assertEqual(val, 'revno: 3')
378
def test_revision_id(self):
379
wt = self.create_branch()
380
val = self.regen(wt, 'revision-id: {revision_id}')
381
self.assertEqual(val, 'revision-id: r3')
383
def test_clean(self):
384
wt = self.create_branch()
385
val = self.regen(wt, 'clean: {clean}', check_for_clean=True)
386
self.assertEqual(val, 'clean: 1')
388
def test_not_clean(self):
389
wt = self.create_branch()
391
self.build_tree(['branch/c'])
392
val = self.regen(wt, 'clean: {clean}', check_for_clean=True)
393
self.assertEqual(val, 'clean: 0')
394
os.remove('branch/c')
396
def test_custom_without_template(self):
397
builder = CustomVersionInfoBuilder(None)
399
self.assertRaises(errors.NoTemplate, builder.generate, sio)
402
class TestBuilder(version_info_formats.VersionInfoBuilder):
406
class TestVersionInfoFormatRegistry(tests.TestCase):
409
super(TestVersionInfoFormatRegistry, self).setUp()
410
self.overrideAttr(version_info_formats,
411
'format_registry', registry.Registry())
413
def test_register_remove(self):
414
registry = version_info_formats.format_registry
415
registry.register('testbuilder',
416
TestBuilder, 'a simple test builder')
417
self.assertIs(TestBuilder, registry.get('testbuilder'))
418
self.assertEqual('a simple test builder',
419
registry.get_help('testbuilder'))
420
registry.remove('testbuilder')
421
self.assertRaises(KeyError, registry.get, 'testbuilder')