mirror of https://github.com/python/cpython.git
SF bug #456621: normpath on Win32 not collapsing c:\\..
I actually rewrote normpath quite a bit: it had no test cases, and as soon as I starting writing some I found several cases that didn't make sense.
This commit is contained in:
parent
0e03f588f5
commit
54a14a373e
Lib
|
@ -407,7 +407,7 @@ def expandvars(path):
|
|||
return res
|
||||
|
||||
|
||||
# Normalize a path, e.g. A//B, A/./B and A/foo/../B all become A/B.
|
||||
# Normalize a path, e.g. A//B, A/./B and A/foo/../B all become A\B.
|
||||
# Previously, this function also truncated pathnames to 8+3 format,
|
||||
# but as this module is called "ntpath", that's obviously wrong!
|
||||
|
||||
|
@ -421,15 +421,18 @@ def normpath(path):
|
|||
comps = path.split("\\")
|
||||
i = 0
|
||||
while i < len(comps):
|
||||
if comps[i] == '.':
|
||||
del comps[i]
|
||||
elif comps[i] == '..' and i > 0 and comps[i-1] not in ('', '..'):
|
||||
del comps[i-1:i+1]
|
||||
i = i - 1
|
||||
elif comps[i] == '' and i > 0 and comps[i-1] != '':
|
||||
if comps[i] in ('.', ''):
|
||||
del comps[i]
|
||||
elif comps[i] == '..':
|
||||
if i > 0 and comps[i-1] != '..':
|
||||
del comps[i-1:i+1]
|
||||
i -= 1
|
||||
elif i == 0 and prefix.endswith("\\"):
|
||||
del comps[i]
|
||||
else:
|
||||
i += 1
|
||||
else:
|
||||
i = i + 1
|
||||
i += 1
|
||||
# If the path is now empty, substitute '.'
|
||||
if not prefix and not comps:
|
||||
comps.append('.')
|
||||
|
|
|
@ -74,6 +74,30 @@ def tester(fn, wantResult):
|
|||
tester("ntpath.join('c:/', 'd:/')", 'd:/')
|
||||
tester("ntpath.join('c:/', 'd:/a/b')", 'd:/a/b')
|
||||
|
||||
tester("ntpath.normpath('A//////././//.//B')", r'A\B')
|
||||
tester("ntpath.normpath('A/./B')", r'A\B')
|
||||
tester("ntpath.normpath('A/foo/../B')", r'A\B')
|
||||
tester("ntpath.normpath('C:A//B')", r'C:A\B')
|
||||
tester("ntpath.normpath('D:A/./B')", r'D:A\B')
|
||||
tester("ntpath.normpath('e:A/foo/../B')", r'e:A\B')
|
||||
|
||||
# Next 3 seem dubious, and especially the 3rd, but normpath is possibly
|
||||
# trying to leave UNC paths alone without actually knowing anything about
|
||||
# them.
|
||||
tester("ntpath.normpath('C:///A//B')", r'C:\\\A\B')
|
||||
tester("ntpath.normpath('D:///A/./B')", r'D:\\\A\B')
|
||||
tester("ntpath.normpath('e:///A/foo/../B')", r'e:\\\A\B')
|
||||
|
||||
tester("ntpath.normpath('..')", r'..')
|
||||
tester("ntpath.normpath('.')", r'.')
|
||||
tester("ntpath.normpath('')", r'.')
|
||||
tester("ntpath.normpath('/')", '\\')
|
||||
tester("ntpath.normpath('c:/')", 'c:\\')
|
||||
tester("ntpath.normpath('/../.././..')", '\\')
|
||||
tester("ntpath.normpath('c:/../../..')", 'c:\\')
|
||||
tester("ntpath.normpath('../.././..')", r'..\..\..')
|
||||
tester("ntpath.normpath('K:../.././..')", r'K:..\..\..')
|
||||
|
||||
if errors:
|
||||
raise TestFailed(str(errors) + " errors.")
|
||||
elif verbose:
|
||||
|
|
Loading…
Reference in New Issue