Refactored site.py into functions. Also moved over to using sets.

New regression test suite.
This commit is contained in:
Brett Cannon 2004-06-05 01:12:51 +00:00
parent 642c8a11fd
commit 0096e262ff
3 changed files with 425 additions and 190 deletions

View File

@ -57,64 +57,104 @@ ImportError exception, it is silently ignored.
""" """
import sys, os import sys
import os
import __builtin__
def makepath(*paths): def makepath(*paths):
dir = os.path.abspath(os.path.join(*paths)) dir = os.path.abspath(os.path.join(*paths))
return dir, os.path.normcase(dir) return dir, os.path.normcase(dir)
for m in sys.modules.values(): def abs__file__():
if hasattr(m, "__file__") and m.__file__: """Set all module' __file__ attribute to an absolute path"""
m.__file__ = os.path.abspath(m.__file__) for m in sys.modules.values():
del m try:
m.__file__ = os.path.abspath(m.__file__)
except AttributeError:
continue
# This ensures that the initial path provided by the interpreter contains def removeduppaths():
# only absolute pathnames, even if we're running from the build directory. """ Remove duplicate entries from sys.path along with making them
L = [] absolute"""
_dirs_in_sys_path = {} # This ensures that the initial path provided by the interpreter contains
dir = dircase = None # sys.path may be empty at this point # only absolute pathnames, even if we're running from the build directory.
for dir in sys.path: L = []
# Filter out duplicate paths (on case-insensitive file systems also known_paths = set()
# if they only differ in case); turn relative paths into absolute for dir in sys.path:
# paths. # Filter out duplicate paths (on case-insensitive file systems also
dir, dircase = makepath(dir) # if they only differ in case); turn relative paths into absolute
if not dircase in _dirs_in_sys_path: # paths.
L.append(dir) dir, dircase = makepath(dir)
_dirs_in_sys_path[dircase] = 1 if not dircase in known_paths:
sys.path[:] = L L.append(dir)
del dir, dircase, L known_paths.add(dircase)
sys.path[:] = L
return known_paths
# Append ./build/lib.<platform> in case we're running in the build dir
# (especially for Guido :-)
# XXX This should not be part of site.py, since it is needed even when # XXX This should not be part of site.py, since it is needed even when
# using the -S option for Python. See http://www.python.org/sf/586680 # using the -S option for Python. See http://www.python.org/sf/586680
if (os.name == "posix" and sys.path and def addbuilddir():
os.path.basename(sys.path[-1]) == "Modules"): """Append ./build/lib.<platform> in case we're running in the build dir
(especially for Guido :-)"""
from distutils.util import get_platform from distutils.util import get_platform
s = "build/lib.%s-%.3s" % (get_platform(), sys.version) s = "build/lib.%s-%.3s" % (get_platform(), sys.version)
s = os.path.join(os.path.dirname(sys.path[-1]), s) s = os.path.join(os.path.dirname(sys.path[-1]), s)
sys.path.append(s) sys.path.append(s)
del get_platform, s
def _init_pathinfo(): def _init_pathinfo():
global _dirs_in_sys_path """Return a set containing all existing directory entries from sys.path"""
_dirs_in_sys_path = d = {} d = set()
for dir in sys.path: for dir in sys.path:
if dir and not os.path.isdir(dir): try:
if os.path.isdir(dir):
dir, dircase = makepath(dir)
d.add(dircase)
except TypeError:
continue continue
dir, dircase = makepath(dir) return d
d[dircase] = 1
def addsitedir(sitedir): def addpackage(sitedir, name, known_paths):
global _dirs_in_sys_path """Add a new path to known_paths by combining sitedir and 'name' or execute
if _dirs_in_sys_path is None: sitedir if it starts with 'import'"""
if known_paths is None:
_init_pathinfo() _init_pathinfo()
reset = 1 reset = 1
else: else:
reset = 0 reset = 0
fullname = os.path.join(sitedir, name)
try:
f = file(fullname, "rU")
except IOError:
return
try:
for line in f:
if line.startswith("#"):
continue
if line.startswith("import"):
exec line
continue
line = line.rstrip()
dir, dircase = makepath(sitedir, line)
if not dircase in known_paths and os.path.exists(dir):
sys.path.append(dir)
known_paths.add(dircase)
finally:
f.close()
if reset:
known_paths = None
return known_paths
def addsitedir(sitedir, known_paths):
"""Add 'sitedir' argument to sys.path if missing and handle .pth files in
'sitedir'"""
if known_paths is None:
d = _init_pathinfo()
reset = 1
else:
reset = 0
sitedir, sitedircase = makepath(sitedir) sitedir, sitedircase = makepath(sitedir)
if not sitedircase in _dirs_in_sys_path: if not sitedircase in known_paths:
sys.path.append(sitedir) # Add path component sys.path.append(sitedir) # Add path component
try: try:
names = os.listdir(sitedir) names = os.listdir(sitedir)
@ -123,82 +163,55 @@ def addsitedir(sitedir):
names.sort() names.sort()
for name in names: for name in names:
if name[-4:] == os.extsep + "pth": if name[-4:] == os.extsep + "pth":
addpackage(sitedir, name) addpackage(sitedir, name, known_paths)
if reset: if reset:
_dirs_in_sys_path = None known_paths = None
return known_paths
def addpackage(sitedir, name): def addsitepackages(known_paths):
global _dirs_in_sys_path """Add site-packages (and possibly site-python) to sys.path"""
if _dirs_in_sys_path is None: prefixes = [sys.prefix]
_init_pathinfo() if sys.exec_prefix != sys.prefix:
reset = 1 prefixes.append(sys.exec_prefix)
else: for prefix in prefixes:
reset = 0 if prefix:
fullname = os.path.join(sitedir, name) if sys.platform in ('os2emx', 'riscos'):
try: sitedirs = [os.path.join(prefix, "Lib", "site-packages")]
f = open(fullname) elif os.sep == '/':
except IOError: sitedirs = [os.path.join(prefix,
return "lib",
while 1: "python" + sys.version[:3],
dir = f.readline() "site-packages"),
if not dir: os.path.join(prefix, "lib", "site-python")]
break else:
if dir[0] == '#': sitedirs = [prefix, os.path.join(prefix, "lib", "site-packages")]
continue if sys.platform == 'darwin':
if dir.startswith("import"): # for framework builds *only* we add the standard Apple
exec dir # locations. Currently only per-user, but /Library and
continue # /Network/Library could be added too
dir = dir.rstrip() if 'Python.framework' in prefix:
dir, dircase = makepath(sitedir, dir) home = os.environ.get('HOME')
if not dircase in _dirs_in_sys_path and os.path.exists(dir): if home:
sys.path.append(dir) sitedirs.append(
_dirs_in_sys_path[dircase] = 1 os.path.join(home,
if reset: 'Library',
_dirs_in_sys_path = None 'Python',
sys.version[:3],
prefixes = [sys.prefix] 'site-packages'))
sitedir = None # make sure sitedir is initialized because of later 'del' for sitedir in sitedirs:
if sys.exec_prefix != sys.prefix: if os.path.isdir(sitedir):
prefixes.append(sys.exec_prefix) addsitedir(sitedir, known_paths)
for prefix in prefixes: return None
if prefix:
if sys.platform in ('os2emx', 'riscos'):
sitedirs = [os.path.join(prefix, "Lib", "site-packages")]
elif os.sep == '/':
sitedirs = [os.path.join(prefix,
"lib",
"python" + sys.version[:3],
"site-packages"),
os.path.join(prefix, "lib", "site-python")]
else:
sitedirs = [prefix, os.path.join(prefix, "lib", "site-packages")]
if sys.platform == 'darwin':
# for framework builds *only* we add the standard Apple
# locations. Currently only per-user, but /Library and
# /Network/Library could be added too
if 'Python.framework' in prefix:
home = os.environ.get('HOME')
if home:
sitedirs.append(
os.path.join(home,
'Library',
'Python',
sys.version[:3],
'site-packages'))
for sitedir in sitedirs:
if os.path.isdir(sitedir):
addsitedir(sitedir)
del prefix, sitedir
_dirs_in_sys_path = None
# the OS/2 EMX port has optional extension modules that do double duty def setBEGINLIBPATH():
# as DLLs (and must use the .DLL file extension) for other extensions. """The OS/2 EMX port has optional extension modules that do double duty
# The library search path needs to be amended so these will be found as DLLs (and must use the .DLL file extension) for other extensions.
# during module import. Use BEGINLIBPATH so that these are at the start The library search path needs to be amended so these will be found
# of the library search path. during module import. Use BEGINLIBPATH so that these are at the start
if sys.platform == 'os2emx': of the library search path.
"""
dllpath = os.path.join(sys.prefix, "Lib", "lib-dynload") dllpath = os.path.join(sys.prefix, "Lib", "lib-dynload")
libpath = os.environ['BEGINLIBPATH'].split(';') libpath = os.environ['BEGINLIBPATH'].split(';')
if libpath[-1]: if libpath[-1]:
@ -208,21 +221,24 @@ if sys.platform == 'os2emx':
os.environ['BEGINLIBPATH'] = ';'.join(libpath) os.environ['BEGINLIBPATH'] = ';'.join(libpath)
# Define new built-ins 'quit' and 'exit'. def setquit():
# These are simply strings that display a hint on how to exit. """Define new built-ins 'quit' and 'exit'.
if os.sep == ':': These are simply strings that display a hint on how to exit.
exit = 'Use Cmd-Q to quit.'
elif os.sep == '\\': """
exit = 'Use Ctrl-Z plus Return to exit.' if os.sep == ':':
else: exit = 'Use Cmd-Q to quit.'
exit = 'Use Ctrl-D (i.e. EOF) to exit.' elif os.sep == '\\':
import __builtin__ exit = 'Use Ctrl-Z plus Return to exit.'
__builtin__.quit = __builtin__.exit = exit else:
del exit exit = 'Use Ctrl-D (i.e. EOF) to exit.'
__builtin__.quit = __builtin__.exit = exit
class _Printer(object):
"""interactive prompt objects for printing the license text, a list of
contributors and the copyright notice."""
# interactive prompt objects for printing the license text, a list of
# contributors and the copyright notice.
class _Printer:
MAXLINES = 23 MAXLINES = 23
def __init__(self, name, data, files=(), dirs=()): def __init__(self, name, data, files=(), dirs=()):
@ -237,10 +253,10 @@ class _Printer:
return return
data = None data = None
for dir in self.__dirs: for dir in self.__dirs:
for file in self.__files: for filename in self.__files:
file = os.path.join(dir, file) filename = os.path.join(dir, filename)
try: try:
fp = open(file) fp = file(filename, "rU")
data = fp.read() data = fp.read()
fp.close() fp.close()
break break
@ -280,26 +296,30 @@ class _Printer:
if key == 'q': if key == 'q':
break break
__builtin__.copyright = _Printer("copyright", sys.copyright) def setcopyright():
if sys.platform[:4] == 'java': """Set 'copyright' and 'credits' in __builtin__"""
__builtin__.credits = _Printer( __builtin__.copyright = _Printer("copyright", sys.copyright)
"credits", if sys.platform[:4] == 'java':
"Jython is maintained by the Jython developers (www.jython.org).") __builtin__.credits = _Printer(
else: "credits",
__builtin__.credits = _Printer("credits", """\ "Jython is maintained by the Jython developers (www.jython.org).")
Thanks to CWI, CNRI, BeOpen.com, Zope Corporation and a cast of thousands else:
for supporting Python development. See www.python.org for more information.""") __builtin__.credits = _Printer("credits", """\
here = os.path.dirname(os.__file__) Thanks to CWI, CNRI, BeOpen.com, Zope Corporation and a cast of thousands
__builtin__.license = _Printer( for supporting Python development. See www.python.org for more information.""")
"license", "See http://www.python.org/%.3s/license.html" % sys.version, here = os.path.dirname(os.__file__)
["LICENSE.txt", "LICENSE"], __builtin__.license = _Printer(
[os.path.join(here, os.pardir), here, os.curdir]) "license", "See http://www.python.org/%.3s/license.html" % sys.version,
["LICENSE.txt", "LICENSE"],
[os.path.join(here, os.pardir), here, os.curdir])
# Define new built-in 'help'. class _Helper(object):
# This is a wrapper around pydoc.help (with a twist). """Define the built-in 'help'.
This is a wrapper around pydoc.help (with a twist).
"""
class _Helper:
def __repr__(self): def __repr__(self):
return "Type help() for interactive help, " \ return "Type help() for interactive help, " \
"or help(object) for help about object." "or help(object) for help about object."
@ -307,61 +327,74 @@ class _Helper:
import pydoc import pydoc
return pydoc.help(*args, **kwds) return pydoc.help(*args, **kwds)
__builtin__.help = _Helper() def sethelper():
__builtin__.help = _Helper()
def aliasmbcs():
"""On Windows, some default encodings are not provided by Python,
while they are always available as "mbcs" in each locale. Make
them usable by aliasing to "mbcs" in such a case."""
if sys.platform == 'win32':
import locale, codecs
enc = locale.getdefaultlocale()[1]
if enc.startswith('cp'): # "cp***" ?
try:
codecs.lookup(enc)
except LookupError:
import encodings
encodings._cache[enc] = encodings._unknown
encodings.aliases.aliases[enc] = 'mbcs'
def setencoding():
"""Set the string encoding used by the Unicode implementation. The
default is 'ascii', but if you're willing to experiment, you can
change this."""
encoding = "ascii" # Default value set by _PyUnicode_Init()
if 0:
# Enable to support locale aware default string encodings.
import locale
loc = locale.getdefaultlocale()
if loc[1]:
encoding = loc[1]
if 0:
# Enable to switch off string to Unicode coercion and implicit
# Unicode to string conversion.
encoding = "undefined"
if encoding != "ascii":
# On Non-Unicode builds this will raise an AttributeError...
sys.setdefaultencoding(encoding) # Needs Python Unicode build !
# On Windows, some default encodings are not provided by Python, def execsitecustomize():
# while they are always available as "mbcs" in each locale. Make """Run custom site specific code, if available."""
# them usable by aliasing to "mbcs" in such a case. try:
import sitecustomize
except ImportError:
pass
if sys.platform == 'win32':
import locale, codecs
enc = locale.getdefaultlocale()[1]
if enc.startswith('cp'): # "cp***" ?
try:
codecs.lookup(enc)
except LookupError:
import encodings
encodings._cache[enc] = encodings._unknown
encodings.aliases.aliases[enc] = 'mbcs'
# Set the string encoding used by the Unicode implementation. The def main():
# default is 'ascii', but if you're willing to experiment, you can abs__file__()
# change this. paths_in_sys = removeduppaths()
if (os.name == "posix" and sys.path and
os.path.basename(sys.path[-1]) == "Modules"):
addbuilddir()
paths_in_sys = addsitepackages(paths_in_sys)
if sys.platform == 'os2emx':
setBEGINLIBPATH()
setquit()
setcopyright()
sethelper()
aliasmbcs()
setencoding()
execsitecustomize()
# Remove sys.setdefaultencoding() so that users cannot change the
# encoding after initialization. The test for presence is needed when
# this module is run as a script, because this code is executed twice.
if hasattr(sys, "setdefaultencoding"):
del sys.setdefaultencoding
encoding = "ascii" # Default value set by _PyUnicode_Init() main()
if 0:
# Enable to support locale aware default string encodings.
import locale
loc = locale.getdefaultlocale()
if loc[1]:
encoding = loc[1]
if 0:
# Enable to switch off string to Unicode coercion and implicit
# Unicode to string conversion.
encoding = "undefined"
if encoding != "ascii":
# On Non-Unicode builds this will raise an AttributeError...
sys.setdefaultencoding(encoding) # Needs Python Unicode build !
#
# Run custom site specific code, if available.
#
try:
import sitecustomize
except ImportError:
pass
#
# Remove sys.setdefaultencoding() so that users cannot change the
# encoding after initialization. The test for presence is needed when
# this module is run as a script, because this code is executed twice.
#
if hasattr(sys, "setdefaultencoding"):
del sys.setdefaultencoding
def _test(): def _test():
print "sys.path = [" print "sys.path = ["

199
Lib/test/test_site.py Normal file
View File

@ -0,0 +1,199 @@
"""Tests for 'site'.
Tests assume the initial paths in sys.path once the interpreter has begun
executing have not been removed.
"""
import unittest
from test.test_support import TestSkipped, run_unittest, TESTFN
import __builtin__
import os
import sys
import encodings
import tempfile
# Need to make sure to not import 'site' if someone specified ``-S`` at the
# command-line. Detect this by just making sure 'site' has not been imported
# already.
if "site" in sys.modules:
import site
else:
raise TestSkipped("importation of site.py suppressed")
class HelperFunctionsTests(unittest.TestCase):
"""Tests for helper functions.
The setting of the encoding (set using sys.setdefaultencoding) used by
the Unicode implementation is not tested.
"""
def setUp(self):
"""Save a copy of sys.path"""
self.sys_path = sys.path[:]
def tearDown(self):
"""Restore sys.path"""
sys.path = self.sys_path
def test_makepath(self):
# Test makepath() have an absolute path for its first return value
# and a case-normalized version of the absolute path for its
# second value.
path_parts = ("Beginning", "End")
original_dir = os.path.join(*path_parts)
abs_dir, norm_dir = site.makepath(*path_parts)
self.failUnlessEqual(os.path.abspath(original_dir), abs_dir)
if original_dir == os.path.normcase(original_dir):
self.failUnlessEqual(abs_dir, norm_dir)
else:
self.failUnlessEqual(os.path.normcase(abs_dir), norm_dir)
def test_init_pathinfo(self):
dir_set = site._init_pathinfo()
for entry in [site.makepath(path)[1] for path in sys.path
if path and os.path.isdir(path)]:
self.failUnless(entry in dir_set,
"%s from sys.path not found in set returned "
"by _init_pathinfo(): %s" % (entry, dir_set))
def test_addpackage(self):
# Make sure addpackage() imports if the line starts with 'import',
# otherwise add a directory combined from sitedir and 'name'.
# Must also skip comment lines.
dir_path, file_name, new_dir = createpth()
try:
site.addpackage(dir_path, file_name, set())
self.failUnless(site.makepath(os.path.join(dir_path, new_dir))[0] in
sys.path)
finally:
cleanuppth(dir_path, file_name, new_dir)
def test_addsitedir(self):
dir_path, file_name, new_dir = createpth()
try:
site.addsitedir(dir_path, set())
self.failUnless(site.makepath(os.path.join(dir_path, new_dir))[0] in
sys.path)
finally:
cleanuppth(dir_path, file_name, new_dir)
def createpth():
"""Create a temporary .pth file at the returned location and return the
directory where it was created, the pth file name, and the directory
specified in the pth file.
Make sure to delete the file when finished.
"""
pth_dirname = "__testdir__"
file_name = TESTFN + ".pth"
full_dirname = os.path.dirname(os.path.abspath(file_name))
FILE = file(os.path.join(full_dirname, file_name), 'w')
try:
print>>FILE, "#import @bad module name"
print>>FILE, ''
print>>FILE, "import os"
print>>FILE, pth_dirname
finally:
FILE.close()
os.mkdir(os.path.join(full_dirname, pth_dirname))
return full_dirname, file_name, pth_dirname
def cleanuppth(full_dirname, file_name, pth_dirname):
"""Clean up what createpth() made"""
os.remove(os.path.join(full_dirname, file_name))
os.rmdir(os.path.join(full_dirname, pth_dirname))
class ImportSideEffectTests(unittest.TestCase):
"""Test side-effects from importing 'site'."""
def setUp(self):
"""Make a copy of sys.path"""
self.sys_path = sys.path[:]
def tearDown(self):
"""Restore sys.path"""
sys.path = self.sys_path
def test_abs__file__(self):
# Make sure all imported modules have their __file__ attribute
# as an absolute path.
# Handled by abs__file__()
site.abs__file__()
for module in sys.modules.values():
try:
self.failUnless(os.path.isabs(module.__file__))
except AttributeError:
continue
def test_no_duplicate_paths(self):
# No duplicate paths should exist in sys.path
# Handled by removeduppaths()
site.removeduppaths()
seen_paths = set()
for path in sys.path:
self.failUnless(path not in seen_paths)
seen_paths.add(path)
def test_add_build_dir(self):
# Test that the build directory's Modules directory is used when it
# should be.
# XXX: implement
pass
def test_sitepackages(self):
# There should be a path that ends in site-packages
for path in sys.path:
if path.endswith("site-packages"):
break
else:
self.fail("'site-packages' directory missing'")
def test_setting_quit(self):
# 'quit' and 'exit' should be injected into __builtin__
self.failUnless(hasattr(__builtin__, "quit"))
self.failUnless(hasattr(__builtin__, "exit"))
def test_setting_copyright(self):
# 'copyright' and 'credits' should be in __builtin__
self.failUnless(hasattr(__builtin__, "copyright"))
self.failUnless(hasattr(__builtin__, "credits"))
def test_setting_help(self):
# 'help' should be set in __builtin__
self.failUnless(hasattr(__builtin__, "help"))
def test_aliasing_mbcs(self):
if sys.platform == "win32":
import locale
if locale.getdefaultlocale()[1].startswith('cp'):
for value in encodings.aliases.aliases.itervalues():
if value == "mbcs":
break
else:
self.fail("did not alias mbcs")
def test_setdefaultencoding_removed(self):
# Make sure sys.setdefaultencoding is gone
self.failUnless(not hasattr(sys, "setdefaultencoding"))
def test_sitecustomize_executed(self):
# If sitecustomize is available, it should have been imported.
if not sys.modules.has_key("sitecustomize"):
try:
import sitecustomize
except ImportError:
pass
else:
self.fail("sitecustomize not imported automatically")
def test_main():
run_unittest(HelperFunctionsTests, ImportSideEffectTests)
if __name__ == "__main__":
test_main()

View File

@ -322,6 +322,9 @@ Extension modules
Library Library
------- -------
- refactored site.py into functions. Also wrote regression tests for the
module.
- asyncore.loop now has repeat count parameter that defaults to infinity. - asyncore.loop now has repeat count parameter that defaults to infinity.
- The distutils sdist command now ignores all .svn directories, in - The distutils sdist command now ignores all .svn directories, in