1
# Copyright (C) 2005-2010 Canonical Ltd
1
# Copyright (C) 2005 by Canonical Ltd
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
5
5
# the Free Software Foundation; either version 2 of the License, or
6
6
# (at your option) any later version.
8
8
# This program is distributed in the hope that it will be useful,
9
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11
11
# GNU General Public License for more details.
13
13
# You should have received a copy of the GNU General Public License
14
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
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17
18
"""Tests of status command.
19
20
Most of these depend on the particular formatting used.
20
As such they really are blackbox tests even though some of the
21
tests are not using self.capture. If we add tests for the programmatic
22
interface later, they will be non blackbox tests.
25
from cStringIO import StringIO
27
from os import mkdir, chdir, rmdir, unlink
29
from tempfile import TemporaryFile
38
from bzrlib.osutils import pathjoin
39
from bzrlib.revisionspec import RevisionSpec
40
from bzrlib.status import show_tree_status
41
from bzrlib.tests import TestCaseWithTransport, TestSkipped
42
from bzrlib.workingtree import WorkingTree
45
class BranchStatus(TestCaseWithTransport):
47
def assertStatus(self, expected_lines, working_tree,
48
revision=None, short=False, pending=True, verbose=False):
49
"""Run status in working_tree and look for output.
51
:param expected_lines: The lines to look for.
52
:param working_tree: The tree to run status in.
54
output_string = self.status_string(working_tree, revision, short,
56
self.assertEqual(expected_lines, output_string.splitlines(True))
58
def status_string(self, wt, revision=None, short=False, pending=True,
60
# use a real file rather than StringIO because it doesn't handle
62
tof = codecs.getwriter('utf-8')(TemporaryFile())
63
show_tree_status(wt, to_file=tof, revision=revision, short=short,
64
show_pending=pending, verbose=verbose)
66
return tof.read().decode('utf-8')
68
def test_branch_status(self):
69
"""Test basic branch status"""
70
wt = self.make_branch_and_tree('.')
72
# status with no commits or files - it must
73
# work and show no output. We do this with no
74
# commits to be sure that it's not going to fail
76
self.assertStatus([], wt)
78
self.build_tree(['hello.c', 'bye.c'])
91
# add a commit to allow showing pending merges.
92
wt.commit('create a parent to allow testing merge output')
94
wt.add_parent_tree_id('pending@pending-0-0')
99
'pending merge tips: (use -v to see all merge revisions)\n',
100
' (ghost) pending@pending-0-0\n',
108
' (ghost) pending@pending-0-0\n',
114
'P (ghost) pending@pending-0-0\n',
127
wt, short=True, pending=False)
129
def test_branch_status_revisions(self):
130
"""Tests branch status with revisions"""
131
wt = self.make_branch_and_tree('.')
133
self.build_tree(['hello.c', 'bye.c'])
136
wt.commit('Test message')
138
revs = [RevisionSpec.from_string('0')]
147
self.build_tree(['more.c'])
149
wt.commit('Another test message')
151
revs.append(RevisionSpec.from_string('1'))
160
def test_pending(self):
161
"""Pending merges display works, including Unicode"""
163
wt = self.make_branch_and_tree('branch')
165
wt.commit("Empty commit 1")
166
b_2_dir = b.bzrdir.sprout('./copy')
167
b_2 = b_2_dir.open_branch()
168
wt2 = b_2_dir.open_workingtree()
169
wt.commit(u"\N{TIBETAN DIGIT TWO} Empty commit 2")
170
wt2.merge_from_branch(wt.branch)
171
message = self.status_string(wt2, verbose=True)
172
self.assertStartsWith(message, "pending merges:\n")
173
self.assertEndsWith(message, "Empty commit 2\n")
175
# must be long to make sure we see elipsis at the end
176
wt.commit("Empty commit 3 " +
177
"blah blah blah blah " * 100)
178
wt2.merge_from_branch(wt.branch)
179
message = self.status_string(wt2, verbose=True)
180
self.assertStartsWith(message, "pending merges:\n")
181
self.assert_("Empty commit 3" in message)
182
self.assertEndsWith(message, "...\n")
184
def test_tree_status_ignores(self):
185
"""Tests branch status with ignores"""
186
wt = self.make_branch_and_tree('.')
187
self.run_bzr('ignore *~')
188
wt.commit('commit .bzrignore')
189
self.build_tree(['foo.c', 'foo.c~'])
200
def test_tree_status_specific_files(self):
201
"""Tests branch status with given specific files"""
202
wt = self.make_branch_and_tree('.')
205
self.build_tree(['directory/','directory/hello.c', 'bye.c','test.c','dir2/'])
214
' directory/hello.c\n'
221
'? directory/hello.c\n'
226
self.assertRaises(errors.PathsDoNotExist,
228
wt, specific_files=['bye.c','test.c','absent.c'],
232
show_tree_status(wt, specific_files=['directory'], to_file=tof)
234
self.assertEquals(tof.readlines(),
236
' directory/hello.c\n'
239
show_tree_status(wt, specific_files=['directory'], to_file=tof,
242
self.assertEquals(tof.readlines(), ['? directory/hello.c\n'])
245
show_tree_status(wt, specific_files=['dir2'], to_file=tof)
247
self.assertEquals(tof.readlines(),
252
show_tree_status(wt, specific_files=['dir2'], to_file=tof, short=True)
254
self.assertEquals(tof.readlines(), ['? dir2/\n'])
257
revs = [RevisionSpec.from_string('0'), RevisionSpec.from_string('1')]
258
show_tree_status(wt, specific_files=['test.c'], to_file=tof,
259
short=True, revision=revs)
261
self.assertEquals(tof.readlines(), ['+N test.c\n'])
263
def test_specific_files_conflicts(self):
264
tree = self.make_branch_and_tree('.')
265
self.build_tree(['dir2/'])
267
tree.commit('added dir2')
268
tree.set_conflicts(conflicts.ConflictList(
269
[conflicts.ContentsConflict('foo')]))
271
show_tree_status(tree, specific_files=['dir2'], to_file=tof)
272
self.assertEqualDiff('', tof.getvalue())
273
tree.set_conflicts(conflicts.ConflictList(
274
[conflicts.ContentsConflict('dir2')]))
276
show_tree_status(tree, specific_files=['dir2'], to_file=tof)
277
self.assertEqualDiff('conflicts:\n Contents conflict in dir2\n',
280
tree.set_conflicts(conflicts.ConflictList(
281
[conflicts.ContentsConflict('dir2/file1')]))
283
show_tree_status(tree, specific_files=['dir2'], to_file=tof)
284
self.assertEqualDiff('conflicts:\n Contents conflict in dir2/file1\n',
287
def _prepare_nonexistent(self):
288
wt = self.make_branch_and_tree('.')
289
self.assertStatus([], wt)
290
self.build_tree(['FILE_A', 'FILE_B', 'FILE_C', 'FILE_D', 'FILE_E', ])
296
wt.commit('Create five empty files.')
297
open('FILE_B', 'w').write('Modification to file FILE_B.')
298
open('FILE_C', 'w').write('Modification to file FILE_C.')
299
unlink('FILE_E') # FILE_E will be versioned but missing
300
open('FILE_Q', 'w').write('FILE_Q is added but not committed.')
301
wt.add('FILE_Q') # FILE_Q will be added but not committed
302
open('UNVERSIONED_BUT_EXISTING', 'w')
305
def test_status_nonexistent_file(self):
306
# files that don't exist in either the basis tree or working tree
307
# should give an error
308
wt = self._prepare_nonexistent()
318
' UNVERSIONED_BUT_EXISTING\n',
326
'? UNVERSIONED_BUT_EXISTING\n',
330
# Okay, everything's looking good with the existent files.
331
# Let's see what happens when we throw in non-existent files.
333
# bzr st [--short] NONEXISTENT '
338
out, err = self.run_bzr('status NONEXISTENT', retcode=3)
339
self.assertEqual(expected, out.splitlines(True))
340
self.assertContainsRe(err,
341
r'.*ERROR: Path\(s\) do not exist: '
346
out, err = self.run_bzr('status --short NONEXISTENT', retcode=3)
347
self.assertContainsRe(err,
348
r'.*ERROR: Path\(s\) do not exist: '
351
def test_status_nonexistent_file_with_others(self):
352
# bzr st [--short] NONEXISTENT ...others..
353
wt = self._prepare_nonexistent()
363
out, err = self.run_bzr('status NONEXISTENT '
364
'FILE_A FILE_B FILE_C FILE_D FILE_E',
366
self.assertEqual(expected, out.splitlines(True))
367
self.assertContainsRe(err,
368
r'.*ERROR: Path\(s\) do not exist: '
376
out, err = self.run_bzr('status --short NONEXISTENT '
377
'FILE_A FILE_B FILE_C FILE_D FILE_E',
379
self.assertEqual(expected, out.splitlines(True))
380
self.assertContainsRe(err,
381
r'.*ERROR: Path\(s\) do not exist: '
384
def test_status_multiple_nonexistent_files(self):
385
# bzr st [--short] NONEXISTENT ... ANOTHER_NONEXISTENT ...
386
wt = self._prepare_nonexistent()
394
' ANOTHER_NONEXISTENT\n',
397
out, err = self.run_bzr('status NONEXISTENT '
398
'FILE_A FILE_B ANOTHER_NONEXISTENT '
399
'FILE_C FILE_D FILE_E', retcode=3)
400
self.assertEqual(expected, out.splitlines(True))
401
self.assertContainsRe(err,
402
r'.*ERROR: Path\(s\) do not exist: '
403
'ANOTHER_NONEXISTENT NONEXISTENT.*')
408
'X ANOTHER_NONEXISTENT\n',
411
out, err = self.run_bzr('status --short NONEXISTENT '
412
'FILE_A FILE_B ANOTHER_NONEXISTENT '
413
'FILE_C FILE_D FILE_E', retcode=3)
414
self.assertEqual(expected, out.splitlines(True))
415
self.assertContainsRe(err,
416
r'.*ERROR: Path\(s\) do not exist: '
417
'ANOTHER_NONEXISTENT NONEXISTENT.*')
419
def test_status_nonexistent_file_with_unversioned(self):
420
# bzr st [--short] NONEXISTENT A B UNVERSIONED_BUT_EXISTING C D E Q
421
wt = self._prepare_nonexistent()
431
' UNVERSIONED_BUT_EXISTING\n',
435
out, err = self.run_bzr('status NONEXISTENT '
436
'FILE_A FILE_B UNVERSIONED_BUT_EXISTING '
437
'FILE_C FILE_D FILE_E FILE_Q', retcode=3)
438
self.assertEqual(expected, out.splitlines(True))
439
self.assertContainsRe(err,
440
r'.*ERROR: Path\(s\) do not exist: '
444
'? UNVERSIONED_BUT_EXISTING\n',
450
out, err = self.run_bzr('status --short NONEXISTENT '
451
'FILE_A FILE_B UNVERSIONED_BUT_EXISTING '
452
'FILE_C FILE_D FILE_E FILE_Q', retcode=3)
453
self.assertEqual(expected, out.splitlines(True))
454
self.assertContainsRe(err,
455
r'.*ERROR: Path\(s\) do not exist: '
458
def test_status_out_of_date(self):
459
"""Simulate status of out-of-date tree after remote push"""
460
tree = self.make_branch_and_tree('.')
461
self.build_tree_contents([('a', 'foo\n')])
465
tree.commit('add test file')
466
# simulate what happens after a remote push
467
tree.set_last_revision("0")
469
# before run another commands we should unlock tree
471
out, err = self.run_bzr('status')
472
self.assertEqual("working tree is out of date, run 'bzr update'\n",
475
def test_status_write_lock(self):
476
"""Test that status works without fetching history and
479
See https://bugs.launchpad.net/bzr/+bug/149270
482
wt = self.make_branch_and_tree('branch1')
484
wt.commit('Empty commit 1')
485
wt2 = b.bzrdir.sprout('branch2').open_workingtree()
486
wt2.commit('Empty commit 2')
487
out, err = self.run_bzr('status branch1 -rbranch:branch2')
488
self.assertEqual('', out)
491
class CheckoutStatus(BranchStatus):
494
super(CheckoutStatus, self).setUp()
498
def make_branch_and_tree(self, relpath):
499
source = self.make_branch(pathjoin('..', relpath))
500
checkout = bzrdir.BzrDirMetaFormat1().initialize(relpath)
501
bzrlib.branch.BranchReferenceFormat().initialize(checkout,
502
target_branch=source)
503
return checkout.create_workingtree()
506
class TestStatus(TestCaseWithTransport):
508
def test_status_plain(self):
509
tree = self.make_branch_and_tree('.')
511
self.build_tree(['hello.txt'])
512
result = self.run_bzr("status")[0]
513
self.assertContainsRe(result, "unknown:\n hello.txt\n")
515
tree.add("hello.txt")
516
result = self.run_bzr("status")[0]
517
self.assertContainsRe(result, "added:\n hello.txt\n")
519
tree.commit(message="added")
520
result = self.run_bzr("status -r 0..1")[0]
521
self.assertContainsRe(result, "added:\n hello.txt\n")
523
result = self.run_bzr("status -c 1")[0]
524
self.assertContainsRe(result, "added:\n hello.txt\n")
526
self.build_tree(['world.txt'])
527
result = self.run_bzr("status -r 0")[0]
528
self.assertContainsRe(result, "added:\n hello.txt\n" \
529
"unknown:\n world.txt\n")
530
result2 = self.run_bzr("status -r 0..")[0]
531
self.assertEquals(result2, result)
533
def test_status_short(self):
534
tree = self.make_branch_and_tree('.')
536
self.build_tree(['hello.txt'])
537
result = self.run_bzr("status --short")[0]
538
self.assertContainsRe(result, "[?] hello.txt\n")
540
tree.add("hello.txt")
541
result = self.run_bzr("status --short")[0]
542
self.assertContainsRe(result, "[+]N hello.txt\n")
544
tree.commit(message="added")
545
result = self.run_bzr("status --short -r 0..1")[0]
546
self.assertContainsRe(result, "[+]N hello.txt\n")
548
self.build_tree(['world.txt'])
549
result = self.run_bzr("status --short -r 0")[0]
550
self.assertContainsRe(result, "[+]N hello.txt\n" \
552
result2 = self.run_bzr("status --short -r 0..")[0]
553
self.assertEquals(result2, result)
555
def test_status_versioned(self):
556
tree = self.make_branch_and_tree('.')
558
self.build_tree(['hello.txt'])
559
result = self.run_bzr("status --versioned")[0]
560
self.assertNotContainsRe(result, "unknown:\n hello.txt\n")
562
tree.add("hello.txt")
563
result = self.run_bzr("status --versioned")[0]
564
self.assertContainsRe(result, "added:\n hello.txt\n")
567
result = self.run_bzr("status --versioned -r 0..1")[0]
568
self.assertContainsRe(result, "added:\n hello.txt\n")
570
self.build_tree(['world.txt'])
571
result = self.run_bzr("status --versioned -r 0")[0]
572
self.assertContainsRe(result, "added:\n hello.txt\n")
573
self.assertNotContainsRe(result, "unknown:\n world.txt\n")
574
result2 = self.run_bzr("status --versioned -r 0..")[0]
575
self.assertEquals(result2, result)
577
def test_status_SV(self):
578
tree = self.make_branch_and_tree('.')
580
self.build_tree(['hello.txt'])
581
result = self.run_bzr("status -SV")[0]
582
self.assertNotContainsRe(result, "hello.txt")
584
tree.add("hello.txt")
585
result = self.run_bzr("status -SV")[0]
586
self.assertContainsRe(result, "[+]N hello.txt\n")
588
tree.commit(message="added")
589
result = self.run_bzr("status -SV -r 0..1")[0]
590
self.assertContainsRe(result, "[+]N hello.txt\n")
592
self.build_tree(['world.txt'])
593
result = self.run_bzr("status -SV -r 0")[0]
594
self.assertContainsRe(result, "[+]N hello.txt\n")
596
result2 = self.run_bzr("status -SV -r 0..")[0]
597
self.assertEquals(result2, result)
599
def assertStatusContains(self, pattern, short=False):
600
"""Run status, and assert it contains the given pattern"""
602
result = self.run_bzr("status --short")[0]
604
result = self.run_bzr("status")[0]
605
self.assertContainsRe(result, pattern)
607
def test_kind_change_plain(self):
608
tree = self.make_branch_and_tree('.')
609
self.build_tree(['file'])
611
tree.commit('added file')
613
self.build_tree(['file/'])
614
self.assertStatusContains('kind changed:\n file \(file => directory\)')
615
tree.rename_one('file', 'directory')
616
self.assertStatusContains('renamed:\n file/ => directory/\n' \
617
'modified:\n directory/\n')
619
self.assertStatusContains('removed:\n file\n')
621
def test_kind_change_short(self):
622
tree = self.make_branch_and_tree('.')
623
self.build_tree(['file'])
625
tree.commit('added file')
627
self.build_tree(['file/'])
628
self.assertStatusContains('K file => file/',
630
tree.rename_one('file', 'directory')
631
self.assertStatusContains('RK file => directory/',
634
self.assertStatusContains('RD file => directory',
637
def test_status_illegal_revision_specifiers(self):
638
out, err = self.run_bzr('status -r 1..23..123', retcode=3)
639
self.assertContainsRe(err, 'one or two revision specifiers')
641
def test_status_no_pending(self):
642
a_tree = self.make_branch_and_tree('a')
643
self.build_tree(['a/a'])
646
b_tree = a_tree.bzrdir.sprout('b').open_workingtree()
647
self.build_tree(['b/b'])
651
self.run_bzr('merge ../b', working_dir='a')
652
out, err = self.run_bzr('status --no-pending', working_dir='a')
653
self.assertEquals(out, "added:\n b\n")
655
def test_pending_specific_files(self):
656
"""With a specific file list, pending merges are not shown."""
657
tree = self.make_branch_and_tree('tree')
658
self.build_tree_contents([('tree/a', 'content of a\n')])
660
r1_id = tree.commit('one')
661
alt = tree.bzrdir.sprout('alt').open_workingtree()
662
self.build_tree_contents([('alt/a', 'content of a\nfrom alt\n')])
663
alt_id = alt.commit('alt')
664
tree.merge_from_branch(alt.branch)
665
output = self.make_utf8_encoded_stringio()
666
show_tree_status(tree, to_file=output)
667
self.assertContainsRe(output.getvalue(), 'pending merge')
668
out, err = self.run_bzr('status tree/a')
669
self.assertNotContainsRe(out, 'pending merge')
672
class TestStatusEncodings(TestCaseWithTransport):
675
TestCaseWithTransport.setUp(self)
676
self.user_encoding = osutils._cached_user_encoding
677
self.stdout = sys.stdout
680
osutils._cached_user_encoding = self.user_encoding
681
sys.stdout = self.stdout
682
TestCaseWithTransport.tearDown(self)
684
def make_uncommitted_tree(self):
685
"""Build a branch with uncommitted unicode named changes in the cwd."""
686
working_tree = self.make_branch_and_tree(u'.')
687
filename = u'hell\u00d8'
689
self.build_tree_contents([(filename, 'contents of hello')])
690
except UnicodeEncodeError:
691
raise TestSkipped("can't build unicode working tree in "
692
"filesystem encoding %s" % sys.getfilesystemencoding())
693
working_tree.add(filename)
696
def test_stdout_ascii(self):
697
sys.stdout = StringIO()
698
osutils._cached_user_encoding = 'ascii'
699
working_tree = self.make_uncommitted_tree()
700
stdout, stderr = self.run_bzr("status")
702
self.assertEquals(stdout, """\
707
def test_stdout_latin1(self):
708
sys.stdout = StringIO()
709
osutils._cached_user_encoding = 'latin-1'
710
working_tree = self.make_uncommitted_tree()
711
stdout, stderr = self.run_bzr('status')
713
self.assertEquals(stdout, u"""\
716
""".encode('latin-1'))
24
from bzrlib.selftest import InTempDir
26
class BranchStatus(InTempDir):
28
"""Basic 'bzr mkdir' operation"""
29
from cStringIO import StringIO
30
from bzrlib.status import show_status
31
from bzrlib.branch import Branch
33
b = Branch('.', init=True)
37
show_status(b, to_file=tof)
38
self.assertEquals(tof.getvalue(), "")
41
self.build_tree(['hello.c', 'bye.c'])
42
show_status(b, to_file=tof)
44
self.assertEquals(tof.readlines(),