1
# Copyright (C) 2005, 2006 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 TestVersionInfo(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 test_rio_null(self):
57
wt = self.make_branch_and_tree('branch')
60
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt)
63
self.assertContainsRe(val, 'build-date:')
64
self.assertContainsRe(val, 'revno: 0')
66
def test_rio_version_text(self):
67
wt = self.create_branch()
71
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt,
78
self.assertContainsRe(val, 'build-date:')
79
self.assertContainsRe(val, 'date:')
80
self.assertContainsRe(val, 'revno: 3')
81
self.assertContainsRe(val, 'revision-id: r3')
83
val = regen(check_for_clean=True)
84
self.assertContainsRe(val, 'clean: True')
86
self.build_tree(['branch/c'])
87
val = regen(check_for_clean=True)
88
self.assertContainsRe(val, 'clean: False')
91
val = regen(include_revision_history=True)
92
self.assertContainsRe(val, 'id: r1')
93
self.assertContainsRe(val, 'message: a')
94
self.assertContainsRe(val, 'id: r2')
95
self.assertContainsRe(val, 'message: b')
96
self.assertContainsRe(val, 'id: r3')
97
self.assertContainsRe(val, 'message: \xc3\xa52') # utf8 encoding '\xe5'
99
def test_rio_version_hook(self):
100
def update_stanza(rev, stanza):
101
stanza.add('bla', 'bloe')
102
RioVersionInfoBuilder.hooks.install_named_hook(
103
'revision', update_stanza, None)
104
wt = self.create_branch()
108
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt,
110
builder.generate(sio)
112
stanzas = list(read_stanzas(sio))
113
self.assertEqual(1, len(stanzas))
117
self.assertEqual(['bloe'], stanza.get_all('bla'))
119
def test_rio_version(self):
120
wt = self.create_branch()
124
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt,
126
builder.generate(sio)
128
stanzas = list(read_stanzas(sio))
129
self.assertEqual(1, len(stanzas))
132
def get_one_stanza(stanza, key):
133
new_stanzas = list(read_stanzas(
134
StringIO(stanza[key].encode('utf8'))))
135
self.assertEqual(1, len(new_stanzas))
136
return new_stanzas[0]
139
self.failUnless('date' in stanza)
140
self.failUnless('build-date' in stanza)
141
self.assertEqual(['3'], stanza.get_all('revno'))
142
self.assertEqual(['r3'], stanza.get_all('revision-id'))
144
stanza = regen(check_for_clean=True)
145
self.assertEqual(['True'], stanza.get_all('clean'))
147
self.build_tree(['branch/c'])
148
stanza = regen(check_for_clean=True, include_file_revisions=True)
149
self.assertEqual(['False'], stanza.get_all('clean'))
151
# This assumes it's being run against a tree that does not update the
152
# root revision on every commit.
153
file_rev_stanza = get_one_stanza(stanza, 'file-revisions')
154
self.assertEqual(['', 'a', 'b', 'c'], file_rev_stanza.get_all('path'))
155
self.assertEqual(['r1', 'r3', 'r2', 'unversioned'],
156
file_rev_stanza.get_all('revision'))
157
os.remove('branch/c')
159
stanza = regen(include_revision_history=True)
160
revision_stanza = get_one_stanza(stanza, 'revisions')
161
self.assertEqual(['r1', 'r2', 'r3'], revision_stanza.get_all('id'))
162
self.assertEqual(['a', 'b', u'\xe52'], revision_stanza.get_all('message'))
163
self.assertEqual(3, len(revision_stanza.get_all('date')))
165
# a was modified, so it should show up modified again
166
self.build_tree(['branch/a', 'branch/c'])
168
wt.rename_one('b', 'd')
169
stanza = regen(check_for_clean=True, include_file_revisions=True)
170
file_rev_stanza = get_one_stanza(stanza, 'file-revisions')
171
self.assertEqual(['', 'a', 'b', 'c', 'd'],
172
file_rev_stanza.get_all('path'))
173
self.assertEqual(['r1', 'modified', 'renamed to d', 'new',
175
file_rev_stanza.get_all('revision'))
177
wt.commit('modified', rev_id='r4')
178
wt.remove(['c', 'd'])
179
os.remove('branch/d')
180
stanza = regen(check_for_clean=True, include_file_revisions=True)
181
file_rev_stanza = get_one_stanza(stanza, 'file-revisions')
182
self.assertEqual(['', 'a', 'c', 'd'], file_rev_stanza.get_all('path'))
183
self.assertEqual(['r1', 'r4', 'unversioned', 'removed'],
184
file_rev_stanza.get_all('revision'))
186
def test_python_null(self):
187
wt = self.make_branch_and_tree('branch')
190
builder = PythonVersionInfoBuilder(wt.branch, working_tree=wt)
191
builder.generate(sio)
193
self.assertContainsRe(val, "'revision_id': None")
194
self.assertContainsRe(val, "'revno': 0")
196
def test_python_version(self):
197
wt = self.create_branch()
200
"""Create a test module, import and return it"""
201
outf = open('test_version_information.py', 'wb')
203
builder = PythonVersionInfoBuilder(wt.branch, working_tree=wt,
205
builder.generate(outf)
208
module_info = imp.find_module('test_version_information',
210
tvi = imp.load_module('tvi', *module_info)
211
# Make sure the module isn't cached
212
sys.modules.pop('tvi', None)
213
sys.modules.pop('test_version_information', None)
214
# Delete the compiled versions, because we are generating
215
# a new file fast enough that python doesn't detect it
216
# needs to recompile, and using sleep() just makes the
218
if os.path.exists('test_version_information.pyc'):
219
os.remove('test_version_information.pyc')
220
if os.path.exists('test_version_information.pyo'):
221
os.remove('test_version_information.pyo')
225
self.assertEqual(3, tvi.version_info['revno'])
226
self.assertEqual('r3', tvi.version_info['revision_id'])
227
self.failUnless(tvi.version_info.has_key('date'))
228
self.assertEqual(None, tvi.version_info['clean'])
230
tvi = regen(check_for_clean=True)
231
self.assertEqual(True, tvi.version_info['clean'])
233
self.build_tree(['branch/c'])
234
tvi = regen(check_for_clean=True, include_file_revisions=True)
235
self.assertEqual(False, tvi.version_info['clean'])
236
self.assertEqual(['', 'a', 'b', 'c'],
237
sorted(tvi.file_revisions.keys()))
238
self.assertEqual('r3', tvi.file_revisions['a'])
239
self.assertEqual('r2', tvi.file_revisions['b'])
240
self.assertEqual('unversioned', tvi.file_revisions['c'])
241
os.remove('branch/c')
243
tvi = regen(include_revision_history=True)
245
rev_info = [(rev, message) for rev, message, timestamp, timezone
247
self.assertEqual([('r1', 'a'), ('r2', 'b'), ('r3', u'\xe52')], rev_info)
249
# a was modified, so it should show up modified again
250
self.build_tree(['branch/a', 'branch/c'])
252
wt.rename_one('b', 'd')
253
tvi = regen(check_for_clean=True, include_file_revisions=True)
254
self.assertEqual(['', 'a', 'b', 'c', 'd'],
255
sorted(tvi.file_revisions.keys()))
256
self.assertEqual('modified', tvi.file_revisions['a'])
257
self.assertEqual('renamed to d', tvi.file_revisions['b'])
258
self.assertEqual('new', tvi.file_revisions['c'])
259
self.assertEqual('renamed from b', tvi.file_revisions['d'])
261
wt.commit('modified', rev_id='r4')
262
wt.remove(['c', 'd'])
263
os.remove('branch/d')
264
tvi = regen(check_for_clean=True, include_file_revisions=True)
265
self.assertEqual(['', 'a', 'c', 'd'],
266
sorted(tvi.file_revisions.keys()))
267
self.assertEqual('r4', tvi.file_revisions['a'])
268
self.assertEqual('unversioned', tvi.file_revisions['c'])
269
self.assertEqual('removed', tvi.file_revisions['d'])
271
def test_custom_null(self):
273
wt = self.make_branch_and_tree('branch')
274
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
275
template='revno: {revno}')
276
builder.generate(sio)
277
self.assertEquals("revno: 0", sio.getvalue())
279
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
280
template='{revno} revid: {revision_id}')
281
# revision_id is not available yet
282
self.assertRaises(errors.MissingTemplateVariable,
283
builder.generate, sio)
285
def test_custom_version_text(self):
286
wt = self.create_branch()
288
def regen(tpl, **kwargs):
290
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
291
template=tpl, **kwargs)
292
builder.generate(sio)
296
val = regen('build-date: "{build_date}"\ndate: "{date}"')
297
self.assertContainsRe(val, 'build-date: "[0-9-+: ]+"')
298
self.assertContainsRe(val, 'date: "[0-9-+: ]+"')
300
val = regen('revno: {revno}')
301
self.assertEqual(val, 'revno: 3')
303
val = regen('revision-id: {revision_id}')
304
self.assertEqual(val, 'revision-id: r3')
306
val = regen('clean: {clean}', check_for_clean=True)
307
self.assertEqual(val, 'clean: 1')
309
self.build_tree(['branch/c'])
310
val = regen('clean: {clean}', check_for_clean=True)
311
self.assertEqual(val, 'clean: 0')
312
os.remove('branch/c')
314
def test_custom_without_template(self):
315
builder = CustomVersionInfoBuilder(None)
317
self.assertRaises(errors.NoTemplate, builder.generate, sio)
320
class TestBuilder(version_info_formats.VersionInfoBuilder):
324
class TestVersionInfoFormatRegistry(tests.TestCase):
327
super(TestVersionInfoFormatRegistry, self).setUp()
328
registry = version_info_formats.format_registry
329
self._default_key = registry._default_key
330
self._dict = registry._dict.copy()
331
self._help_dict = registry._help_dict.copy()
332
self._info_dict = registry._info_dict.copy()
333
self.addCleanup(self._cleanup)
336
# Restore the registry to pristine state after the test runs
337
registry = version_info_formats.format_registry
338
registry._default_key = self._default_key
339
registry._dict = self._dict
340
registry._help_dict = self._help_dict
341
registry._info_dict = self._info_dict
343
def test_register_remove(self):
344
registry = version_info_formats.format_registry
345
registry.register('testbuilder',
346
TestBuilder, 'a simple test builder')
347
self.assertIs(TestBuilder, registry.get('testbuilder'))
348
self.assertEqual('a simple test builder',
349
registry.get_help('testbuilder'))
350
registry.remove('testbuilder')
351
self.assertRaises(KeyError, registry.get, 'testbuilder')