| 1 | r"""OS routines for Mac, NT, or Posix depending on what system we're on.
|
|---|
| 2 |
|
|---|
| 3 | This exports:
|
|---|
| 4 | - all functions from posix, nt, os2, mac, or ce, e.g. unlink, stat, etc.
|
|---|
| 5 | - os.path is one of the modules posixpath, ntpath, or macpath
|
|---|
| 6 | - os.name is 'posix', 'nt', 'os2', 'mac', 'ce' or 'riscos'
|
|---|
| 7 | - os.curdir is a string representing the current directory ('.' or ':')
|
|---|
| 8 | - os.pardir is a string representing the parent directory ('..' or '::')
|
|---|
| 9 | - os.sep is the (or a most common) pathname separator ('/' or ':' or '\\')
|
|---|
| 10 | - os.extsep is the extension separator ('.' or '/')
|
|---|
| 11 | - os.altsep is the alternate pathname separator (None or '/')
|
|---|
| 12 | - os.pathsep is the component separator used in $PATH etc
|
|---|
| 13 | - os.linesep is the line separator in text files ('\r' or '\n' or '\r\n')
|
|---|
| 14 | - os.defpath is the default search path for executables
|
|---|
| 15 | - os.devnull is the file path of the null device ('/dev/null', etc.)
|
|---|
| 16 |
|
|---|
| 17 | Programs that import and use 'os' stand a better chance of being
|
|---|
| 18 | portable between different platforms. Of course, they must then
|
|---|
| 19 | only use functions that are defined by all platforms (e.g., unlink
|
|---|
| 20 | and opendir), and leave all pathname manipulation to os.path
|
|---|
| 21 | (e.g., split and join).
|
|---|
| 22 | """
|
|---|
| 23 |
|
|---|
| 24 | #'
|
|---|
| 25 |
|
|---|
| 26 | import sys
|
|---|
| 27 |
|
|---|
| 28 | _names = sys.builtin_module_names
|
|---|
| 29 |
|
|---|
| 30 | # Note: more names are added to __all__ later.
|
|---|
| 31 | __all__ = ["altsep", "curdir", "pardir", "sep", "pathsep", "linesep",
|
|---|
| 32 | "defpath", "name", "path", "devnull",
|
|---|
| 33 | "SEEK_SET", "SEEK_CUR", "SEEK_END"]
|
|---|
| 34 |
|
|---|
| 35 | def _get_exports_list(module):
|
|---|
| 36 | try:
|
|---|
| 37 | return list(module.__all__)
|
|---|
| 38 | except AttributeError:
|
|---|
| 39 | return [n for n in dir(module) if n[0] != '_']
|
|---|
| 40 |
|
|---|
| 41 | if 'posix' in _names:
|
|---|
| 42 | name = 'posix'
|
|---|
| 43 | linesep = '\n'
|
|---|
| 44 | from posix import *
|
|---|
| 45 | try:
|
|---|
| 46 | from posix import _exit
|
|---|
| 47 | except ImportError:
|
|---|
| 48 | pass
|
|---|
| 49 | import posixpath as path
|
|---|
| 50 |
|
|---|
| 51 | import posix
|
|---|
| 52 | __all__.extend(_get_exports_list(posix))
|
|---|
| 53 | del posix
|
|---|
| 54 |
|
|---|
| 55 | elif 'nt' in _names:
|
|---|
| 56 | name = 'nt'
|
|---|
| 57 | linesep = '\r\n'
|
|---|
| 58 | from nt import *
|
|---|
| 59 | try:
|
|---|
| 60 | from nt import _exit
|
|---|
| 61 | except ImportError:
|
|---|
| 62 | pass
|
|---|
| 63 | import ntpath as path
|
|---|
| 64 |
|
|---|
| 65 | import nt
|
|---|
| 66 | __all__.extend(_get_exports_list(nt))
|
|---|
| 67 | del nt
|
|---|
| 68 |
|
|---|
| 69 | elif 'os2' in _names:
|
|---|
| 70 | name = 'os2'
|
|---|
| 71 | linesep = '\r\n'
|
|---|
| 72 | from os2 import *
|
|---|
| 73 | try:
|
|---|
| 74 | from os2 import _exit
|
|---|
| 75 | except ImportError:
|
|---|
| 76 | pass
|
|---|
| 77 | if sys.version.find('EMX GCC') == -1:
|
|---|
| 78 | import ntpath as path
|
|---|
| 79 | else:
|
|---|
| 80 | import os2emxpath as path
|
|---|
| 81 | from _emx_link import link
|
|---|
| 82 |
|
|---|
| 83 | import os2
|
|---|
| 84 | __all__.extend(_get_exports_list(os2))
|
|---|
| 85 | del os2
|
|---|
| 86 |
|
|---|
| 87 | elif 'mac' in _names:
|
|---|
| 88 | name = 'mac'
|
|---|
| 89 | linesep = '\r'
|
|---|
| 90 | from mac import *
|
|---|
| 91 | try:
|
|---|
| 92 | from mac import _exit
|
|---|
| 93 | except ImportError:
|
|---|
| 94 | pass
|
|---|
| 95 | import macpath as path
|
|---|
| 96 |
|
|---|
| 97 | import mac
|
|---|
| 98 | __all__.extend(_get_exports_list(mac))
|
|---|
| 99 | del mac
|
|---|
| 100 |
|
|---|
| 101 | elif 'ce' in _names:
|
|---|
| 102 | name = 'ce'
|
|---|
| 103 | linesep = '\r\n'
|
|---|
| 104 | from ce import *
|
|---|
| 105 | try:
|
|---|
| 106 | from ce import _exit
|
|---|
| 107 | except ImportError:
|
|---|
| 108 | pass
|
|---|
| 109 | # We can use the standard Windows path.
|
|---|
| 110 | import ntpath as path
|
|---|
| 111 |
|
|---|
| 112 | import ce
|
|---|
| 113 | __all__.extend(_get_exports_list(ce))
|
|---|
| 114 | del ce
|
|---|
| 115 |
|
|---|
| 116 | elif 'riscos' in _names:
|
|---|
| 117 | name = 'riscos'
|
|---|
| 118 | linesep = '\n'
|
|---|
| 119 | from riscos import *
|
|---|
| 120 | try:
|
|---|
| 121 | from riscos import _exit
|
|---|
| 122 | except ImportError:
|
|---|
| 123 | pass
|
|---|
| 124 | import riscospath as path
|
|---|
| 125 |
|
|---|
| 126 | import riscos
|
|---|
| 127 | __all__.extend(_get_exports_list(riscos))
|
|---|
| 128 | del riscos
|
|---|
| 129 |
|
|---|
| 130 | else:
|
|---|
| 131 | raise ImportError, 'no os specific module found'
|
|---|
| 132 |
|
|---|
| 133 | sys.modules['os.path'] = path
|
|---|
| 134 | from os.path import (curdir, pardir, sep, pathsep, defpath, extsep, altsep,
|
|---|
| 135 | devnull)
|
|---|
| 136 |
|
|---|
| 137 | del _names
|
|---|
| 138 |
|
|---|
| 139 | # Python uses fixed values for the SEEK_ constants; they are mapped
|
|---|
| 140 | # to native constants if necessary in posixmodule.c
|
|---|
| 141 | SEEK_SET = 0
|
|---|
| 142 | SEEK_CUR = 1
|
|---|
| 143 | SEEK_END = 2
|
|---|
| 144 |
|
|---|
| 145 | #'
|
|---|
| 146 |
|
|---|
| 147 | # Super directory utilities.
|
|---|
| 148 | # (Inspired by Eric Raymond; the doc strings are mostly his)
|
|---|
| 149 |
|
|---|
| 150 | def makedirs(name, mode=0777):
|
|---|
| 151 | """makedirs(path [, mode=0777])
|
|---|
| 152 |
|
|---|
| 153 | Super-mkdir; create a leaf directory and all intermediate ones.
|
|---|
| 154 | Works like mkdir, except that any intermediate path segment (not
|
|---|
| 155 | just the rightmost) will be created if it does not exist. This is
|
|---|
| 156 | recursive.
|
|---|
| 157 |
|
|---|
| 158 | """
|
|---|
| 159 | head, tail = path.split(name)
|
|---|
| 160 | if not tail:
|
|---|
| 161 | head, tail = path.split(head)
|
|---|
| 162 | if head and tail and not path.exists(head):
|
|---|
| 163 | makedirs(head, mode)
|
|---|
| 164 | if tail == curdir: # xxx/newdir/. exists if xxx/newdir exists
|
|---|
| 165 | return
|
|---|
| 166 | mkdir(name, mode)
|
|---|
| 167 |
|
|---|
| 168 | def removedirs(name):
|
|---|
| 169 | """removedirs(path)
|
|---|
| 170 |
|
|---|
| 171 | Super-rmdir; remove a leaf directory and all empty intermediate
|
|---|
| 172 | ones. Works like rmdir except that, if the leaf directory is
|
|---|
| 173 | successfully removed, directories corresponding to rightmost path
|
|---|
| 174 | segments will be pruned away until either the whole path is
|
|---|
| 175 | consumed or an error occurs. Errors during this latter phase are
|
|---|
| 176 | ignored -- they generally mean that a directory was not empty.
|
|---|
| 177 |
|
|---|
| 178 | """
|
|---|
| 179 | rmdir(name)
|
|---|
| 180 | head, tail = path.split(name)
|
|---|
| 181 | if not tail:
|
|---|
| 182 | head, tail = path.split(head)
|
|---|
| 183 | while head and tail:
|
|---|
| 184 | try:
|
|---|
| 185 | rmdir(head)
|
|---|
| 186 | except error:
|
|---|
| 187 | break
|
|---|
| 188 | head, tail = path.split(head)
|
|---|
| 189 |
|
|---|
| 190 | def renames(old, new):
|
|---|
| 191 | """renames(old, new)
|
|---|
| 192 |
|
|---|
| 193 | Super-rename; create directories as necessary and delete any left
|
|---|
| 194 | empty. Works like rename, except creation of any intermediate
|
|---|
| 195 | directories needed to make the new pathname good is attempted
|
|---|
| 196 | first. After the rename, directories corresponding to rightmost
|
|---|
| 197 | path segments of the old name will be pruned way until either the
|
|---|
| 198 | whole path is consumed or a nonempty directory is found.
|
|---|
| 199 |
|
|---|
| 200 | Note: this function can fail with the new directory structure made
|
|---|
| 201 | if you lack permissions needed to unlink the leaf directory or
|
|---|
| 202 | file.
|
|---|
| 203 |
|
|---|
| 204 | """
|
|---|
| 205 | head, tail = path.split(new)
|
|---|
| 206 | if head and tail and not path.exists(head):
|
|---|
| 207 | makedirs(head)
|
|---|
| 208 | rename(old, new)
|
|---|
| 209 | head, tail = path.split(old)
|
|---|
| 210 | if head and tail:
|
|---|
| 211 | try:
|
|---|
| 212 | removedirs(head)
|
|---|
| 213 | except error:
|
|---|
| 214 | pass
|
|---|
| 215 |
|
|---|
| 216 | __all__.extend(["makedirs", "removedirs", "renames"])
|
|---|
| 217 |
|
|---|
| 218 | def walk(top, topdown=True, onerror=None):
|
|---|
| 219 | """Directory tree generator.
|
|---|
| 220 |
|
|---|
| 221 | For each directory in the directory tree rooted at top (including top
|
|---|
| 222 | itself, but excluding '.' and '..'), yields a 3-tuple
|
|---|
| 223 |
|
|---|
| 224 | dirpath, dirnames, filenames
|
|---|
| 225 |
|
|---|
| 226 | dirpath is a string, the path to the directory. dirnames is a list of
|
|---|
| 227 | the names of the subdirectories in dirpath (excluding '.' and '..').
|
|---|
| 228 | filenames is a list of the names of the non-directory files in dirpath.
|
|---|
| 229 | Note that the names in the lists are just names, with no path components.
|
|---|
| 230 | To get a full path (which begins with top) to a file or directory in
|
|---|
| 231 | dirpath, do os.path.join(dirpath, name).
|
|---|
| 232 |
|
|---|
| 233 | If optional arg 'topdown' is true or not specified, the triple for a
|
|---|
| 234 | directory is generated before the triples for any of its subdirectories
|
|---|
| 235 | (directories are generated top down). If topdown is false, the triple
|
|---|
| 236 | for a directory is generated after the triples for all of its
|
|---|
| 237 | subdirectories (directories are generated bottom up).
|
|---|
| 238 |
|
|---|
| 239 | When topdown is true, the caller can modify the dirnames list in-place
|
|---|
| 240 | (e.g., via del or slice assignment), and walk will only recurse into the
|
|---|
| 241 | subdirectories whose names remain in dirnames; this can be used to prune
|
|---|
| 242 | the search, or to impose a specific order of visiting. Modifying
|
|---|
| 243 | dirnames when topdown is false is ineffective, since the directories in
|
|---|
| 244 | dirnames have already been generated by the time dirnames itself is
|
|---|
| 245 | generated.
|
|---|
| 246 |
|
|---|
| 247 | By default errors from the os.listdir() call are ignored. If
|
|---|
| 248 | optional arg 'onerror' is specified, it should be a function; it
|
|---|
| 249 | will be called with one argument, an os.error instance. It can
|
|---|
| 250 | report the error to continue with the walk, or raise the exception
|
|---|
| 251 | to abort the walk. Note that the filename is available as the
|
|---|
| 252 | filename attribute of the exception object.
|
|---|
| 253 |
|
|---|
| 254 | Caution: if you pass a relative pathname for top, don't change the
|
|---|
| 255 | current working directory between resumptions of walk. walk never
|
|---|
| 256 | changes the current directory, and assumes that the client doesn't
|
|---|
| 257 | either.
|
|---|
| 258 |
|
|---|
| 259 | Example:
|
|---|
| 260 |
|
|---|
| 261 | from os.path import join, getsize
|
|---|
| 262 | for root, dirs, files in walk('python/Lib/email'):
|
|---|
| 263 | print root, "consumes",
|
|---|
| 264 | print sum([getsize(join(root, name)) for name in files]),
|
|---|
| 265 | print "bytes in", len(files), "non-directory files"
|
|---|
| 266 | if 'CVS' in dirs:
|
|---|
| 267 | dirs.remove('CVS') # don't visit CVS directories
|
|---|
| 268 | """
|
|---|
| 269 |
|
|---|
| 270 | from os.path import join, isdir, islink
|
|---|
| 271 |
|
|---|
| 272 | # We may not have read permission for top, in which case we can't
|
|---|
| 273 | # get a list of the files the directory contains. os.path.walk
|
|---|
| 274 | # always suppressed the exception then, rather than blow up for a
|
|---|
| 275 | # minor reason when (say) a thousand readable directories are still
|
|---|
| 276 | # left to visit. That logic is copied here.
|
|---|
| 277 | try:
|
|---|
| 278 | # Note that listdir and error are globals in this module due
|
|---|
| 279 | # to earlier import-*.
|
|---|
| 280 | names = listdir(top)
|
|---|
| 281 | except error, err:
|
|---|
| 282 | if onerror is not None:
|
|---|
| 283 | onerror(err)
|
|---|
| 284 | return
|
|---|
| 285 |
|
|---|
| 286 | dirs, nondirs = [], []
|
|---|
| 287 | for name in names:
|
|---|
| 288 | if isdir(join(top, name)):
|
|---|
| 289 | dirs.append(name)
|
|---|
| 290 | else:
|
|---|
| 291 | nondirs.append(name)
|
|---|
| 292 |
|
|---|
| 293 | if topdown:
|
|---|
| 294 | yield top, dirs, nondirs
|
|---|
| 295 | for name in dirs:
|
|---|
| 296 | path = join(top, name)
|
|---|
| 297 | if not islink(path):
|
|---|
| 298 | for x in walk(path, topdown, onerror):
|
|---|
| 299 | yield x
|
|---|
| 300 | if not topdown:
|
|---|
| 301 | yield top, dirs, nondirs
|
|---|
| 302 |
|
|---|
| 303 | __all__.append("walk")
|
|---|
| 304 |
|
|---|
| 305 | # Make sure os.environ exists, at least
|
|---|
| 306 | try:
|
|---|
| 307 | environ
|
|---|
| 308 | except NameError:
|
|---|
| 309 | environ = {}
|
|---|
| 310 |
|
|---|
| 311 | def execl(file, *args):
|
|---|
| 312 | """execl(file, *args)
|
|---|
| 313 |
|
|---|
| 314 | Execute the executable file with argument list args, replacing the
|
|---|
| 315 | current process. """
|
|---|
| 316 | execv(file, args)
|
|---|
| 317 |
|
|---|
| 318 | def execle(file, *args):
|
|---|
| 319 | """execle(file, *args, env)
|
|---|
| 320 |
|
|---|
| 321 | Execute the executable file with argument list args and
|
|---|
| 322 | environment env, replacing the current process. """
|
|---|
| 323 | env = args[-1]
|
|---|
| 324 | execve(file, args[:-1], env)
|
|---|
| 325 |
|
|---|
| 326 | def execlp(file, *args):
|
|---|
| 327 | """execlp(file, *args)
|
|---|
| 328 |
|
|---|
| 329 | Execute the executable file (which is searched for along $PATH)
|
|---|
| 330 | with argument list args, replacing the current process. """
|
|---|
| 331 | execvp(file, args)
|
|---|
| 332 |
|
|---|
| 333 | def execlpe(file, *args):
|
|---|
| 334 | """execlpe(file, *args, env)
|
|---|
| 335 |
|
|---|
| 336 | Execute the executable file (which is searched for along $PATH)
|
|---|
| 337 | with argument list args and environment env, replacing the current
|
|---|
| 338 | process. """
|
|---|
| 339 | env = args[-1]
|
|---|
| 340 | execvpe(file, args[:-1], env)
|
|---|
| 341 |
|
|---|
| 342 | def execvp(file, args):
|
|---|
| 343 | """execp(file, args)
|
|---|
| 344 |
|
|---|
| 345 | Execute the executable file (which is searched for along $PATH)
|
|---|
| 346 | with argument list args, replacing the current process.
|
|---|
| 347 | args may be a list or tuple of strings. """
|
|---|
| 348 | _execvpe(file, args)
|
|---|
| 349 |
|
|---|
| 350 | def execvpe(file, args, env):
|
|---|
| 351 | """execvpe(file, args, env)
|
|---|
| 352 |
|
|---|
| 353 | Execute the executable file (which is searched for along $PATH)
|
|---|
| 354 | with argument list args and environment env , replacing the
|
|---|
| 355 | current process.
|
|---|
| 356 | args may be a list or tuple of strings. """
|
|---|
| 357 | _execvpe(file, args, env)
|
|---|
| 358 |
|
|---|
| 359 | __all__.extend(["execl","execle","execlp","execlpe","execvp","execvpe"])
|
|---|
| 360 |
|
|---|
| 361 | def _execvpe(file, args, env=None):
|
|---|
| 362 | from errno import ENOENT, ENOTDIR
|
|---|
| 363 |
|
|---|
| 364 | if env is not None:
|
|---|
| 365 | func = execve
|
|---|
| 366 | argrest = (args, env)
|
|---|
| 367 | else:
|
|---|
| 368 | func = execv
|
|---|
| 369 | argrest = (args,)
|
|---|
| 370 | env = environ
|
|---|
| 371 |
|
|---|
| 372 | head, tail = path.split(file)
|
|---|
| 373 | if head:
|
|---|
| 374 | func(file, *argrest)
|
|---|
| 375 | return
|
|---|
| 376 | if 'PATH' in env:
|
|---|
| 377 | envpath = env['PATH']
|
|---|
| 378 | else:
|
|---|
| 379 | envpath = defpath
|
|---|
| 380 | PATH = envpath.split(pathsep)
|
|---|
| 381 | saved_exc = None
|
|---|
| 382 | saved_tb = None
|
|---|
| 383 | for dir in PATH:
|
|---|
| 384 | fullname = path.join(dir, file)
|
|---|
| 385 | try:
|
|---|
| 386 | func(fullname, *argrest)
|
|---|
| 387 | except error, e:
|
|---|
| 388 | tb = sys.exc_info()[2]
|
|---|
| 389 | if (e.errno != ENOENT and e.errno != ENOTDIR
|
|---|
| 390 | and saved_exc is None):
|
|---|
| 391 | saved_exc = e
|
|---|
| 392 | saved_tb = tb
|
|---|
| 393 | if saved_exc:
|
|---|
| 394 | raise error, saved_exc, saved_tb
|
|---|
| 395 | raise error, e, tb
|
|---|
| 396 |
|
|---|
|
|---|