1
# Copyright (C) 2005 by 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17
"""Tests for the osutils wrapper."""
26
from bzrlib.errors import BzrBadParameterNotUnicode, InvalidURL
27
import bzrlib.osutils as osutils
28
from bzrlib.tests import (
36
class TestOSUtils(TestCaseInTempDir):
38
def test_fancy_rename(self):
39
# This should work everywhere
41
osutils.fancy_rename(a, b,
42
rename_func=os.rename,
43
unlink_func=os.unlink)
45
open('a', 'wb').write('something in a\n')
47
self.failIfExists('a')
48
self.failUnlessExists('b')
49
self.check_file_contents('b', 'something in a\n')
51
open('a', 'wb').write('new something in a\n')
54
self.check_file_contents('a', 'something in a\n')
56
def test_rename(self):
57
# Rename should be semi-atomic on all platforms
58
open('a', 'wb').write('something in a\n')
59
osutils.rename('a', 'b')
60
self.failIfExists('a')
61
self.failUnlessExists('b')
62
self.check_file_contents('b', 'something in a\n')
64
open('a', 'wb').write('new something in a\n')
65
osutils.rename('b', 'a')
67
self.check_file_contents('a', 'something in a\n')
69
# TODO: test fancy_rename using a MemoryTransport
71
def test_01_rand_chars_empty(self):
72
result = osutils.rand_chars(0)
73
self.assertEqual(result, '')
75
def test_02_rand_chars_100(self):
76
result = osutils.rand_chars(100)
77
self.assertEqual(len(result), 100)
78
self.assertEqual(type(result), str)
79
self.assertContainsRe(result, r'^[a-z0-9]{100}$')
82
def test_rmtree(self):
83
# Check to remove tree with read-only files/dirs
85
f = file('dir/file', 'w')
88
# would like to also try making the directory readonly, but at the
89
# moment python shutil.rmtree doesn't handle that properly - it would
90
# need to chmod the directory before removing things inside it - deferred
91
# for now -- mbp 20060505
92
# osutils.make_readonly('dir')
93
osutils.make_readonly('dir/file')
97
self.failIfExists('dir/file')
98
self.failIfExists('dir')
100
def test_file_kind(self):
101
self.build_tree(['file', 'dir/'])
102
self.assertEquals('file', osutils.file_kind('file'))
103
self.assertEquals('directory', osutils.file_kind('dir/'))
104
if osutils.has_symlinks():
105
os.symlink('symlink', 'symlink')
106
self.assertEquals('symlink', osutils.file_kind('symlink'))
108
# TODO: jam 20060529 Test a block device
110
os.lstat('/dev/null')
112
if e.errno not in (errno.ENOENT,):
115
self.assertEquals('chardev', osutils.file_kind('/dev/null'))
117
mkfifo = getattr(os, 'mkfifo', None)
121
self.assertEquals('fifo', osutils.file_kind('fifo'))
125
AF_UNIX = getattr(socket, 'AF_UNIX', None)
127
s = socket.socket(AF_UNIX)
130
self.assertEquals('socket', osutils.file_kind('socket'))
135
class TestSafeUnicode(TestCase):
137
def test_from_ascii_string(self):
138
self.assertEqual(u'foobar', osutils.safe_unicode('foobar'))
140
def test_from_unicode_string_ascii_contents(self):
141
self.assertEqual(u'bargam', osutils.safe_unicode(u'bargam'))
143
def test_from_unicode_string_unicode_contents(self):
144
self.assertEqual(u'bargam\xae', osutils.safe_unicode(u'bargam\xae'))
146
def test_from_utf8_string(self):
147
self.assertEqual(u'foo\xae', osutils.safe_unicode('foo\xc2\xae'))
149
def test_bad_utf8_string(self):
150
self.assertRaises(BzrBadParameterNotUnicode,
151
osutils.safe_unicode,
155
class TestWin32Funcs(TestCase):
156
"""Test that the _win32 versions of os utilities return appropriate paths."""
158
def test_abspath(self):
159
self.assertEqual('C:/foo', osutils._win32_abspath('C:\\foo'))
160
self.assertEqual('C:/foo', osutils._win32_abspath('C:/foo'))
162
def test_realpath(self):
163
self.assertEqual('C:/foo', osutils._win32_realpath('C:\\foo'))
164
self.assertEqual('C:/foo', osutils._win32_realpath('C:/foo'))
166
def test_pathjoin(self):
167
self.assertEqual('path/to/foo', osutils._win32_pathjoin('path', 'to', 'foo'))
168
self.assertEqual('C:/foo', osutils._win32_pathjoin('path\\to', 'C:\\foo'))
169
self.assertEqual('C:/foo', osutils._win32_pathjoin('path/to', 'C:/foo'))
170
self.assertEqual('path/to/foo', osutils._win32_pathjoin('path/to/', 'foo'))
171
self.assertEqual('/foo', osutils._win32_pathjoin('C:/path/to/', '/foo'))
172
self.assertEqual('/foo', osutils._win32_pathjoin('C:\\path\\to\\', '\\foo'))
174
def test_normpath(self):
175
self.assertEqual('path/to/foo', osutils._win32_normpath(r'path\\from\..\to\.\foo'))
176
self.assertEqual('path/to/foo', osutils._win32_normpath('path//from/../to/./foo'))
178
def test_getcwd(self):
179
self.assertEqual(os.getcwdu().replace('\\', '/'), osutils._win32_getcwd())
182
class TestWin32FuncsDirs(TestCaseInTempDir):
183
"""Test win32 functions that create files."""
185
def test_getcwd(self):
186
# Make sure getcwd can handle unicode filenames
188
os.mkdir(u'B\xe5gfors')
190
raise TestSkipped("Unable to create Unicode filename")
192
os.chdir(u'B\xe5gfors')
193
# TODO: jam 20060427 This will probably fail on Mac OSX because
194
# it will change the normalization of B\xe5gfors
195
# Consider using a different unicode character, or make
196
# osutils.getcwd() renormalize the path.
197
self.assertTrue(osutils._win32_getcwd().endswith(u'/B\xe5gfors'))
199
def test_mkdtemp(self):
200
tmpdir = osutils._win32_mkdtemp(dir='.')
201
self.assertFalse('\\' in tmpdir)
203
def test_rename(self):
211
osutils._win32_rename('b', 'a')
212
self.failUnlessExists('a')
213
self.failIfExists('b')
214
self.assertFileEqual('baz\n', 'a')
217
class TestSplitLines(TestCase):
219
def test_split_unicode(self):
220
self.assertEqual([u'foo\n', u'bar\xae'],
221
osutils.split_lines(u'foo\nbar\xae'))
222
self.assertEqual([u'foo\n', u'bar\xae\n'],
223
osutils.split_lines(u'foo\nbar\xae\n'))
225
def test_split_with_carriage_returns(self):
226
self.assertEqual(['foo\rbar\n'],
227
osutils.split_lines('foo\rbar\n'))
230
class TestWalkDirs(TestCaseInTempDir):
232
def test_walkdirs(self):
241
self.build_tree(tree)
242
expected_dirblocks = [
244
('0file', '0file', 'file'),
245
('1dir', '1dir', 'directory'),
246
('2file', '2file', 'file'),
249
('1dir/0file', '0file', 'file'),
250
('1dir/1dir', '1dir', 'directory'),
257
for dirblock in osutils.walkdirs('.'):
258
if len(dirblock) and dirblock[0][1] == '.bzr':
259
# this tests the filtering of selected paths
262
result.append(dirblock)
264
self.assertTrue(found_bzrdir)
265
self.assertEqual(expected_dirblocks,
266
[[line[0:3] for line in block] for block in result])
267
# you can search a subdir only, with a supplied prefix.
269
for dirblock in osutils.walkdirs('1dir', '1dir'):
270
result.append(dirblock)
271
self.assertEqual(expected_dirblocks[1:],
272
[[line[0:3] for line in block] for block in result])
274
def assertPathCompare(self, path_less, path_greater):
275
"""check that path_less and path_greater compare correctly."""
276
self.assertEqual(0, osutils.compare_paths_prefix_order(
277
path_less, path_less))
278
self.assertEqual(0, osutils.compare_paths_prefix_order(
279
path_greater, path_greater))
280
self.assertEqual(-1, osutils.compare_paths_prefix_order(
281
path_less, path_greater))
282
self.assertEqual(1, osutils.compare_paths_prefix_order(
283
path_greater, path_less))
285
def test_compare_paths_prefix_order(self):
286
# root before all else
287
self.assertPathCompare("/", "/a")
289
self.assertPathCompare("/a", "/b")
290
self.assertPathCompare("/b", "/z")
291
# high dirs before lower.
292
self.assertPathCompare("/z", "/a/a")
293
# except if the deeper dir should be output first
294
self.assertPathCompare("/a/b/c", "/d/g")
295
# lexical betwen dirs of the same height
296
self.assertPathCompare("/a/z", "/z/z")
297
self.assertPathCompare("/a/c/z", "/a/d/e")
299
# this should also be consistent for no leading / paths
300
# root before all else
301
self.assertPathCompare("", "a")
303
self.assertPathCompare("a", "b")
304
self.assertPathCompare("b", "z")
305
# high dirs before lower.
306
self.assertPathCompare("z", "a/a")
307
# except if the deeper dir should be output first
308
self.assertPathCompare("a/b/c", "d/g")
309
# lexical betwen dirs of the same height
310
self.assertPathCompare("a/z", "z/z")
311
self.assertPathCompare("a/c/z", "a/d/e")
313
def test_path_prefix_sorting(self):
314
"""Doing a sort on path prefix should match our sample data."""
345
sorted(original_paths, key=osutils.path_prefix_key))
346
# using the comparison routine shoudl work too:
349
sorted(original_paths, cmp=osutils.compare_paths_prefix_order))
352
class TestTerminalEncoding(TestCase):
353
"""Test the auto-detection of proper terminal encoding."""
356
self._stdout = sys.stdout
357
self._stderr = sys.stderr
358
self._stdin = sys.stdin
359
self._user_encoding = bzrlib.user_encoding
361
self.addCleanup(self._reset)
363
sys.stdout = StringIOWrapper()
364
sys.stdout.encoding = 'stdout_encoding'
365
sys.stderr = StringIOWrapper()
366
sys.stderr.encoding = 'stderr_encoding'
367
sys.stdin = StringIOWrapper()
368
sys.stdin.encoding = 'stdin_encoding'
369
bzrlib.user_encoding = 'user_encoding'
372
sys.stdout = self._stdout
373
sys.stderr = self._stderr
374
sys.stdin = self._stdin
375
bzrlib.user_encoding = self._user_encoding
377
def test_get_terminal_encoding(self):
378
# first preference is stdout encoding
379
self.assertEqual('stdout_encoding', osutils.get_terminal_encoding())
381
sys.stdout.encoding = None
382
# if sys.stdout is None, fall back to sys.stdin
383
self.assertEqual('stdin_encoding', osutils.get_terminal_encoding())
385
sys.stdin.encoding = None
386
# and in the worst case, use bzrlib.user_encoding
387
self.assertEqual('user_encoding', osutils.get_terminal_encoding())