2000-02-04 11:10:34 -04:00
|
|
|
"""Pathname and path-related operations for the Macintosh."""
|
1990-12-26 11:40:07 -04:00
|
|
|
|
1998-03-03 17:49:01 -04:00
|
|
|
import os
|
2008-05-08 21:27:01 -03:00
|
|
|
import warnings
|
1990-12-26 11:40:07 -04:00
|
|
|
from stat import *
|
2007-03-07 07:04:33 -04:00
|
|
|
import genericpath
|
2006-08-26 15:42:06 -03:00
|
|
|
from genericpath import *
|
2015-02-13 06:02:05 -04:00
|
|
|
from genericpath import _unicode
|
1990-12-26 11:40:07 -04:00
|
|
|
|
2001-01-24 02:27:27 -04:00
|
|
|
__all__ = ["normcase","isabs","join","splitdrive","split","splitext",
|
|
|
|
"basename","dirname","commonprefix","getsize","getmtime",
|
2005-08-22 15:02:59 -03:00
|
|
|
"getatime","getctime", "islink","exists","lexists","isdir","isfile",
|
2002-10-07 23:44:31 -03:00
|
|
|
"walk","expanduser","expandvars","normpath","abspath",
|
2003-02-14 15:35:31 -04:00
|
|
|
"curdir","pardir","sep","pathsep","defpath","altsep","extsep",
|
2004-06-08 05:29:33 -03:00
|
|
|
"devnull","realpath","supports_unicode_filenames"]
|
1991-01-01 14:10:40 -04:00
|
|
|
|
2003-02-14 15:35:31 -04:00
|
|
|
# strings representing various path-related bits and pieces
|
|
|
|
curdir = ':'
|
|
|
|
pardir = '::'
|
|
|
|
extsep = '.'
|
|
|
|
sep = ':'
|
|
|
|
pathsep = '\n'
|
|
|
|
defpath = ':'
|
|
|
|
altsep = None
|
2004-06-08 05:29:33 -03:00
|
|
|
devnull = 'Dev:Null'
|
2003-02-14 15:35:31 -04:00
|
|
|
|
2000-09-28 13:25:20 -03:00
|
|
|
# Normalize the case of a pathname. Dummy in Posix, but <s>.lower() here.
|
1992-01-14 14:28:18 -04:00
|
|
|
|
2000-09-28 13:25:20 -03:00
|
|
|
def normcase(path):
|
|
|
|
return path.lower()
|
1992-01-14 14:28:18 -04:00
|
|
|
|
|
|
|
|
1990-12-26 11:40:07 -04:00
|
|
|
def isabs(s):
|
2000-02-04 11:10:34 -04:00
|
|
|
"""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)."""
|
|
|
|
|
2000-12-12 19:20:45 -04:00
|
|
|
return ':' in s and s[0] != ':'
|
1990-12-26 11:40:07 -04:00
|
|
|
|
1991-01-01 14:10:40 -04:00
|
|
|
|
1997-02-18 17:53:25 -04:00
|
|
|
def join(s, *p):
|
2000-02-04 11:10:34 -04:00
|
|
|
path = s
|
|
|
|
for t in p:
|
2014-09-27 12:53:01 -03:00
|
|
|
if (not path) or isabs(t):
|
2000-02-04 11:10:34 -04:00
|
|
|
path = t
|
|
|
|
continue
|
|
|
|
if t[:1] == ':':
|
|
|
|
t = t[1:]
|
|
|
|
if ':' not in path:
|
|
|
|
path = ':' + path
|
2000-12-12 19:20:45 -04:00
|
|
|
if path[-1:] != ':':
|
2000-02-04 11:10:34 -04:00
|
|
|
path = path + ':'
|
|
|
|
path = path + t
|
|
|
|
return path
|
|
|
|
|
1991-01-01 14:10:40 -04:00
|
|
|
|
|
|
|
def split(s):
|
2000-02-04 11:10:34 -04:00
|
|
|
"""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."""
|
1991-01-01 14:10:40 -04:00
|
|
|
|
2000-02-04 11:10:34 -04:00
|
|
|
if ':' not in s: return '', s
|
|
|
|
colon = 0
|
|
|
|
for i in range(len(s)):
|
2000-09-28 13:25:20 -03:00
|
|
|
if s[i] == ':': colon = i + 1
|
2000-02-04 11:10:34 -04:00
|
|
|
path, file = s[:colon-1], s[colon:]
|
|
|
|
if path and not ':' in path:
|
|
|
|
path = path + ':'
|
|
|
|
return path, file
|
1991-01-01 14:10:40 -04:00
|
|
|
|
1996-07-22 23:28:32 -03:00
|
|
|
|
|
|
|
def splitext(p):
|
2007-03-07 07:04:33 -04:00
|
|
|
return genericpath._splitext(p, sep, altsep, extsep)
|
|
|
|
splitext.__doc__ = genericpath._splitext.__doc__
|
1992-11-05 06:43:02 -04:00
|
|
|
|
1995-08-10 15:09:16 -03:00
|
|
|
def splitdrive(p):
|
2000-02-04 11:10:34 -04:00
|
|
|
"""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
|
1992-11-05 06:43:02 -04:00
|
|
|
|
|
|
|
|
1995-08-10 15:09:16 -03:00
|
|
|
# Short interfaces to split()
|
1990-12-26 11:40:07 -04:00
|
|
|
|
1995-08-10 15:09:16 -03:00
|
|
|
def dirname(s): return split(s)[0]
|
|
|
|
def basename(s): return split(s)[1]
|
1990-12-26 11:40:07 -04:00
|
|
|
|
2003-01-15 18:45:48 -04:00
|
|
|
def ismount(s):
|
2003-01-28 23:49:43 -04:00
|
|
|
if not isabs(s):
|
|
|
|
return False
|
|
|
|
components = split(s)
|
|
|
|
return len(components) == 2 and components[1] == ''
|
1991-01-01 14:10:40 -04:00
|
|
|
|
1995-01-26 22:41:45 -04:00
|
|
|
def islink(s):
|
2002-04-22 10:55:43 -03:00
|
|
|
"""Return true if the pathname refers to a symbolic link."""
|
1995-01-26 22:41:45 -04:00
|
|
|
|
2002-04-22 10:55:43 -03:00
|
|
|
try:
|
2003-02-27 19:18:46 -04:00
|
|
|
import Carbon.File
|
|
|
|
return Carbon.File.ResolveAliasFile(s, 0)[2]
|
2002-04-22 10:55:43 -03:00
|
|
|
except:
|
|
|
|
return False
|
1995-01-26 22:41:45 -04:00
|
|
|
|
2004-08-30 07:19:56 -03:00
|
|
|
# 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 os.error:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
1995-12-15 09:23:37 -04:00
|
|
|
def expandvars(path):
|
2000-02-04 11:10:34 -04:00
|
|
|
"""Dummy to retain interface-compatibility with other operating systems."""
|
|
|
|
return path
|
1995-12-15 09:23:37 -04:00
|
|
|
|
1992-11-05 06:43:02 -04:00
|
|
|
|
2000-02-04 11:10:34 -04:00
|
|
|
def expanduser(path):
|
|
|
|
"""Dummy to retain interface-compatibility with other operating systems."""
|
|
|
|
return path
|
1995-08-10 15:09:16 -03:00
|
|
|
|
2002-03-31 10:06:41 -04:00
|
|
|
class norm_error(Exception):
|
|
|
|
"""Path cannot be normalized"""
|
1992-11-05 06:43:02 -04:00
|
|
|
|
|
|
|
def normpath(s):
|
2000-08-06 18:18:35 -03:00
|
|
|
"""Normalize a pathname. Will return the same result for
|
|
|
|
equivalent paths."""
|
|
|
|
|
|
|
|
if ":" not in s:
|
|
|
|
return ":"+s
|
|
|
|
|
2000-09-28 13:25:20 -03:00
|
|
|
comps = s.split(":")
|
2000-08-06 18:18:35 -03:00
|
|
|
i = 1
|
|
|
|
while i < len(comps)-1:
|
|
|
|
if comps[i] == "" and comps[i-1] != "":
|
|
|
|
if i > 1:
|
|
|
|
del comps[i-1:i+1]
|
2000-09-28 13:25:20 -03:00
|
|
|
i = i - 1
|
2000-08-06 18:18:35 -03:00
|
|
|
else:
|
|
|
|
# best way to handle this is to raise an exception
|
2001-08-08 17:55:10 -03:00
|
|
|
raise norm_error, 'Cannot use :: immediately after volume name'
|
2000-02-04 11:10:34 -04:00
|
|
|
else:
|
2000-08-06 18:18:35 -03:00
|
|
|
i = i + 1
|
|
|
|
|
2000-09-28 13:25:20 -03:00
|
|
|
s = ":".join(comps)
|
2000-08-06 18:18:35 -03:00
|
|
|
|
|
|
|
# remove trailing ":" except for ":" and "Volume:"
|
|
|
|
if s[-1] == ":" and len(comps) > 2 and s != ":"*len(s):
|
|
|
|
s = s[:-1]
|
2000-02-04 11:10:34 -04:00
|
|
|
return s
|
|
|
|
|
1995-08-07 11:09:27 -03:00
|
|
|
|
|
|
|
def walk(top, func, arg):
|
2001-10-10 01:16:20 -03:00
|
|
|
"""Directory tree walk with callback function.
|
|
|
|
|
|
|
|
For each directory in the directory tree rooted at top (including top
|
|
|
|
itself, but excluding '.' and '..'), call func(arg, dirname, fnames).
|
|
|
|
dirname is the name of the directory, and fnames a list of the names of
|
|
|
|
the files and subdirectories in dirname (excluding '.' and '..'). func
|
|
|
|
may modify the fnames list in-place (e.g. via del or slice assignment),
|
|
|
|
and walk will only recurse into the subdirectories whose names remain in
|
|
|
|
fnames; this can be used to implement a filter, or to impose a specific
|
|
|
|
order of visiting. No semantics are defined for, or required of, arg,
|
|
|
|
beyond that arg is always passed to func. It can be used, e.g., to pass
|
|
|
|
a filename pattern, or a mutable object designed to accumulate
|
|
|
|
statistics. Passing None for arg is common."""
|
2009-05-07 23:28:39 -03:00
|
|
|
warnings.warnpy3k("In 3.x, os.path.walk is removed in favor of os.walk.",
|
|
|
|
stacklevel=2)
|
2000-02-04 11:10:34 -04:00
|
|
|
try:
|
|
|
|
names = os.listdir(top)
|
|
|
|
except os.error:
|
|
|
|
return
|
|
|
|
func(arg, top, names)
|
|
|
|
for name in names:
|
|
|
|
name = join(top, name)
|
2002-04-22 10:55:43 -03:00
|
|
|
if isdir(name) and not islink(name):
|
2000-02-04 11:10:34 -04:00
|
|
|
walk(name, func, arg)
|
|
|
|
|
|
|
|
|
1999-01-29 14:05:18 -04:00
|
|
|
def abspath(path):
|
2000-02-04 11:10:34 -04:00
|
|
|
"""Return an absolute path."""
|
1999-01-29 14:05:18 -04:00
|
|
|
if not isabs(path):
|
2015-02-13 06:02:05 -04:00
|
|
|
if isinstance(path, _unicode):
|
2010-02-20 04:09:39 -04:00
|
|
|
cwd = os.getcwdu()
|
|
|
|
else:
|
|
|
|
cwd = os.getcwd()
|
|
|
|
path = join(cwd, path)
|
1999-01-29 14:05:18 -04:00
|
|
|
return normpath(path)
|
2001-09-17 12:16:09 -03:00
|
|
|
|
|
|
|
# realpath is a no-op on systems without islink support
|
2002-04-22 10:55:43 -03:00
|
|
|
def realpath(path):
|
2002-05-23 12:15:30 -03:00
|
|
|
path = abspath(path)
|
|
|
|
try:
|
2003-02-27 19:18:46 -04:00
|
|
|
import Carbon.File
|
2002-05-23 12:15:30 -03:00
|
|
|
except ImportError:
|
|
|
|
return path
|
|
|
|
if not path:
|
|
|
|
return path
|
|
|
|
components = path.split(':')
|
|
|
|
path = components[0] + ':'
|
|
|
|
for c in components[1:]:
|
|
|
|
path = join(path, c)
|
2010-03-21 09:27:20 -03:00
|
|
|
try:
|
|
|
|
path = Carbon.File.FSResolveAliasFile(path, 1)[0].as_pathname()
|
|
|
|
except Carbon.File.Error:
|
|
|
|
pass
|
2002-05-23 12:15:30 -03:00
|
|
|
return path
|
2002-10-07 23:44:31 -03:00
|
|
|
|
2010-09-13 17:31:34 -03:00
|
|
|
supports_unicode_filenames = True
|