Kaydet (Commit) 0096e262 authored tarafından Brett Cannon's avatar Brett Cannon

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

New regression test suite.
üst 642c8a11
...@@ -57,64 +57,104 @@ ImportError exception, it is silently ignored. ...@@ -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__)
# This ensures that the initial path provided by the interpreter contains except AttributeError:
# only absolute pathnames, even if we're running from the build directory. continue
L = []
_dirs_in_sys_path = {} def removeduppaths():
dir = dircase = None # sys.path may be empty at this point """ Remove duplicate entries from sys.path along with making them
for dir in sys.path: absolute"""
# Filter out duplicate paths (on case-insensitive file systems also # This ensures that the initial path provided by the interpreter contains
# if they only differ in case); turn relative paths into absolute # only absolute pathnames, even if we're running from the build directory.
# paths. L = []
dir, dircase = makepath(dir) known_paths = set()
if not dircase in _dirs_in_sys_path: for dir in sys.path:
L.append(dir) # Filter out duplicate paths (on case-insensitive file systems also
_dirs_in_sys_path[dircase] = 1 # if they only differ in case); turn relative paths into absolute
sys.path[:] = L # paths.
del dir, dircase, L dir, dircase = makepath(dir)
if not dircase in known_paths:
# Append ./build/lib.<platform> in case we're running in the build dir L.append(dir)
# (especially for Guido :-) known_paths.add(dircase)
sys.path[:] = L
return known_paths
# 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): ...@@ -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:
_dirs_in_sys_path = None
def addpackage(sitedir, name):
global _dirs_in_sys_path
if _dirs_in_sys_path is None:
_init_pathinfo()
reset = 1
else:
reset = 0
fullname = os.path.join(sitedir, name)
try:
f = open(fullname)
except IOError:
return
while 1:
dir = f.readline()
if not dir:
break
if dir[0] == '#':
continue
if dir.startswith("import"):
exec dir
continue
dir = dir.rstrip()
dir, dircase = makepath(sitedir, dir)
if not dircase in _dirs_in_sys_path and os.path.exists(dir):
sys.path.append(dir)
_dirs_in_sys_path[dircase] = 1
if reset: if reset:
_dirs_in_sys_path = None known_paths = None
return known_paths
prefixes = [sys.prefix]
sitedir = None # make sure sitedir is initialized because of later 'del' def addsitepackages(known_paths):
if sys.exec_prefix != sys.prefix: """Add site-packages (and possibly site-python) to sys.path"""
prefixes.append(sys.exec_prefix) prefixes = [sys.prefix]
for prefix in prefixes: if sys.exec_prefix != sys.prefix:
if prefix: prefixes.append(sys.exec_prefix)
if sys.platform in ('os2emx', 'riscos'): for prefix in prefixes:
sitedirs = [os.path.join(prefix, "Lib", "site-packages")] if prefix:
elif os.sep == '/': if sys.platform in ('os2emx', 'riscos'):
sitedirs = [os.path.join(prefix, sitedirs = [os.path.join(prefix, "Lib", "site-packages")]
"lib", elif os.sep == '/':
"python" + sys.version[:3], sitedirs = [os.path.join(prefix,
"site-packages"), "lib",
os.path.join(prefix, "lib", "site-python")] "python" + sys.version[:3],
else: "site-packages"),
sitedirs = [prefix, os.path.join(prefix, "lib", "site-packages")] os.path.join(prefix, "lib", "site-python")]
if sys.platform == 'darwin': else:
# for framework builds *only* we add the standard Apple sitedirs = [prefix, os.path.join(prefix, "lib", "site-packages")]
# locations. Currently only per-user, but /Library and if sys.platform == 'darwin':
# /Network/Library could be added too # for framework builds *only* we add the standard Apple
if 'Python.framework' in prefix: # locations. Currently only per-user, but /Library and
home = os.environ.get('HOME') # /Network/Library could be added too
if home: if 'Python.framework' in prefix:
sitedirs.append( home = os.environ.get('HOME')
os.path.join(home, if home:
'Library', sitedirs.append(
'Python', os.path.join(home,
sys.version[:3], 'Library',
'site-packages')) 'Python',
for sitedir in sitedirs: sys.version[:3],
if os.path.isdir(sitedir): 'site-packages'))
addsitedir(sitedir) for sitedir in sitedirs:
del prefix, sitedir if os.path.isdir(sitedir):
addsitedir(sitedir, known_paths)
_dirs_in_sys_path = None return 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': ...@@ -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: ...@@ -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: ...@@ -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'.
# This is a wrapper around pydoc.help (with a twist).
class _Helper(object):
class _Helper: """Define the built-in 'help'.
This is a wrapper around pydoc.help (with a twist).
"""
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: ...@@ -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()
# On Windows, some default encodings are not provided by Python, def aliasmbcs():
# while they are always available as "mbcs" in each locale. Make """On Windows, some default encodings are not provided by Python,
# them usable by aliasing to "mbcs" in such a case. 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': if sys.platform == 'win32':
import locale, codecs import locale, codecs
enc = locale.getdefaultlocale()[1] enc = locale.getdefaultlocale()[1]
if enc.startswith('cp'): # "cp***" ? if enc.startswith('cp'): # "cp***" ?
try: try:
codecs.lookup(enc) codecs.lookup(enc)
except LookupError: except LookupError:
import encodings import encodings
encodings._cache[enc] = encodings._unknown encodings._cache[enc] = encodings._unknown
encodings.aliases.aliases[enc] = 'mbcs' encodings.aliases.aliases[enc] = 'mbcs'
# Set the string encoding used by the Unicode implementation. The def setencoding():
# default is 'ascii', but if you're willing to experiment, you can """Set the string encoding used by the Unicode implementation. The
# change this. default is 'ascii', but if you're willing to experiment, you can
change this."""
encoding = "ascii" # Default value set by _PyUnicode_Init() encoding = "ascii" # Default value set by _PyUnicode_Init()
if 0:
if 0: # Enable to support locale aware default string encodings.
# Enable to support locale aware default string encodings. import locale
import locale loc = locale.getdefaultlocale()
loc = locale.getdefaultlocale() if loc[1]:
if loc[1]: encoding = loc[1]
encoding = loc[1] if 0:
# Enable to switch off string to Unicode coercion and implicit
if 0: # Unicode to string conversion.
# Enable to switch off string to Unicode coercion and implicit encoding = "undefined"
# Unicode to string conversion. if encoding != "ascii":
encoding = "undefined" # On Non-Unicode builds this will raise an AttributeError...
sys.setdefaultencoding(encoding) # Needs Python Unicode build !
if encoding != "ascii":
# On Non-Unicode builds this will raise an AttributeError...
sys.setdefaultencoding(encoding) # Needs Python Unicode build ! def execsitecustomize():
"""Run custom site specific code, if available."""
# try:
# Run custom site specific code, if available. import sitecustomize
# except ImportError:
try: pass
import sitecustomize
except ImportError:
pass def main():
abs__file__()
# paths_in_sys = removeduppaths()
# Remove sys.setdefaultencoding() so that users cannot change the if (os.name == "posix" and sys.path and
# encoding after initialization. The test for presence is needed when os.path.basename(sys.path[-1]) == "Modules"):
# this module is run as a script, because this code is executed twice. addbuilddir()
# paths_in_sys = addsitepackages(paths_in_sys)
if hasattr(sys, "setdefaultencoding"): if sys.platform == 'os2emx':
del sys.setdefaultencoding 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
main()
def _test(): def _test():
print "sys.path = [" print "sys.path = ["
......
"""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()
...@@ -322,6 +322,9 @@ Extension modules ...@@ -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
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment