Issue #5768: Change to Unicode output logic and test case for same.
[python.git] / Lib / site.py
blobabe668759cd4ad37e5745247c6a0ce5d2c459535
1 """Append module search paths for third-party packages to sys.path.
3 ****************************************************************
4 * This module is automatically imported during initialization. *
5 ****************************************************************
7 In earlier versions of Python (up to 1.5a3), scripts or modules that
8 needed to use site-specific modules would place ``import site''
9 somewhere near the top of their code. Because of the automatic
10 import, this is no longer necessary (but code that does it still
11 works).
13 This will append site-specific paths to the module search path. On
14 Unix (including Mac OSX), it starts with sys.prefix and
15 sys.exec_prefix (if different) and appends
16 lib/python<version>/site-packages as well as lib/site-python.
17 On other platforms (such as Windows), it tries each of the
18 prefixes directly, as well as with lib/site-packages appended. The
19 resulting directories, if they exist, are appended to sys.path, and
20 also inspected for path configuration files.
22 A path configuration file is a file whose name has the form
23 <package>.pth; its contents are additional directories (one per line)
24 to be added to sys.path. Non-existing directories (or
25 non-directories) are never added to sys.path; no directory is added to
26 sys.path more than once. Blank lines and lines beginning with
27 '#' are skipped. Lines starting with 'import' are executed.
29 For example, suppose sys.prefix and sys.exec_prefix are set to
30 /usr/local and there is a directory /usr/local/lib/python2.5/site-packages
31 with three subdirectories, foo, bar and spam, and two path
32 configuration files, foo.pth and bar.pth. Assume foo.pth contains the
33 following:
35 # foo package configuration
36 foo
37 bar
38 bletch
40 and bar.pth contains:
42 # bar package configuration
43 bar
45 Then the following directories are added to sys.path, in this order:
47 /usr/local/lib/python2.5/site-packages/bar
48 /usr/local/lib/python2.5/site-packages/foo
50 Note that bletch is omitted because it doesn't exist; bar precedes foo
51 because bar.pth comes alphabetically before foo.pth; and spam is
52 omitted because it is not mentioned in either path configuration file.
54 After these path manipulations, an attempt is made to import a module
55 named sitecustomize, which can perform arbitrary additional
56 site-specific customizations. If this import fails with an
57 ImportError exception, it is silently ignored.
59 """
61 import sys
62 import os
63 import __builtin__
65 # Prefixes for site-packages; add additional prefixes like /usr/local here
66 PREFIXES = [sys.prefix, sys.exec_prefix]
67 # Enable per user site-packages directory
68 # set it to False to disable the feature or True to force the feature
69 ENABLE_USER_SITE = None
70 # for distutils.commands.install
71 USER_SITE = None
72 USER_BASE = None
75 def makepath(*paths):
76 dir = os.path.abspath(os.path.join(*paths))
77 return dir, os.path.normcase(dir)
80 def abs__file__():
81 """Set all module' __file__ attribute to an absolute path"""
82 for m in sys.modules.values():
83 if hasattr(m, '__loader__'):
84 continue # don't mess with a PEP 302-supplied __file__
85 try:
86 m.__file__ = os.path.abspath(m.__file__)
87 except AttributeError:
88 continue
91 def removeduppaths():
92 """ Remove duplicate entries from sys.path along with making them
93 absolute"""
94 # This ensures that the initial path provided by the interpreter contains
95 # only absolute pathnames, even if we're running from the build directory.
96 L = []
97 known_paths = set()
98 for dir in sys.path:
99 # Filter out duplicate paths (on case-insensitive file systems also
100 # if they only differ in case); turn relative paths into absolute
101 # paths.
102 dir, dircase = makepath(dir)
103 if not dircase in known_paths:
104 L.append(dir)
105 known_paths.add(dircase)
106 sys.path[:] = L
107 return known_paths
109 # XXX This should not be part of site.py, since it is needed even when
110 # using the -S option for Python. See http://www.python.org/sf/586680
111 def addbuilddir():
112 """Append ./build/lib.<platform> in case we're running in the build dir
113 (especially for Guido :-)"""
114 from distutils.util import get_platform
115 s = "build/lib.%s-%.3s" % (get_platform(), sys.version)
116 if hasattr(sys, 'gettotalrefcount'):
117 s += '-pydebug'
118 s = os.path.join(os.path.dirname(sys.path[-1]), s)
119 sys.path.append(s)
122 def _init_pathinfo():
123 """Return a set containing all existing directory entries from sys.path"""
124 d = set()
125 for dir in sys.path:
126 try:
127 if os.path.isdir(dir):
128 dir, dircase = makepath(dir)
129 d.add(dircase)
130 except TypeError:
131 continue
132 return d
135 def addpackage(sitedir, name, known_paths):
136 """Process a .pth file within the site-packages directory:
137 For each line in the file, either combine it with sitedir to a path
138 and add that to known_paths, or execute it if it starts with 'import '.
140 if known_paths is None:
141 _init_pathinfo()
142 reset = 1
143 else:
144 reset = 0
145 fullname = os.path.join(sitedir, name)
146 try:
147 f = open(fullname, "rU")
148 except IOError:
149 return
150 with f:
151 for line in f:
152 if line.startswith("#"):
153 continue
154 if line.startswith(("import ", "import\t")):
155 exec line
156 continue
157 line = line.rstrip()
158 dir, dircase = makepath(sitedir, line)
159 if not dircase in known_paths and os.path.exists(dir):
160 sys.path.append(dir)
161 known_paths.add(dircase)
162 if reset:
163 known_paths = None
164 return known_paths
167 def addsitedir(sitedir, known_paths=None):
168 """Add 'sitedir' argument to sys.path if missing and handle .pth files in
169 'sitedir'"""
170 if known_paths is None:
171 known_paths = _init_pathinfo()
172 reset = 1
173 else:
174 reset = 0
175 sitedir, sitedircase = makepath(sitedir)
176 if not sitedircase in known_paths:
177 sys.path.append(sitedir) # Add path component
178 try:
179 names = os.listdir(sitedir)
180 except os.error:
181 return
182 dotpth = os.extsep + "pth"
183 names = [name for name in names if name.endswith(dotpth)]
184 for name in sorted(names):
185 addpackage(sitedir, name, known_paths)
186 if reset:
187 known_paths = None
188 return known_paths
191 def check_enableusersite():
192 """Check if user site directory is safe for inclusion
194 The function tests for the command line flag (including environment var),
195 process uid/gid equal to effective uid/gid.
197 None: Disabled for security reasons
198 False: Disabled by user (command line option)
199 True: Safe and enabled
201 if sys.flags.no_user_site:
202 return False
204 if hasattr(os, "getuid") and hasattr(os, "geteuid"):
205 # check process uid == effective uid
206 if os.geteuid() != os.getuid():
207 return None
208 if hasattr(os, "getgid") and hasattr(os, "getegid"):
209 # check process gid == effective gid
210 if os.getegid() != os.getgid():
211 return None
213 return True
216 def addusersitepackages(known_paths):
217 """Add a per user site-package to sys.path
219 Each user has its own python directory with site-packages in the
220 home directory.
222 USER_BASE is the root directory for all Python versions
224 USER_SITE is the user specific site-packages directory
226 USER_SITE/.. can be used for data.
228 global USER_BASE, USER_SITE, ENABLE_USER_SITE
229 env_base = os.environ.get("PYTHONUSERBASE", None)
231 def joinuser(*args):
232 return os.path.expanduser(os.path.join(*args))
234 #if sys.platform in ('os2emx', 'riscos'):
235 # # Don't know what to put here
236 # USER_BASE = ''
237 # USER_SITE = ''
238 if os.name == "nt":
239 base = os.environ.get("APPDATA") or "~"
240 USER_BASE = env_base if env_base else joinuser(base, "Python")
241 USER_SITE = os.path.join(USER_BASE,
242 "Python" + sys.version[0] + sys.version[2],
243 "site-packages")
244 else:
245 USER_BASE = env_base if env_base else joinuser("~", ".local")
246 USER_SITE = os.path.join(USER_BASE, "lib",
247 "python" + sys.version[:3],
248 "site-packages")
250 if ENABLE_USER_SITE and os.path.isdir(USER_SITE):
251 addsitedir(USER_SITE, known_paths)
252 return known_paths
255 def addsitepackages(known_paths):
256 """Add site-packages (and possibly site-python) to sys.path"""
257 sitedirs = []
258 seen = []
260 for prefix in PREFIXES:
261 if not prefix or prefix in seen:
262 continue
263 seen.append(prefix)
265 if sys.platform in ('os2emx', 'riscos'):
266 sitedirs.append(os.path.join(prefix, "Lib", "site-packages"))
267 elif os.sep == '/':
268 sitedirs.append(os.path.join(prefix, "lib",
269 "python" + sys.version[:3],
270 "site-packages"))
271 sitedirs.append(os.path.join(prefix, "lib", "site-python"))
272 else:
273 sitedirs.append(prefix)
274 sitedirs.append(os.path.join(prefix, "lib", "site-packages"))
276 if sys.platform == "darwin":
277 # for framework builds *only* we add the standard Apple
278 # locations.
279 if 'Python.framework' in prefix:
280 sitedirs.append(
281 os.path.expanduser(
282 os.path.join("~", "Library", "Python",
283 sys.version[:3], "site-packages")))
284 sitedirs.append(
285 os.path.join("/Library", "Python",
286 sys.version[:3], "site-packages"))
288 for sitedir in sitedirs:
289 if os.path.isdir(sitedir):
290 addsitedir(sitedir, known_paths)
292 return known_paths
295 def setBEGINLIBPATH():
296 """The OS/2 EMX port has optional extension modules that do double duty
297 as DLLs (and must use the .DLL file extension) for other extensions.
298 The library search path needs to be amended so these will be found
299 during module import. Use BEGINLIBPATH so that these are at the start
300 of the library search path.
303 dllpath = os.path.join(sys.prefix, "Lib", "lib-dynload")
304 libpath = os.environ['BEGINLIBPATH'].split(';')
305 if libpath[-1]:
306 libpath.append(dllpath)
307 else:
308 libpath[-1] = dllpath
309 os.environ['BEGINLIBPATH'] = ';'.join(libpath)
312 def setquit():
313 """Define new built-ins 'quit' and 'exit'.
314 These are simply strings that display a hint on how to exit.
317 if os.sep == ':':
318 eof = 'Cmd-Q'
319 elif os.sep == '\\':
320 eof = 'Ctrl-Z plus Return'
321 else:
322 eof = 'Ctrl-D (i.e. EOF)'
324 class Quitter(object):
325 def __init__(self, name):
326 self.name = name
327 def __repr__(self):
328 return 'Use %s() or %s to exit' % (self.name, eof)
329 def __call__(self, code=None):
330 # Shells like IDLE catch the SystemExit, but listen when their
331 # stdin wrapper is closed.
332 try:
333 sys.stdin.close()
334 except:
335 pass
336 raise SystemExit(code)
337 __builtin__.quit = Quitter('quit')
338 __builtin__.exit = Quitter('exit')
341 class _Printer(object):
342 """interactive prompt objects for printing the license text, a list of
343 contributors and the copyright notice."""
345 MAXLINES = 23
347 def __init__(self, name, data, files=(), dirs=()):
348 self.__name = name
349 self.__data = data
350 self.__files = files
351 self.__dirs = dirs
352 self.__lines = None
354 def __setup(self):
355 if self.__lines:
356 return
357 data = None
358 for dir in self.__dirs:
359 for filename in self.__files:
360 filename = os.path.join(dir, filename)
361 try:
362 fp = file(filename, "rU")
363 data = fp.read()
364 fp.close()
365 break
366 except IOError:
367 pass
368 if data:
369 break
370 if not data:
371 data = self.__data
372 self.__lines = data.split('\n')
373 self.__linecnt = len(self.__lines)
375 def __repr__(self):
376 self.__setup()
377 if len(self.__lines) <= self.MAXLINES:
378 return "\n".join(self.__lines)
379 else:
380 return "Type %s() to see the full %s text" % ((self.__name,)*2)
382 def __call__(self):
383 self.__setup()
384 prompt = 'Hit Return for more, or q (and Return) to quit: '
385 lineno = 0
386 while 1:
387 try:
388 for i in range(lineno, lineno + self.MAXLINES):
389 print self.__lines[i]
390 except IndexError:
391 break
392 else:
393 lineno += self.MAXLINES
394 key = None
395 while key is None:
396 key = raw_input(prompt)
397 if key not in ('', 'q'):
398 key = None
399 if key == 'q':
400 break
402 def setcopyright():
403 """Set 'copyright' and 'credits' in __builtin__"""
404 __builtin__.copyright = _Printer("copyright", sys.copyright)
405 if sys.platform[:4] == 'java':
406 __builtin__.credits = _Printer(
407 "credits",
408 "Jython is maintained by the Jython developers (www.jython.org).")
409 else:
410 __builtin__.credits = _Printer("credits", """\
411 Thanks to CWI, CNRI, BeOpen.com, Zope Corporation and a cast of thousands
412 for supporting Python development. See www.python.org for more information.""")
413 here = os.path.dirname(os.__file__)
414 __builtin__.license = _Printer(
415 "license", "See http://www.python.org/%.3s/license.html" % sys.version,
416 ["LICENSE.txt", "LICENSE"],
417 [os.path.join(here, os.pardir), here, os.curdir])
420 class _Helper(object):
421 """Define the built-in 'help'.
422 This is a wrapper around pydoc.help (with a twist).
426 def __repr__(self):
427 return "Type help() for interactive help, " \
428 "or help(object) for help about object."
429 def __call__(self, *args, **kwds):
430 import pydoc
431 return pydoc.help(*args, **kwds)
433 def sethelper():
434 __builtin__.help = _Helper()
436 def aliasmbcs():
437 """On Windows, some default encodings are not provided by Python,
438 while they are always available as "mbcs" in each locale. Make
439 them usable by aliasing to "mbcs" in such a case."""
440 if sys.platform == 'win32':
441 import locale, codecs
442 enc = locale.getdefaultlocale()[1]
443 if enc.startswith('cp'): # "cp***" ?
444 try:
445 codecs.lookup(enc)
446 except LookupError:
447 import encodings
448 encodings._cache[enc] = encodings._unknown
449 encodings.aliases.aliases[enc] = 'mbcs'
451 def setencoding():
452 """Set the string encoding used by the Unicode implementation. The
453 default is 'ascii', but if you're willing to experiment, you can
454 change this."""
455 encoding = "ascii" # Default value set by _PyUnicode_Init()
456 if 0:
457 # Enable to support locale aware default string encodings.
458 import locale
459 loc = locale.getdefaultlocale()
460 if loc[1]:
461 encoding = loc[1]
462 if 0:
463 # Enable to switch off string to Unicode coercion and implicit
464 # Unicode to string conversion.
465 encoding = "undefined"
466 if encoding != "ascii":
467 # On Non-Unicode builds this will raise an AttributeError...
468 sys.setdefaultencoding(encoding) # Needs Python Unicode build !
471 def execsitecustomize():
472 """Run custom site specific code, if available."""
473 try:
474 import sitecustomize
475 except ImportError:
476 pass
479 def execusercustomize():
480 """Run custom user specific code, if available."""
481 try:
482 import usercustomize
483 except ImportError:
484 pass
487 def main():
488 global ENABLE_USER_SITE
490 abs__file__()
491 known_paths = removeduppaths()
492 if (os.name == "posix" and sys.path and
493 os.path.basename(sys.path[-1]) == "Modules"):
494 addbuilddir()
495 if ENABLE_USER_SITE is None:
496 ENABLE_USER_SITE = check_enableusersite()
497 known_paths = addusersitepackages(known_paths)
498 known_paths = addsitepackages(known_paths)
499 if sys.platform == 'os2emx':
500 setBEGINLIBPATH()
501 setquit()
502 setcopyright()
503 sethelper()
504 aliasmbcs()
505 setencoding()
506 execsitecustomize()
507 if ENABLE_USER_SITE:
508 execusercustomize()
509 # Remove sys.setdefaultencoding() so that users cannot change the
510 # encoding after initialization. The test for presence is needed when
511 # this module is run as a script, because this code is executed twice.
512 if hasattr(sys, "setdefaultencoding"):
513 del sys.setdefaultencoding
515 main()
517 def _script():
518 help = """\
519 %s [--user-base] [--user-site]
521 Without arguments print some useful information
522 With arguments print the value of USER_BASE and/or USER_SITE separated
523 by '%s'.
525 Exit codes with --user-base or --user-site:
526 0 - user site directory is enabled
527 1 - user site directory is disabled by user
528 2 - uses site directory is disabled by super user
529 or for security reasons
530 >2 - unknown error
532 args = sys.argv[1:]
533 if not args:
534 print "sys.path = ["
535 for dir in sys.path:
536 print " %r," % (dir,)
537 print "]"
538 print "USER_BASE: %r (%s)" % (USER_BASE,
539 "exists" if os.path.isdir(USER_BASE) else "doesn't exist")
540 print "USER_SITE: %r (%s)" % (USER_SITE,
541 "exists" if os.path.isdir(USER_SITE) else "doesn't exist")
542 print "ENABLE_USER_SITE: %r" % ENABLE_USER_SITE
543 sys.exit(0)
545 buffer = []
546 if '--user-base' in args:
547 buffer.append(USER_BASE)
548 if '--user-site' in args:
549 buffer.append(USER_SITE)
551 if buffer:
552 print os.pathsep.join(buffer)
553 if ENABLE_USER_SITE:
554 sys.exit(0)
555 elif ENABLE_USER_SITE is False:
556 sys.exit(1)
557 elif ENABLE_USER_SITE is None:
558 sys.exit(2)
559 else:
560 sys.exit(3)
561 else:
562 import textwrap
563 print textwrap.dedent(help % (sys.argv[0], os.pathsep))
564 sys.exit(10)
566 if __name__ == '__main__':
567 _script()