15
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19
18
from cStringIO import StringIO
23
21
from tempfile import TemporaryFile
25
from bzrlib import tests
26
23
from bzrlib.diff import (
36
from bzrlib.errors import BinaryFile, NoDiff, ExecutableMissing
32
from bzrlib.errors import BinaryFile, NoDiff
37
33
import bzrlib.osutils as osutils
38
import bzrlib.transform as transform
39
34
import bzrlib.patiencediff
40
35
import bzrlib._patiencediff_py
41
36
from bzrlib.tests import (Feature, TestCase, TestCaseWithTransport,
42
37
TestCaseInTempDir, TestSkipped)
45
class _AttribFeature(Feature):
48
if (sys.platform not in ('cygwin', 'win32')):
51
proc = subprocess.Popen(['attrib', '.'], stdout=subprocess.PIPE)
54
return (0 == proc.wait())
56
def feature_name(self):
57
return 'attrib Windows command-line tool'
59
AttribFeature = _AttribFeature()
62
40
class _CompiledPatienceDiffFeature(Feature):
74
52
CompiledPatienceDiffFeature = _CompiledPatienceDiffFeature()
55
class _UnicodeFilename(Feature):
56
"""Does the filesystem support Unicode filenames?"""
61
except UnicodeEncodeError:
63
except (IOError, OSError):
64
# The filesystem allows the Unicode filename but the file doesn't
68
# The filesystem allows the Unicode filename and the file exists,
72
UnicodeFilename = _UnicodeFilename()
75
class TestUnicodeFilename(TestCase):
77
def test_probe_passes(self):
78
"""UnicodeFilename._probe passes."""
79
# We can't test much more than that because the behaviour depends
81
UnicodeFilename._probe()
77
84
def udiff_lines(old, new, allow_binary=False):
78
85
output = StringIO()
79
86
internal_diff('old', old, 'new', new, output, allow_binary)
499
506
self.assertContainsRe(diff, '-contents\n'
500
507
'\\+new contents\n')
503
def test_internal_diff_exec_property(self):
504
tree = self.make_branch_and_tree('tree')
506
tt = transform.TreeTransform(tree)
507
tt.new_file('a', tt.root, 'contents\n', 'a-id', True)
508
tt.new_file('b', tt.root, 'contents\n', 'b-id', False)
509
tt.new_file('c', tt.root, 'contents\n', 'c-id', True)
510
tt.new_file('d', tt.root, 'contents\n', 'd-id', False)
511
tt.new_file('e', tt.root, 'contents\n', 'control-e-id', True)
512
tt.new_file('f', tt.root, 'contents\n', 'control-f-id', False)
514
tree.commit('one', rev_id='rev-1')
516
tt = transform.TreeTransform(tree)
517
tt.set_executability(False, tt.trans_id_file_id('a-id'))
518
tt.set_executability(True, tt.trans_id_file_id('b-id'))
519
tt.set_executability(False, tt.trans_id_file_id('c-id'))
520
tt.set_executability(True, tt.trans_id_file_id('d-id'))
522
tree.rename_one('c', 'new-c')
523
tree.rename_one('d', 'new-d')
525
diff = self.get_diff(tree.basis_tree(), tree)
527
self.assertContainsRe(diff, r"file 'a'.*\(properties changed:.*\+x to -x.*\)")
528
self.assertContainsRe(diff, r"file 'b'.*\(properties changed:.*-x to \+x.*\)")
529
self.assertContainsRe(diff, r"file 'c'.*\(properties changed:.*\+x to -x.*\)")
530
self.assertContainsRe(diff, r"file 'd'.*\(properties changed:.*-x to \+x.*\)")
531
self.assertNotContainsRe(diff, r"file 'e'")
532
self.assertNotContainsRe(diff, r"file 'f'")
535
509
def test_binary_unicode_filenames(self):
536
510
"""Test that contents of files are *not* encoded in UTF-8 when there
537
511
is a binary file in the diff.
539
513
# See https://bugs.launchpad.net/bugs/110092.
540
self.requireFeature(tests.UnicodeFilenameFeature)
514
self.requireFeature(UnicodeFilename)
542
516
# This bug isn't triggered with cStringIO.
543
517
from StringIO import StringIO
1260
1233
from bzrlib._patiencediff_py import recurse_matches_py
1261
1234
self.assertIs(recurse_matches_py,
1262
1235
bzrlib.patiencediff.recurse_matches)
1265
class TestDiffFromTool(TestCaseWithTransport):
1267
def test_from_string(self):
1268
diff_obj = DiffFromTool.from_string('diff', None, None, None)
1269
self.addCleanup(diff_obj.finish)
1270
self.assertEqual(['diff', '%(old_path)s', '%(new_path)s'],
1271
diff_obj.command_template)
1273
def test_from_string_u5(self):
1274
diff_obj = DiffFromTool.from_string('diff -u\\ 5', None, None, None)
1275
self.addCleanup(diff_obj.finish)
1276
self.assertEqual(['diff', '-u 5', '%(old_path)s', '%(new_path)s'],
1277
diff_obj.command_template)
1278
self.assertEqual(['diff', '-u 5', 'old-path', 'new-path'],
1279
diff_obj._get_command('old-path', 'new-path'))
1281
def test_execute(self):
1283
diff_obj = DiffFromTool(['python', '-c',
1284
'print "%(old_path)s %(new_path)s"'],
1286
self.addCleanup(diff_obj.finish)
1287
diff_obj._execute('old', 'new')
1288
self.assertEqual(output.getvalue().rstrip(), 'old new')
1290
def test_excute_missing(self):
1291
diff_obj = DiffFromTool(['a-tool-which-is-unlikely-to-exist'],
1293
self.addCleanup(diff_obj.finish)
1294
e = self.assertRaises(ExecutableMissing, diff_obj._execute, 'old',
1296
self.assertEqual('a-tool-which-is-unlikely-to-exist could not be found'
1297
' on this machine', str(e))
1299
def test_prepare_files_creates_paths_readable_by_windows_tool(self):
1300
self.requireFeature(AttribFeature)
1302
tree = self.make_branch_and_tree('tree')
1303
self.build_tree_contents([('tree/file', 'content')])
1304
tree.add('file', 'file-id')
1305
tree.commit('old tree')
1307
self.addCleanup(tree.unlock)
1308
diff_obj = DiffFromTool(['python', '-c',
1309
'print "%(old_path)s %(new_path)s"'],
1311
diff_obj._prepare_files('file-id', 'file', 'file')
1312
self.assertReadableByAttrib(diff_obj._root, 'old\\file', r'old\\file')
1313
self.assertReadableByAttrib(diff_obj._root, 'new\\file', r'new\\file')
1315
def assertReadableByAttrib(self, cwd, relpath, regex):
1316
proc = subprocess.Popen(['attrib', relpath],
1317
stdout=subprocess.PIPE,
1320
result = proc.stdout.read()
1321
self.assertContainsRe(result, regex)
1323
def test_prepare_files(self):
1325
tree = self.make_branch_and_tree('tree')
1326
self.build_tree_contents([('tree/oldname', 'oldcontent')])
1327
self.build_tree_contents([('tree/oldname2', 'oldcontent2')])
1328
tree.add('oldname', 'file-id')
1329
tree.add('oldname2', 'file2-id')
1330
tree.commit('old tree', timestamp=0)
1331
tree.rename_one('oldname', 'newname')
1332
tree.rename_one('oldname2', 'newname2')
1333
self.build_tree_contents([('tree/newname', 'newcontent')])
1334
self.build_tree_contents([('tree/newname2', 'newcontent2')])
1335
old_tree = tree.basis_tree()
1336
old_tree.lock_read()
1337
self.addCleanup(old_tree.unlock)
1339
self.addCleanup(tree.unlock)
1340
diff_obj = DiffFromTool(['python', '-c',
1341
'print "%(old_path)s %(new_path)s"'],
1342
old_tree, tree, output)
1343
self.addCleanup(diff_obj.finish)
1344
self.assertContainsRe(diff_obj._root, 'bzr-diff-[^/]*')
1345
old_path, new_path = diff_obj._prepare_files('file-id', 'oldname',
1347
self.assertContainsRe(old_path, 'old/oldname$')
1348
self.assertEqual(0, os.stat(old_path).st_mtime)
1349
self.assertContainsRe(new_path, 'new/newname$')
1350
self.assertFileEqual('oldcontent', old_path)
1351
self.assertFileEqual('newcontent', new_path)
1352
if osutils.host_os_dereferences_symlinks():
1353
self.assertTrue(os.path.samefile('tree/newname', new_path))
1354
# make sure we can create files with the same parent directories
1355
diff_obj._prepare_files('file2-id', 'oldname2', 'newname2')