123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211 |
- """Pathname and path-related operations for the Macintosh."""
- import os
- from stat import *
- import genericpath
- from genericpath import *
- __all__ = ["normcase","isabs","join","splitdrive","split","splitext",
- "basename","dirname","commonprefix","getsize","getmtime",
- "getatime","getctime", "islink","exists","lexists","isdir","isfile",
- "expanduser","expandvars","normpath","abspath",
- "curdir","pardir","sep","pathsep","defpath","altsep","extsep",
- "devnull","realpath","supports_unicode_filenames"]
- # strings representing various path-related bits and pieces
- # These are primarily for export; internally, they are hardcoded.
- curdir = ':'
- pardir = '::'
- extsep = '.'
- sep = ':'
- pathsep = '\n'
- defpath = ':'
- altsep = None
- devnull = 'Dev:Null'
- def _get_colon(path):
- if isinstance(path, bytes):
- return b':'
- else:
- return ':'
- # Normalize the case of a pathname. Dummy in Posix, but <s>.lower() here.
- def normcase(path):
- if not isinstance(path, (bytes, str)):
- raise TypeError("normcase() argument must be str or bytes, "
- "not '{}'".format(path.__class__.__name__))
- return path.lower()
- def isabs(s):
- """Return true if a path is absolute.
- On the Mac, relative paths begin with a colon,
- but as a special case, paths with no colons at all are also relative.
- Anything else is absolute (the string up to the first colon is the
- volume name)."""
- colon = _get_colon(s)
- return colon in s and s[:1] != colon
- def join(s, *p):
- try:
- colon = _get_colon(s)
- path = s
- if not p:
- path[:0] + colon #23780: Ensure compatible data type even if p is null.
- for t in p:
- if (not path) or isabs(t):
- path = t
- continue
- if t[:1] == colon:
- t = t[1:]
- if colon not in path:
- path = colon + path
- if path[-1:] != colon:
- path = path + colon
- path = path + t
- return path
- except (TypeError, AttributeError, BytesWarning):
- genericpath._check_arg_types('join', s, *p)
- raise
- def split(s):
- """Split a pathname into two parts: the directory leading up to the final
- bit, and the basename (the filename, without colons, in that directory).
- The result (s, t) is such that join(s, t) yields the original argument."""
- colon = _get_colon(s)
- if colon not in s: return s[:0], s
- col = 0
- for i in range(len(s)):
- if s[i:i+1] == colon: col = i + 1
- path, file = s[:col-1], s[col:]
- if path and not colon in path:
- path = path + colon
- return path, file
- def splitext(p):
- if isinstance(p, bytes):
- return genericpath._splitext(p, b':', altsep, b'.')
- else:
- return genericpath._splitext(p, sep, altsep, extsep)
- splitext.__doc__ = genericpath._splitext.__doc__
- def splitdrive(p):
- """Split a pathname into a drive specification and the rest of the
- path. Useful on DOS/Windows/NT; on the Mac, the drive is always
- empty (don't use the volume name -- it doesn't have the same
- syntactic and semantic oddities as DOS drive letters, such as there
- being a separate current directory per drive)."""
- return p[:0], p
- # Short interfaces to split()
- def dirname(s): return split(s)[0]
- def basename(s): return split(s)[1]
- def ismount(s):
- if not isabs(s):
- return False
- components = split(s)
- return len(components) == 2 and not components[1]
- def islink(s):
- """Return true if the pathname refers to a symbolic link."""
- try:
- import Carbon.File
- return Carbon.File.ResolveAliasFile(s, 0)[2]
- except:
- return False
- # Is `stat`/`lstat` a meaningful difference on the Mac? This is safe in any
- # case.
- def lexists(path):
- """Test whether a path exists. Returns True for broken symbolic links"""
- try:
- st = os.lstat(path)
- except OSError:
- return False
- return True
- def expandvars(path):
- """Dummy to retain interface-compatibility with other operating systems."""
- return path
- def expanduser(path):
- """Dummy to retain interface-compatibility with other operating systems."""
- return path
- class norm_error(Exception):
- """Path cannot be normalized"""
- def normpath(s):
- """Normalize a pathname. Will return the same result for
- equivalent paths."""
- colon = _get_colon(s)
- if colon not in s:
- return colon + s
- comps = s.split(colon)
- i = 1
- while i < len(comps)-1:
- if not comps[i] and comps[i-1]:
- if i > 1:
- del comps[i-1:i+1]
- i = i - 1
- else:
- # best way to handle this is to raise an exception
- raise norm_error('Cannot use :: immediately after volume name')
- else:
- i = i + 1
- s = colon.join(comps)
- # remove trailing ":" except for ":" and "Volume:"
- if s[-1:] == colon and len(comps) > 2 and s != colon*len(s):
- s = s[:-1]
- return s
- def abspath(path):
- """Return an absolute path."""
- if not isabs(path):
- if isinstance(path, bytes):
- cwd = os.getcwdb()
- else:
- cwd = os.getcwd()
- path = join(cwd, path)
- return normpath(path)
- # realpath is a no-op on systems without islink support
- def realpath(path):
- path = abspath(path)
- try:
- import Carbon.File
- except ImportError:
- return path
- if not path:
- return path
- colon = _get_colon(path)
- components = path.split(colon)
- path = components[0] + colon
- for c in components[1:]:
- path = join(path, c)
- try:
- path = Carbon.File.FSResolveAliasFile(path, 1)[0].as_pathname()
- except Carbon.File.Error:
- pass
- return path
- supports_unicode_filenames = True
|