1
# Copyright (C) 2006 by Canonical Ltd
2
# Authors: Robert Collins <robert.collins@canonical.com>
3
# -*- coding: utf-8 -*-
1
# Copyright (C) 2006-2011 Canonical Ltd
5
3
# This program is free software; you can redistribute it and/or modify
6
4
# it under the terms of the GNU General Public License as published by
7
5
# the Free Software Foundation; either version 2 of the License, or
8
6
# (at your option) any later version.
10
8
# This program is distributed in the hope that it will be useful,
11
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
12
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
11
# GNU General Public License for more details.
15
13
# You should have received a copy of the GNU General Public License
16
14
# along with this program; if not, write to the Free Software
17
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19
17
"""Black box tests for the upgrade ui."""
23
import bzrlib.bzrdir as bzrdir
24
import bzrlib.repository as repository
25
from bzrlib.tests import TestCaseWithTransport
26
from bzrlib.tests.blackbox import TestUIFactory
28
from bzrlib.tests import (
30
TestCaseWithTransport,
27
32
from bzrlib.tests.test_sftp_transport import TestCaseWithSFTPServer
28
from bzrlib.transport import get_transport
29
import bzrlib.ui as ui
33
from bzrlib.repofmt.knitpack_repo import RepositoryFormatKnitPack1
36
class OldBzrDir(bzrdir.BzrDirMeta1):
37
"""An test bzr dir implementation"""
39
def needs_format_conversion(self, format):
40
return not isinstance(format, self.__class__)
43
class ConvertOldTestToMeta(controldir.Converter):
44
"""A trivial converter, used for testing."""
46
def convert(self, to_convert, pb):
47
ui.ui_factory.note('starting upgrade from old test format to 2a')
48
to_convert.control_transport.put_bytes(
50
bzrdir.BzrDirMetaFormat1().get_format_string(),
51
mode=to_convert._get_file_mode())
52
return controldir.ControlDir.open(to_convert.user_url)
55
class OldBzrDirFormat(bzrdir.BzrDirMetaFormat1):
57
_lock_class = lockable_files.TransportLock
59
def get_converter(self, format=None):
60
return ConvertOldTestToMeta()
63
def get_format_string(cls):
64
return "Ancient Test Format"
66
def _open(self, transport):
67
return OldBzrDir(transport, self)
32
70
class TestWithUpgradableBranches(TestCaseWithTransport):
35
73
super(TestWithUpgradableBranches, self).setUp()
36
self.old_format = bzrdir.BzrDirFormat.get_default_format()
37
self.old_ui_factory = ui.ui_factory
38
self.addCleanup(self.restoreDefaults)
40
ui.ui_factory = TestUIFactory()
41
bzrdir.BzrDirFormat.set_default_format(bzrdir.BzrDirMetaFormat1())
42
# FIXME RBC 20060120 we should be able to do this via ui calls only.
43
# setup a format 5 branch we can upgrade from.
44
t = get_transport(self.get_url())
45
t.mkdir('format_5_branch')
46
bzrdir.BzrDirFormat5().initialize(self.get_url('format_5_branch'))
47
bzrdir.BzrDir.create_standalone_workingtree('current_format_branch')
48
d = bzrdir.BzrDir.create('metadir_weave_branch')
51
d.create_workingtree()
52
self.run_bzr('checkout',
54
self.get_url('current_format_branch'),
55
'current_format_checkout')
57
def restoreDefaults(self):
58
bzrdir.BzrDirFormat.set_default_format(self.old_format)
59
ui.ui_factory = self.old_ui_factory
75
def make_current_format_branch_and_checkout(self):
76
current_tree = self.make_branch_and_tree('current_format_branch',
78
current_tree.branch.create_checkout(
79
self.get_url('current_format_checkout'), lightweight=True)
61
81
def test_readonly_url_error(self):
62
(out, err) = self.run_bzr_captured(
63
['upgrade', self.get_readonly_url('format_5_branch')], 3)
64
self.assertEqual(out, "")
65
self.assertEqual(err, "bzr: ERROR: Upgrade URL cannot work with readonly URL's.\n")
82
self.make_branch_and_tree("old_format_branch", format="knit")
83
(out, err) = self.run_bzr(
84
['upgrade', self.get_readonly_url("old_format_branch")], retcode=3)
85
err_msg = 'Upgrade URL cannot work with readonly URLs.'
86
self.assertEqualDiff('conversion error: %s\nbzr: ERROR: %s\n'
67
90
def test_upgrade_up_to_date(self):
91
self.make_current_format_branch_and_checkout()
68
92
# when up to date we should get a message to that effect
69
(out, err) = self.run_bzr_captured(
70
['upgrade', 'current_format_branch'], 3)
71
self.assertEqual("", out)
72
self.assertEqualDiff("bzr: ERROR: The branch format Bazaar-NG meta "
73
"directory, format 1 is already at the most "
74
"recent format.\n", err)
93
burl = self.get_transport('current_format_branch').local_abspath(".")
94
(out, err) = self.run_bzr('upgrade current_format_branch', retcode=0)
96
'Upgrading branch %s/ ...\n'
97
'The branch format %s is already at the most recent format.\n'
98
% (burl, 'Meta directory format 1'),
76
101
def test_upgrade_up_to_date_checkout_warns_branch_left_alone(self):
102
self.make_current_format_branch_and_checkout()
77
103
# when upgrading a checkout, the branch location and a suggestion
78
# to upgrade it should be emitted even if the checkout is up to
104
# to upgrade it should be emitted even if the checkout is up to
80
(out, err) = self.run_bzr_captured(
81
['upgrade', 'current_format_checkout'], 3)
82
self.assertEqual("This is a checkout. The branch (%s) needs to be "
83
"upgraded separately.\n"
84
% get_transport(self.get_url('current_format_branch')).base,
86
self.assertEqualDiff("bzr: ERROR: The branch format Bazaar-NG meta "
87
"directory, format 1 is already at the most "
88
"recent format.\n", err)
106
burl = self.get_transport('current_format_branch').local_abspath(".")
107
curl = self.get_transport('current_format_checkout').local_abspath(".")
108
(out, err) = self.run_bzr('upgrade current_format_checkout', retcode=0)
110
'Upgrading branch %s/ ...\nThis is a checkout.'
111
' The branch (%s/) needs to be upgraded separately.\n'
112
'The branch format %s is already at the most recent format.\n'
113
% (curl, burl, 'Meta directory format 1'),
90
116
def test_upgrade_checkout(self):
91
117
# upgrading a checkout should work
94
120
def test_upgrade_repository_scans_branches(self):
95
# we should get individual upgrade notes for each branch even the
121
# we should get individual upgrade notes for each branch even the
96
122
# anonymous branch
99
def test_ugrade_branch_in_repo(self):
125
def test_upgrade_branch_in_repo(self):
100
126
# upgrading a branch in a repo should warn about not upgrading the repo
103
def test_upgrade_explicit_metaformat(self):
104
# users can force an upgrade to metadir format.
105
url = get_transport(self.get_url('format_5_branch')).base
129
def test_upgrade_control_dir(self):
130
old_format = OldBzrDirFormat()
131
self.addCleanup(bzrdir.BzrProber.formats.remove,
132
old_format.get_format_string())
133
bzrdir.BzrProber.formats.register(old_format.get_format_string(),
135
self.addCleanup(controldir.ControlDirFormat._set_default_format,
136
controldir.ControlDirFormat.get_default_format())
138
# setup an old format branch we can upgrade from.
139
path = 'old_format_branch'
140
self.make_branch_and_tree(path, format=old_format)
141
transport = self.get_transport(path)
143
display_url = transport.local_abspath('.')
106
144
# check --format takes effect
107
bzrdir.BzrDirFormat.set_default_format(bzrdir.BzrDirFormat5())
108
(out, err) = self.run_bzr_captured(
109
['upgrade', '--format=metadir', url])
110
self.assertEqualDiff("""starting upgrade of %s
111
making backup of tree history
112
%s.bzr has been backed up to %s.bzr.backup
113
if conversion fails, you can move this directory back to .bzr
114
if it succeeds, you can remove this directory if you wish
115
starting upgrade from format 5 to 6
116
adding prefixes to weaves
117
adding prefixes to revision-store
118
starting upgrade from format 6 to metadir
145
controldir.ControlDirFormat._set_default_format(old_format)
146
backup_dir = 'backup.bzr.~1~'
147
(out, err) = self.run_bzr(
148
['upgrade', '--format=2a', url])
149
self.assertEqualDiff("""Upgrading branch %s/ ...
150
starting upgrade of %s/
151
making backup of %s/.bzr
153
starting upgrade from old test format to 2a
120
""" % (url, url, url), out)
155
""" % (display_url, display_url, display_url, display_url, backup_dir), out)
121
156
self.assertEqualDiff("", err)
122
157
self.assertTrue(isinstance(
123
bzrdir.BzrDir.open(self.get_url('format_5_branch'))._format,
158
controldir.ControlDir.open(self.get_url(path))._format,
124
159
bzrdir.BzrDirMetaFormat1))
126
161
def test_upgrade_explicit_knit(self):
127
# users can force an upgrade to knit format from a metadir weave
129
url = get_transport(self.get_url('metadir_weave_branch')).base
162
# users can force an upgrade to knit format from a metadir pack 0.92
163
# branch to a 2a branch.
164
self.make_branch_and_tree('branch', format='knit')
165
transport = self.get_transport('branch')
167
display_url = transport.local_abspath('.')
130
168
# check --format takes effect
131
bzrdir.BzrDirFormat.set_default_format(bzrdir.BzrDirFormat5())
132
(out, err) = self.run_bzr_captured(
133
['upgrade', '--format=knit', url])
134
self.assertEqualDiff("""starting upgrade of %s
135
making backup of tree history
136
%s.bzr has been backed up to %s.bzr.backup
137
if conversion fails, you can move this directory back to .bzr
138
if it succeeds, you can remove this directory if you wish
169
backup_dir = 'backup.bzr.~1~'
170
(out, err) = self.run_bzr(
171
['upgrade', '--format=pack-0.92', url])
172
self.assertEqualDiff("""Upgrading branch %s/ ...
173
starting upgrade of %s/
174
making backup of %s/.bzr
139
176
starting repository conversion
140
177
repository converted
142
""" % (url, url, url), out)
179
""" % (display_url, display_url, display_url, display_url, backup_dir),
143
181
self.assertEqualDiff("", err)
144
converted_dir = bzrdir.BzrDir.open(self.get_url('metadir_weave_branch'))
182
converted_dir = controldir.ControlDir.open(self.get_url('branch'))
145
183
self.assertTrue(isinstance(converted_dir._format,
146
184
bzrdir.BzrDirMetaFormat1))
147
185
self.assertTrue(isinstance(converted_dir.open_repository()._format,
148
repository.RepositoryFormatKnit1))
186
RepositoryFormatKnitPack1))
150
188
def test_upgrade_repo(self):
151
self.run_bzr('init-repository', '--format=metadir', 'repo')
152
self.run_bzr('upgrade', '--format=knit', 'repo')
189
self.run_bzr('init-repository --format=pack-0.92 repo')
190
self.run_bzr('upgrade --format=2a repo')
192
def assertLegalOption(self, option_str):
193
# Confirm that an option is legal. (Lower level tests are
194
# expected to validate the actual functionality.)
195
self.run_bzr('init --format=pack-0.92 branch-foo')
196
self.run_bzr('upgrade --format=2a branch-foo %s' % (option_str,))
198
def assertBranchFormat(self, dir, format):
199
branch = controldir.ControlDir.open_tree_or_branch(self.get_url(dir))[1]
200
branch_format = branch._format
201
meta_format = controldir.format_registry.make_bzrdir(format)
202
expected_format = meta_format.get_branch_format()
203
self.assertEqual(expected_format, branch_format)
205
def test_upgrade_clean_supported(self):
206
self.assertLegalOption('--clean')
207
self.assertBranchFormat('branch-foo', '2a')
208
backup_bzr_dir = os.path.join("branch-foo", "backup.bzr.~1~")
209
self.assertFalse(os.path.exists(backup_bzr_dir))
211
def test_upgrade_dry_run_supported(self):
212
self.assertLegalOption('--dry-run')
213
self.assertBranchFormat('branch-foo', 'pack-0.92')
215
def test_upgrade_permission_check(self):
216
"""'backup.bzr' should retain permissions of .bzr. Bug #262450"""
217
self.requireFeature(features.posix_permissions_feature)
218
old_perms = stat.S_IRUSR | stat.S_IWUSR | stat.S_IXUSR
219
backup_dir = 'backup.bzr.~1~'
220
self.run_bzr('init --format=1.6')
221
os.chmod('.bzr', old_perms)
222
self.run_bzr('upgrade')
223
new_perms = os.stat(backup_dir).st_mode & 0777
224
self.assertTrue(new_perms == old_perms)
226
def test_upgrade_with_existing_backup_dir(self):
227
self.make_branch_and_tree("old_format_branch", format="knit")
228
t = self.get_transport("old_format_branch")
230
display_url = t.local_abspath('.')
231
backup_dir1 = 'backup.bzr.~1~'
232
backup_dir2 = 'backup.bzr.~2~'
233
# explicitly create backup_dir1. bzr should create the .~2~ directory
236
(out, err) = self.run_bzr(
237
['upgrade', '--format=2a', url])
238
self.assertEqualDiff("""Upgrading branch %s/ ...
239
starting upgrade of %s/
240
making backup of %s/.bzr
242
starting repository conversion
245
""" % (display_url, display_url, display_url, display_url, backup_dir2), out)
246
self.assertEqualDiff("", err)
247
self.assertTrue(isinstance(
248
controldir.ControlDir.open(self.get_url("old_format_branch"))._format,
249
bzrdir.BzrDirMetaFormat1))
250
self.assertTrue(t.has(backup_dir2))
155
253
class SFTPTests(TestCaseWithSFTPServer):
156
254
"""Tests for upgrade over sftp."""
159
super(SFTPTests, self).setUp()
160
self.old_ui_factory = ui.ui_factory
161
self.addCleanup(self.restoreDefaults)
163
ui.ui_factory = TestUIFactory()
165
def restoreDefaults(self):
166
ui.ui_factory = self.old_ui_factory
168
256
def test_upgrade_url(self):
169
self.run_bzr('init', '--format=weave')
170
t = get_transport(self.get_url())
257
self.run_bzr('init --format=pack-0.92')
258
t = self.get_transport()
172
out, err = self.run_bzr('upgrade', '--format=knit', url)
173
self.assertEqualDiff("""starting upgrade of %s
174
making backup of tree history
175
%s.bzr has been backed up to %s.bzr.backup
176
if conversion fails, you can move this directory back to .bzr
177
if it succeeds, you can remove this directory if you wish
178
starting upgrade from format 6 to metadir
260
display_url = urlutils.unescape_for_display(url,
262
out, err = self.run_bzr(['upgrade', '--format=2a', url])
263
backup_dir = 'backup.bzr.~1~'
264
self.assertEqualDiff("""Upgrading branch %s ...
265
starting upgrade of %s
266
making backup of %s.bzr
179
268
starting repository conversion
180
269
repository converted
182
""" % (url, url, url), out)
271
""" % (display_url, display_url, display_url, display_url, backup_dir), out)
183
272
self.assertEqual('', err)
275
class UpgradeRecommendedTests(TestCaseWithTransport):
277
def test_recommend_upgrade_wt4(self):
278
# using a deprecated format gives a warning
279
self.run_bzr('init --knit a')
280
out, err = self.run_bzr('status a')
281
self.assertContainsRe(err, 'bzr upgrade .*[/\\\\]a')
283
def test_no_upgrade_recommendation_from_bzrdir(self):
284
# we should only get a recommendation to upgrade when we're accessing
285
# the actual workingtree, not when we only open a bzrdir that contains
287
self.run_bzr('init --knit a')
288
out, err = self.run_bzr('revno a')
289
if err.find('upgrade') > -1:
290
self.fail("message shouldn't suggest upgrade:\n%s" % err)
292
def test_upgrade_shared_repo(self):
293
repo = self.make_repository('repo', format='2a', shared=True)
294
branch = self.make_branch_and_tree('repo/branch', format="pack-0.92")
295
self.get_transport('repo/branch/.bzr/repository').delete_tree('.')
296
out, err = self.run_bzr(['upgrade'], working_dir='repo/branch')