1
# Copyright (C) 2005-2012, 2016 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"""
27
from ..sixish import (
30
from . import TestCaseWithTransport
31
from ..rio import read_stanzas
33
from ..version_info_formats.format_custom import (
34
CustomVersionInfoBuilder,
35
MissingTemplateVariable,
38
from ..version_info_formats.format_rio import RioVersionInfoBuilder
39
from ..version_info_formats.format_python import PythonVersionInfoBuilder
42
class VersionInfoTestCase(TestCaseWithTransport):
44
def create_branch(self):
45
wt = self.make_branch_and_tree('branch')
47
self.build_tree(['branch/a'])
49
wt.commit('a', rev_id='r1')
51
self.build_tree(['branch/b'])
53
wt.commit('b', rev_id='r2')
55
self.build_tree_contents([('branch/a', 'new contents\n')])
56
wt.commit(u'\xe52', rev_id='r3')
60
def create_tree_with_dotted_revno(self):
61
wt = self.make_branch_and_tree('branch')
62
self.build_tree(['branch/a'])
64
wt.commit('a', rev_id='r1')
66
other = wt.controldir.sprout('other').open_workingtree()
67
self.build_tree(['other/b.a'])
69
other.commit('b.a', rev_id='o2')
72
self.run_bzr('merge ../other')
73
wt.commit('merge', rev_id='merge')
75
wt.update(revision='o2')
80
class TestVersionInfoRio(VersionInfoTestCase):
82
def test_rio_null(self):
83
wt = self.make_branch_and_tree('branch')
86
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt)
89
self.assertContainsRe(val, 'build-date:')
90
self.assertContainsRe(val, 'revno: 0')
92
def test_rio_dotted_revno(self):
93
wt = self.create_tree_with_dotted_revno()
96
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt)
99
self.assertContainsRe(val, 'revno: 1.1.1')
101
def regen_text(self, wt, **kwargs):
103
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt, **kwargs)
104
builder.generate(sio)
108
def test_simple(self):
109
wt = self.create_branch()
111
val = self.regen_text(wt)
112
self.assertContainsRe(val, 'build-date:')
113
self.assertContainsRe(val, 'date:')
114
self.assertContainsRe(val, 'revno: 3')
115
self.assertContainsRe(val, 'revision-id: r3')
117
def test_clean(self):
118
wt = self.create_branch()
119
val = self.regen_text(wt, check_for_clean=True)
120
self.assertContainsRe(val, 'clean: True')
122
def test_no_clean(self):
123
wt = self.create_branch()
124
self.build_tree(['branch/c'])
125
val = self.regen_text(wt, check_for_clean=True)
126
self.assertContainsRe(val, 'clean: False')
128
def test_history(self):
129
wt = self.create_branch()
131
val = self.regen_text(wt, include_revision_history=True)
132
self.assertContainsRe(val, 'id: r1')
133
self.assertContainsRe(val, 'message: a')
134
self.assertContainsRe(val, 'id: r2')
135
self.assertContainsRe(val, 'message: b')
136
self.assertContainsRe(val, 'id: r3')
137
self.assertContainsRe(val, 'message: \xc3\xa52') # utf8 encoding '\xe5'
139
def regen(self, wt, **kwargs):
141
builder = RioVersionInfoBuilder(wt.branch, working_tree=wt, **kwargs)
142
builder.generate(sio)
144
stanzas = list(read_stanzas(sio))
145
self.assertEqual(1, len(stanzas))
148
def test_rio_version_hook(self):
149
def update_stanza(rev, stanza):
150
stanza.add('bla', 'bloe')
151
RioVersionInfoBuilder.hooks.install_named_hook(
152
'revision', update_stanza, None)
153
wt = self.create_branch()
155
stanza = self.regen(wt)
156
self.assertEqual(['bloe'], stanza.get_all('bla'))
158
def get_one_stanza(self, stanza, key):
159
new_stanzas = list(read_stanzas(BytesIO(stanza[key].encode('utf8'))))
160
self.assertEqual(1, len(new_stanzas))
161
return new_stanzas[0]
163
def test_build_date(self):
164
wt = self.create_branch()
165
stanza = self.regen(wt)
166
self.assertTrue('date' in stanza)
167
self.assertTrue('build-date' in stanza)
168
self.assertEqual(['3'], stanza.get_all('revno'))
169
self.assertEqual(['r3'], stanza.get_all('revision-id'))
171
def test_not_clean(self):
172
wt = self.create_branch()
173
self.build_tree(['branch/c'])
174
stanza = self.regen(wt, check_for_clean=True, include_file_revisions=True)
175
self.assertEqual(['False'], stanza.get_all('clean'))
177
def test_file_revisions(self):
178
wt = self.create_branch()
179
self.build_tree(['branch/c'])
180
stanza = self.regen(wt, check_for_clean=True, include_file_revisions=True)
181
# This assumes it's being run against a tree that does not update the
182
# root revision on every commit.
183
file_rev_stanza = self.get_one_stanza(stanza, 'file-revisions')
184
self.assertEqual(['', 'a', 'b', 'c'], file_rev_stanza.get_all('path'))
185
self.assertEqual(['r1', 'r3', 'r2', 'unversioned'],
186
file_rev_stanza.get_all('revision'))
188
def test_revision_history(self):
189
wt = self.create_branch()
190
stanza = self.regen(wt, include_revision_history=True)
191
revision_stanza = self.get_one_stanza(stanza, 'revisions')
192
self.assertEqual(['r1', 'r2', 'r3'], revision_stanza.get_all('id'))
193
self.assertEqual(['a', 'b', u'\xe52'], revision_stanza.get_all('message'))
194
self.assertEqual(3, len(revision_stanza.get_all('date')))
196
def test_file_revisions_with_rename(self):
197
# a was modified, so it should show up modified again
198
wt = self.create_branch()
199
self.build_tree(['branch/a', 'branch/c'])
201
wt.rename_one('b', 'd')
202
stanza = self.regen(wt, check_for_clean=True, include_file_revisions=True)
203
file_rev_stanza = self.get_one_stanza(stanza, 'file-revisions')
204
self.assertEqual(['', 'a', 'b', 'c', 'd'],
205
file_rev_stanza.get_all('path'))
206
self.assertEqual(['r1', 'modified', 'renamed to d', 'new',
208
file_rev_stanza.get_all('revision'))
210
def test_file_revisions_with_removal(self):
211
wt = self.create_branch()
212
self.build_tree(['branch/a', 'branch/c'])
214
wt.rename_one('b', 'd')
216
wt.commit('modified', rev_id='r4')
218
wt.remove(['c', 'd'])
219
os.remove('branch/d')
220
stanza = self.regen(wt, check_for_clean=True, include_file_revisions=True)
221
file_rev_stanza = self.get_one_stanza(stanza, 'file-revisions')
222
self.assertEqual(['', 'a', 'c', 'd'], file_rev_stanza.get_all('path'))
223
self.assertEqual(['r1', 'r4', 'unversioned', 'removed'],
224
file_rev_stanza.get_all('revision'))
226
def test_revision(self):
227
wt = self.create_branch()
228
self.build_tree(['branch/a', 'branch/c'])
230
wt.rename_one('b', 'd')
232
stanza = self.regen(wt, check_for_clean=True,
233
include_file_revisions=True, revision_id=wt.last_revision())
234
file_rev_stanza = self.get_one_stanza(stanza, 'file-revisions')
235
self.assertEqual(['', 'a', 'b'], file_rev_stanza.get_all('path'))
236
self.assertEqual(['r1', 'r3', 'r2'],
237
file_rev_stanza.get_all('revision'))
240
class PythonVersionInfoTests(VersionInfoTestCase):
242
def test_python_null(self):
243
wt = self.make_branch_and_tree('branch')
246
builder = PythonVersionInfoBuilder(wt.branch, working_tree=wt)
247
builder.generate(sio)
249
self.assertContainsRe(val, "'revision_id': None")
250
self.assertContainsRe(val, "'revno': '0'")
251
self.assertNotContainsString(val, '\n\n\n\n')
253
def test_python_dotted_revno(self):
254
wt = self.create_tree_with_dotted_revno()
257
builder = PythonVersionInfoBuilder(wt.branch, working_tree=wt)
258
builder.generate(sio)
260
self.assertContainsRe(val, "'revno': '1.1.1'")
262
def regen(self, wt, **kwargs):
263
"""Create a test module, import and return it"""
264
outf = open('test_version_information.py', 'wb')
266
builder = PythonVersionInfoBuilder(wt.branch, working_tree=wt,
268
builder.generate(outf)
272
module_info = imp.find_module('test_version_information',
274
tvi = imp.load_module('tvi', *module_info)
275
# Make sure the module isn't cached
276
sys.modules.pop('tvi', None)
277
sys.modules.pop('test_version_information', None)
278
# Delete the compiled versions, because we are generating
279
# a new file fast enough that python doesn't detect it
280
# needs to recompile, and using sleep() just makes the
282
if os.path.exists('test_version_information.pyc'):
283
os.remove('test_version_information.pyc')
284
if os.path.exists('test_version_information.pyo'):
285
os.remove('test_version_information.pyo')
288
def test_python_version(self):
289
wt = self.create_branch()
292
self.assertEqual('3', tvi.version_info['revno'])
293
self.assertEqual('r3', tvi.version_info['revision_id'])
294
self.assertTrue('date' in tvi.version_info)
295
self.assertEqual(None, tvi.version_info['clean'])
297
tvi = self.regen(wt, check_for_clean=True)
298
self.assertEqual(True, tvi.version_info['clean'])
300
self.build_tree(['branch/c'])
301
tvi = self.regen(wt, check_for_clean=True, include_file_revisions=True)
302
self.assertEqual(False, tvi.version_info['clean'])
303
self.assertEqual(['', 'a', 'b', 'c'],
304
sorted(tvi.file_revisions.keys()))
305
self.assertEqual('r3', tvi.file_revisions['a'])
306
self.assertEqual('r2', tvi.file_revisions['b'])
307
self.assertEqual('unversioned', tvi.file_revisions['c'])
308
os.remove('branch/c')
310
tvi = self.regen(wt, include_revision_history=True)
312
rev_info = [(rev, message) for rev, message, timestamp, timezone
314
self.assertEqual([('r1', 'a'), ('r2', 'b'), ('r3', u'\xe52')], rev_info)
316
# a was modified, so it should show up modified again
317
self.build_tree(['branch/a', 'branch/c'])
319
wt.rename_one('b', 'd')
320
tvi = self.regen(wt, check_for_clean=True, include_file_revisions=True)
321
self.assertEqual(['', 'a', 'b', 'c', 'd'],
322
sorted(tvi.file_revisions.keys()))
323
self.assertEqual('modified', tvi.file_revisions['a'])
324
self.assertEqual('renamed to d', tvi.file_revisions['b'])
325
self.assertEqual('new', tvi.file_revisions['c'])
326
self.assertEqual('renamed from b', tvi.file_revisions['d'])
328
wt.commit('modified', rev_id='r4')
329
wt.remove(['c', 'd'])
330
os.remove('branch/d')
331
tvi = self.regen(wt, check_for_clean=True, include_file_revisions=True)
332
self.assertEqual(['', 'a', 'c', 'd'],
333
sorted(tvi.file_revisions.keys()))
334
self.assertEqual('r4', tvi.file_revisions['a'])
335
self.assertEqual('unversioned', tvi.file_revisions['c'])
336
self.assertEqual('removed', tvi.file_revisions['d'])
339
class CustomVersionInfoTests(VersionInfoTestCase):
341
def test_custom_null(self):
343
wt = self.make_branch_and_tree('branch')
344
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
345
template='revno: {revno}')
346
builder.generate(sio)
347
self.assertEqual("revno: 0", sio.getvalue())
349
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
350
template='{revno} revid: {revision_id}')
351
# revision_id is not available yet
352
self.assertRaises(MissingTemplateVariable, builder.generate, sio)
354
def test_custom_dotted_revno(self):
356
wt = self.create_tree_with_dotted_revno()
357
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
358
template='{revno} revid: {revision_id}')
359
builder.generate(sio)
360
self.assertEqual("1.1.1 revid: o2", sio.getvalue())
362
def regen(self, wt, tpl, **kwargs):
364
builder = CustomVersionInfoBuilder(wt.branch, working_tree=wt,
365
template=tpl, **kwargs)
366
builder.generate(sio)
370
def test_build_date(self):
371
wt = self.create_branch()
373
val = self.regen(wt, 'build-date: "{build_date}"\ndate: "{date}"')
374
self.assertContainsRe(val, 'build-date: "[0-9-+: ]+"')
375
self.assertContainsRe(val, 'date: "[0-9-+: ]+"')
377
def test_revno(self):
378
wt = self.create_branch()
379
val = self.regen(wt, 'revno: {revno}')
380
self.assertEqual(val, 'revno: 3')
382
def test_revision_id(self):
383
wt = self.create_branch()
384
val = self.regen(wt, 'revision-id: {revision_id}')
385
self.assertEqual(val, 'revision-id: r3')
387
def test_clean(self):
388
wt = self.create_branch()
389
val = self.regen(wt, 'clean: {clean}', check_for_clean=True)
390
self.assertEqual(val, 'clean: 1')
392
def test_not_clean(self):
393
wt = self.create_branch()
395
self.build_tree(['branch/c'])
396
val = self.regen(wt, 'clean: {clean}', check_for_clean=True)
397
self.assertEqual(val, 'clean: 0')
398
os.remove('branch/c')
400
def test_custom_without_template(self):
401
builder = CustomVersionInfoBuilder(None)
403
self.assertRaises(NoTemplate, builder.generate, sio)
406
class TestBuilder(version_info_formats.VersionInfoBuilder):
410
class TestVersionInfoFormatRegistry(tests.TestCase):
413
super(TestVersionInfoFormatRegistry, self).setUp()
414
self.overrideAttr(version_info_formats,
415
'format_registry', registry.Registry())
417
def test_register_remove(self):
418
registry = version_info_formats.format_registry
419
registry.register('testbuilder',
420
TestBuilder, 'a simple test builder')
421
self.assertIs(TestBuilder, registry.get('testbuilder'))
422
self.assertEqual('a simple test builder',
423
registry.get_help('testbuilder'))
424
registry.remove('testbuilder')
425
self.assertRaises(KeyError, registry.get, 'testbuilder')