196
181
self.assertEqual('path/..', dirname('path/../foo'))
197
182
self.assertEqual('../path', dirname('../path/foo'))
200
def test(expected, *args):
201
joined = urlutils.join(*args)
202
self.assertEqual(expected, joined)
204
# Test relative path joining
205
test('foo', 'foo') # relative fragment with nothing is preserved.
206
test('foo/bar', 'foo', 'bar')
207
test('http://foo/bar', 'http://foo', 'bar')
208
test('http://foo/bar', 'http://foo', '.', 'bar')
209
test('http://foo/baz', 'http://foo', 'bar', '../baz')
210
test('http://foo/bar/baz', 'http://foo', 'bar/baz')
211
test('http://foo/baz', 'http://foo', 'bar/../baz')
212
test('http://foo/baz', 'http://foo/bar/', '../baz')
215
test('http://foo', 'http://foo') # abs url with nothing is preserved.
216
test('http://bar', 'http://foo', 'http://bar')
217
test('sftp://bzr/foo', 'http://foo', 'bar', 'sftp://bzr/foo')
218
test('file:///bar', 'foo', 'file:///bar')
219
test('http://bar/', 'http://foo', 'http://bar/')
220
test('http://bar/a', 'http://foo', 'http://bar/a')
221
test('http://bar/a/', 'http://foo', 'http://bar/a/')
224
test('file:///foo', 'file:///', 'foo')
225
test('file:///bar/foo', 'file:///bar/', 'foo')
226
test('http://host/foo', 'http://host/', 'foo')
227
test('http://host/', 'http://host', '')
230
# Cannot go above root
231
# Implicitly at root:
232
self.assertRaises(InvalidURLJoin, urlutils.join,
233
'http://foo', '../baz')
234
self.assertRaises(InvalidURLJoin, urlutils.join,
236
# Joining from a path explicitly under the root.
237
self.assertRaises(InvalidURLJoin, urlutils.join,
238
'http://foo/a', '../../b')
240
def test_joinpath(self):
241
def test(expected, *args):
242
joined = urlutils.joinpath(*args)
243
self.assertEqual(expected, joined)
245
# Test a single element
248
# Test relative path joining
249
test('foo/bar', 'foo', 'bar')
250
test('foo/bar', 'foo', '.', 'bar')
251
test('foo/baz', 'foo', 'bar', '../baz')
252
test('foo/bar/baz', 'foo', 'bar/baz')
253
test('foo/baz', 'foo', 'bar/../baz')
255
# Test joining to an absolute path
257
test('/foo', '/foo', '.')
258
test('/foo/bar', '/foo', 'bar')
259
test('/', '/foo', '..')
261
# Test joining with an absolute path
262
test('/bar', 'foo', '/bar')
264
# Test joining to a path with a trailing slash
265
test('foo/bar', 'foo/', 'bar')
268
# Cannot go above root
269
self.assertRaises(InvalidURLJoin, urlutils.joinpath, '/', '../baz')
270
self.assertRaises(InvalidURLJoin, urlutils.joinpath, '/', '..')
271
self.assertRaises(InvalidURLJoin, urlutils.joinpath, '/', '/..')
273
184
def test_function_type(self):
274
185
if sys.platform == 'win32':
275
186
self.assertEqual(urlutils._win32_local_path_to_url, urlutils.local_path_to_url)
299
204
from_url('file:///path/to/r%C3%A4ksm%C3%B6rg%C3%A5s'))
300
205
self.assertEqual(u'/path/to/r\xe4ksm\xf6rg\xe5s',
301
206
from_url('file:///path/to/r%c3%a4ksm%c3%b6rg%c3%a5s'))
302
self.assertEqual(u'/path/to/r\xe4ksm\xf6rg\xe5s',
303
from_url('file://localhost/path/to/r%c3%a4ksm%c3%b6rg%c3%a5s'))
305
208
self.assertRaises(InvalidURL, from_url, '/path/to/foo')
307
InvalidURL, from_url,
308
'file://remotehost/path/to/r%c3%a4ksm%c3%b6rg%c3%a5s')
310
210
def test_win32_local_path_to_url(self):
311
211
to_url = urlutils._win32_local_path_to_url
312
self.assertEqual('file:///C:/path/to/foo',
212
self.assertEqual('file:///C|/path/to/foo',
313
213
to_url('C:/path/to/foo'))
314
# BOGUS: on win32, ntpath.abspath will strip trailing
315
# whitespace, so this will always fail
316
# Though under linux, it fakes abspath support
317
# and thus will succeed
318
# self.assertEqual('file:///C:/path/to/foo%20',
319
# to_url('C:/path/to/foo '))
320
self.assertEqual('file:///C:/path/to/f%20oo',
321
to_url('C:/path/to/f oo'))
323
self.assertEqual('file:///', to_url('/'))
326
result = to_url(u'd:/path/to/r\xe4ksm\xf6rg\xe5s')
328
raise TestSkipped("local encoding cannot handle unicode")
330
self.assertEqual('file:///D:/path/to/r%C3%A4ksm%C3%B6rg%C3%A5s', result)
331
self.assertFalse(isinstance(result, unicode))
333
def test_win32_unc_path_to_url(self):
334
to_url = urlutils._win32_local_path_to_url
335
self.assertEqual('file://HOST/path',
336
to_url(r'\\HOST\path'))
337
self.assertEqual('file://HOST/path',
338
to_url('//HOST/path'))
341
result = to_url(u'//HOST/path/to/r\xe4ksm\xf6rg\xe5s')
343
raise TestSkipped("local encoding cannot handle unicode")
345
self.assertEqual('file://HOST/path/to/r%C3%A4ksm%C3%B6rg%C3%A5s', result)
346
self.assertFalse(isinstance(result, unicode))
214
self.assertEqual('file:///D|/path/to/r%C3%A4ksm%C3%B6rg%C3%A5s',
215
to_url(u'd:/path/to/r\xe4ksm\xf6rg\xe5s'))
348
217
def test_win32_local_path_from_url(self):
349
218
from_url = urlutils._win32_local_path_from_url
352
221
self.assertEqual(u'D:/path/to/r\xe4ksm\xf6rg\xe5s',
353
222
from_url('file:///d|/path/to/r%C3%A4ksm%C3%B6rg%C3%A5s'))
354
223
self.assertEqual(u'D:/path/to/r\xe4ksm\xf6rg\xe5s',
355
from_url('file:///d:/path/to/r%c3%a4ksm%c3%b6rg%c3%a5s'))
356
self.assertEqual('/', from_url('file:///'))
224
from_url('file:///d|/path/to/r%c3%a4ksm%c3%b6rg%c3%a5s'))
358
226
self.assertRaises(InvalidURL, from_url, '/path/to/foo')
359
227
# Not a valid _win32 url, no drive letter
360
228
self.assertRaises(InvalidURL, from_url, 'file:///path/to/foo')
362
def test_win32_unc_path_from_url(self):
363
from_url = urlutils._win32_local_path_from_url
364
self.assertEqual('//HOST/path', from_url('file://HOST/path'))
365
# despite IE allows 2, 4, 5 and 6 slashes in URL to another machine
366
# we want to use only 2 slashes
367
# Firefox understand only 5 slashes in URL, but it's ugly
368
self.assertRaises(InvalidURL, from_url, 'file:////HOST/path')
369
self.assertRaises(InvalidURL, from_url, 'file://///HOST/path')
370
self.assertRaises(InvalidURL, from_url, 'file://////HOST/path')
371
# check for file://C:/ instead of file:///C:/
372
self.assertRaises(InvalidURL, from_url, 'file://C:/path')
374
def test_win32_extract_drive_letter(self):
375
extract = urlutils._win32_extract_drive_letter
376
self.assertEqual(('file:///C:', '/foo'), extract('file://', '/C:/foo'))
377
self.assertEqual(('file:///d|', '/path'), extract('file://', '/d|/path'))
378
self.assertRaises(InvalidURL, extract, 'file://', '/path')
380
230
def test_split(self):
381
231
# Test bzrlib.urlutils.split()
382
232
split = urlutils.split
383
233
if sys.platform == 'win32':
384
234
self.assertRaises(InvalidURL, split, 'file:///path/to/foo')
385
235
self.assertEqual(('file:///C|/', 'foo'), split('file:///C|/foo'))
386
self.assertEqual(('file:///C:/', ''), split('file:///C:/'))
236
self.assertEqual(('file:///C|/', ''), split('file:///C|/'))
388
238
self.assertEqual(('file:///', 'foo'), split('file:///foo'))
389
239
self.assertEqual(('file:///', ''), split('file:///'))
455
295
self.assertEqual('../to/foo', sts('../to/foo/'))
456
296
self.assertEqual('path/../foo', sts('path/../foo/'))
458
def test_unescape_for_display_utf8(self):
298
def test_unescape_for_display(self):
459
299
# Test that URLs are converted to nice unicode strings for display
460
def test(expected, url, encoding='utf-8'):
461
disp_url = urlutils.unescape_for_display(url, encoding=encoding)
462
self.assertIsInstance(disp_url, unicode)
463
self.assertEqual(expected, disp_url)
465
test('http://foo', 'http://foo')
300
disp = urlutils.unescape_for_display
301
eq = self.assertEqual
302
eq('http://foo', disp('http://foo'))
466
303
if sys.platform == 'win32':
467
test('C:/foo/path', 'file:///C|/foo/path')
468
test('C:/foo/path', 'file:///C:/foo/path')
304
eq('C:/foo/path', disp('file:///C|foo/path'))
470
test('/foo/path', 'file:///foo/path')
306
eq('/foo/path', disp('file:///foo/path'))
472
test('http://foo/%2Fbaz', 'http://foo/%2Fbaz')
473
test(u'http://host/r\xe4ksm\xf6rg\xe5s',
474
'http://host/r%C3%A4ksm%C3%B6rg%C3%A5s')
308
eq('http://foo/%2Fbaz', disp('http://foo/%2Fbaz'))
309
eq(u'http://host/r\xe4ksm\xf6rg\xe5s', disp('http://host/r%C3%A4ksm%C3%B6rg%C3%A5s'))
476
311
# Make sure special escaped characters stay escaped
477
test(u'http://host/%3B%2F%3F%3A%40%26%3D%2B%24%2C%23',
478
'http://host/%3B%2F%3F%3A%40%26%3D%2B%24%2C%23')
312
eq(u'http://host/%3B%2F%3F%3A%40%26%3D%2B%24%2C%23',
313
disp('http://host/%3B%2F%3F%3A%40%26%3D%2B%24%2C%23'))
480
315
# Can we handle sections that don't have utf-8 encoding?
481
test(u'http://host/%EE%EE%EE/r\xe4ksm\xf6rg\xe5s',
482
'http://host/%EE%EE%EE/r%C3%A4ksm%C3%B6rg%C3%A5s')
484
# Test encoding into output that can handle some characters
485
test(u'http://host/%EE%EE%EE/r\xe4ksm\xf6rg\xe5s',
486
'http://host/%EE%EE%EE/r%C3%A4ksm%C3%B6rg%C3%A5s',
487
encoding='iso-8859-1')
489
# This one can be encoded into utf8
490
test(u'http://host/\u062c\u0648\u062c\u0648',
491
'http://host/%d8%ac%d9%88%d8%ac%d9%88',
494
# This can't be put into 8859-1 and so stays as escapes
495
test(u'http://host/%d8%ac%d9%88%d8%ac%d9%88',
496
'http://host/%d8%ac%d9%88%d8%ac%d9%88',
497
encoding='iso-8859-1')
316
eq(u'http://host/%EE%EE%EE/r\xe4ksm\xf6rg\xe5s',
317
disp('http://host/%EE%EE%EE/r%C3%A4ksm%C3%B6rg%C3%A5s'))
499
319
def test_escape(self):
500
320
self.assertEqual('%25', urlutils.escape('%'))
501
321
self.assertEqual('%C3%A5', urlutils.escape(u'\xe5'))
502
self.assertFalse(isinstance(urlutils.escape(u'\xe5'), unicode))
504
def test_escape_tildes(self):
505
self.assertEqual('~foo', urlutils.escape('~foo'))
507
323
def test_unescape(self):
508
324
self.assertEqual('%', urlutils.unescape('%25'))
516
332
self.assertEqual(u'\xe5', urlutils.unescape(urlutils.escape(u'\xe5')))
517
333
self.assertEqual('%', urlutils.unescape(urlutils.escape('%')))
519
def test_relative_url(self):
520
def test(expected, base, other):
521
result = urlutils.relative_url(base, other)
522
self.assertEqual(expected, result)
524
test('a', 'http://host/', 'http://host/a')
525
test('http://entirely/different', 'sftp://host/branch',
526
'http://entirely/different')
527
test('../person/feature', 'http://host/branch/mainline',
528
'http://host/branch/person/feature')
529
test('..', 'http://host/branch', 'http://host/')
530
test('http://host2/branch', 'http://host1/branch', 'http://host2/branch')
531
test('.', 'http://host1/branch', 'http://host1/branch')
532
test('../../../branch/2b', 'file:///home/jelmer/foo/bar/2b',
533
'file:///home/jelmer/branch/2b')
534
test('../../branch/2b', 'sftp://host/home/jelmer/bar/2b',
535
'sftp://host/home/jelmer/branch/2b')
536
test('../../branch/feature/%2b', 'http://host/home/jelmer/bar/%2b',
537
'http://host/home/jelmer/branch/feature/%2b')
538
test('../../branch/feature/2b', 'http://host/home/jelmer/bar/2b/',
539
'http://host/home/jelmer/branch/feature/2b')
540
# relative_url should preserve a trailing slash
541
test('../../branch/feature/2b/', 'http://host/home/jelmer/bar/2b/',
542
'http://host/home/jelmer/branch/feature/2b/')
543
test('../../branch/feature/2b/', 'http://host/home/jelmer/bar/2b',
544
'http://host/home/jelmer/branch/feature/2b/')
546
# TODO: treat http://host as http://host/
547
# relative_url is typically called from a branch.base or
548
# transport.base which always ends with a /
549
#test('a', 'http://host', 'http://host/a')
550
test('http://host/a', 'http://host', 'http://host/a')
551
#test('.', 'http://host', 'http://host/')
552
test('http://host/', 'http://host', 'http://host/')
553
#test('.', 'http://host/', 'http://host')
554
test('http://host', 'http://host/', 'http://host')
556
# On Windows file:///C:/path/to and file:///D:/other/path
557
# should not use relative url over the non-existent '/' directory.
558
if sys.platform == 'win32':
560
test('../../other/path',
561
'file:///C:/path/to', 'file:///C:/other/path')
562
#~next two tests is failed, i.e. urlutils.relative_url expects
563
#~to see normalized file URLs?
564
#~test('../../other/path',
565
#~ 'file:///C:/path/to', 'file:///c:/other/path')
566
#~test('../../other/path',
567
#~ 'file:///C:/path/to', 'file:///C|/other/path')
569
# check UNC paths too
570
test('../../other/path',
571
'file://HOST/base/path/to', 'file://HOST/base/other/path')
572
# on different drives
573
test('file:///D:/other/path',
574
'file:///C:/path/to', 'file:///D:/other/path')
575
# TODO: strictly saying in UNC path //HOST/base is full analog
576
# of drive letter for hard disk, and this situation is also
577
# should be exception from rules. [bialix 20071221]
580
class TestCwdToURL(TestCaseInTempDir):
581
"""Test that local_path_to_url works base on the cwd"""
584
# This test will fail if getcwd is not ascii
588
url = urlutils.local_path_to_url('.')
589
self.assertEndsWith(url, '/mytest')
591
def test_non_ascii(self):
592
if win32utils.winver == 'Windows 98':
593
raise TestSkipped('Windows 98 cannot handle unicode filenames')
598
raise TestSkipped('cannot create unicode directory')
602
# On Mac OSX this directory is actually:
603
# u'/dode\u0301' => '/dode\xcc\x81
604
# but we should normalize it back to
605
# u'/dod\xe9' => '/dod\xc3\xa9'
606
url = urlutils.local_path_to_url('.')
607
self.assertEndsWith(url, '/dod%C3%A9')
610
class TestDeriveToLocation(TestCase):
611
"""Test that the mapping of FROM_LOCATION to TO_LOCATION works."""
613
def test_to_locations_derived_from_paths(self):
614
derive = urlutils.derive_to_location
615
self.assertEqual("bar", derive("bar"))
616
self.assertEqual("bar", derive("../bar"))
617
self.assertEqual("bar", derive("/foo/bar"))
618
self.assertEqual("bar", derive("c:/foo/bar"))
619
self.assertEqual("bar", derive("c:bar"))
621
def test_to_locations_derived_from_urls(self):
622
derive = urlutils.derive_to_location
623
self.assertEqual("bar", derive("http://foo/bar"))
624
self.assertEqual("bar", derive("bzr+ssh://foo/bar"))
625
self.assertEqual("foo-bar", derive("lp:foo-bar"))
628
class TestRebaseURL(TestCase):
629
"""Test the behavior of rebase_url."""
631
def test_non_relative(self):
632
result = urlutils.rebase_url('file://foo', 'file://foo',
634
self.assertEqual('file://foo', result)
635
result = urlutils.rebase_url('/foo', 'file://foo',
637
self.assertEqual('/foo', result)
639
def test_different_ports(self):
640
e = self.assertRaises(InvalidRebaseURLs, urlutils.rebase_url,
641
'foo', 'http://bar:80', 'http://bar:81')
642
self.assertEqual(str(e), "URLs differ by more than path:"
643
" 'http://bar:80' and 'http://bar:81'")
645
def test_different_hosts(self):
646
e = self.assertRaises(InvalidRebaseURLs, urlutils.rebase_url,
647
'foo', 'http://bar', 'http://baz')
648
self.assertEqual(str(e), "URLs differ by more than path: 'http://bar'"
651
def test_different_protocol(self):
652
e = self.assertRaises(InvalidRebaseURLs, urlutils.rebase_url,
653
'foo', 'http://bar', 'ftp://bar')
654
self.assertEqual(str(e), "URLs differ by more than path: 'http://bar'"
657
def test_rebase_success(self):
658
self.assertEqual('../bar', urlutils.rebase_url('bar', 'http://baz/',
660
self.assertEqual('qux/bar', urlutils.rebase_url('bar',
661
'http://baz/qux', 'http://baz/'))
662
self.assertEqual('.', urlutils.rebase_url('foo',
663
'http://bar/', 'http://bar/foo/'))
664
self.assertEqual('qux/bar', urlutils.rebase_url('../bar',
665
'http://baz/qux/foo', 'http://baz/'))
667
def test_determine_relative_path(self):
668
self.assertEqual('../../baz/bar',
669
urlutils.determine_relative_path(
670
'/qux/quxx', '/baz/bar'))
671
self.assertEqual('..',
672
urlutils.determine_relative_path(
674
self.assertEqual('baz',
675
urlutils.determine_relative_path(
677
self.assertEqual('.', urlutils.determine_relative_path(
681
class TestParseURL(TestCase):
683
def test_parse_url(self):
684
self.assertEqual(urlutils.parse_url('http://example.com:80/one'),
685
('http', None, None, 'example.com', 80, '/one'))
686
self.assertEqual(urlutils.parse_url('http://[1:2:3::40]/one'),
687
('http', None, None, '1:2:3::40', None, '/one'))
688
self.assertEqual(urlutils.parse_url('http://[1:2:3::40]:80/one'),
689
('http', None, None, '1:2:3::40', 80, '/one'))