Kaydet (Commit) af748c3a authored tarafından Christian Heimes's avatar Christian Heimes

Implemented PEP 370

üst 323c40d4
...@@ -85,3 +85,51 @@ Note that for some non-Unix systems, ``sys.prefix`` and ``sys.exec_prefix`` are ...@@ -85,3 +85,51 @@ Note that for some non-Unix systems, ``sys.prefix`` and ``sys.exec_prefix`` are
empty, and the path manipulations are skipped; however the import of empty, and the path manipulations are skipped; however the import of
:mod:`sitecustomize` is still attempted. :mod:`sitecustomize` is still attempted.
.. data:: PREFIXES
A list of prefixes for site package directories
.. versionadded:: 2.6
.. data:: ENABLE_USER_SITE
Flag showing the status of the user site directory. True means the
user site directory is enabled and added to sys.path. When the flag
is None the user site directory is disabled for security reasons.
.. versionadded:: 2.6
.. data:: USER_SITE
Path to the user site directory for the current Python version or None
.. versionadded:: 2.6
.. data:: USER_BASE
Path to the base directory for user site directories
.. versionadded:: 2.6
.. envvar:: PYTHONNOUSERSITE
.. versionadded:: 2.6
.. envvar:: PYTHONUSERBASE
.. versionadded:: 2.6
.. function:: addsitedir(sitedir, known_paths=None)
Adds a directory to sys.path and processes its pth files.
XXX Update documentation
XXX document python -m site --user-base --user-site
...@@ -21,7 +21,7 @@ Command line ...@@ -21,7 +21,7 @@ Command line
When invoking Python, you may specify any of these options:: When invoking Python, you may specify any of these options::
python [-dEiOQStuUvxX3?] [-c command | -m module-name | script | - ] [args] python [-dEiOQsStuUvxX3?] [-c command | -m module-name | script | - ] [args]
The most common use case is, of course, a simple invocation of a script:: The most common use case is, of course, a simple invocation of a script::
...@@ -240,6 +240,17 @@ Miscellaneous options ...@@ -240,6 +240,17 @@ Miscellaneous options
:pep:`238` -- Changing the division operator :pep:`238` -- Changing the division operator
.. cmdoption:: -s
Don't add user site directory to sys.path
.. versionadded:: 2.6
.. seealso::
:pep:`370` -- Per user site-packages directory
.. cmdoption:: -S .. cmdoption:: -S
Disable the import of the module :mod:`site` and the site-dependent Disable the import of the module :mod:`site` and the site-dependent
...@@ -471,6 +482,28 @@ These environment variables influence Python's behavior. ...@@ -471,6 +482,28 @@ These environment variables influence Python's behavior.
.. versionadded:: 2.6 .. versionadded:: 2.6
.. envvar:: PYTHONNOUSERSITE
If this is set, Python won't add the user site directory to sys.path
.. versionadded:: 2.6
.. seealso::
:pep:`370` -- Per user site-packages directory
.. envvar:: PYTHONUSERBASE
Sets the base directory for the user site directory
.. versionadded:: 2.6
.. seealso::
:pep:`370` -- Per user site-packages directory
.. envvar:: PYTHONEXECUTABLE .. envvar:: PYTHONEXECUTABLE
If this environment variable is set, ``sys.argv[0]`` will be set to its If this environment variable is set, ``sys.argv[0]`` will be set to its
......
...@@ -19,6 +19,7 @@ PyAPI_DATA(int) Py_UnicodeFlag; ...@@ -19,6 +19,7 @@ PyAPI_DATA(int) Py_UnicodeFlag;
PyAPI_DATA(int) Py_IgnoreEnvironmentFlag; PyAPI_DATA(int) Py_IgnoreEnvironmentFlag;
PyAPI_DATA(int) Py_DivisionWarningFlag; PyAPI_DATA(int) Py_DivisionWarningFlag;
PyAPI_DATA(int) Py_DontWriteBytecodeFlag; PyAPI_DATA(int) Py_DontWriteBytecodeFlag;
PyAPI_DATA(int) Py_NoUserSiteDirectory;
/* _XXX Py_QnewFlag should go away in 3.0. It's true iff -Qnew is passed, /* _XXX Py_QnewFlag should go away in 3.0. It's true iff -Qnew is passed,
on the command line, and is used in 2.2 by ceval.c to make all "/" divisions on the command line, and is used in 2.2 by ceval.c to make all "/" divisions
true divisions (which they will be in 3.0). */ true divisions (which they will be in 3.0). */
......
...@@ -18,6 +18,9 @@ from distutils.file_util import write_file ...@@ -18,6 +18,9 @@ from distutils.file_util import write_file
from distutils.util import convert_path, subst_vars, change_root from distutils.util import convert_path, subst_vars, change_root
from distutils.util import get_platform from distutils.util import get_platform
from distutils.errors import DistutilsOptionError from distutils.errors import DistutilsOptionError
from site import USER_BASE
from site import USER_SITE
if sys.version < "2.2": if sys.version < "2.2":
WINDOWS_SCHEME = { WINDOWS_SCHEME = {
...@@ -51,7 +54,21 @@ INSTALL_SCHEMES = { ...@@ -51,7 +54,21 @@ INSTALL_SCHEMES = {
'scripts': '$base/bin', 'scripts': '$base/bin',
'data' : '$base', 'data' : '$base',
}, },
'unix_user': {
'purelib': '$usersite',
'platlib': '$usersite',
'headers': '$userbase/include/python$py_version_short/$dist_name',
'scripts': '$userbase/bin',
'data' : '$userbase',
},
'nt': WINDOWS_SCHEME, 'nt': WINDOWS_SCHEME,
'nt_user': {
'purelib': '$usersite',
'platlib': '$usersite',
'headers': '$userbase/Python$py_version_nodot/Include/$dist_name',
'scripts': '$userbase/Scripts',
'data' : '$userbase',
},
'mac': { 'mac': {
'purelib': '$base/Lib/site-packages', 'purelib': '$base/Lib/site-packages',
'platlib': '$base/Lib/site-packages', 'platlib': '$base/Lib/site-packages',
...@@ -59,13 +76,27 @@ INSTALL_SCHEMES = { ...@@ -59,13 +76,27 @@ INSTALL_SCHEMES = {
'scripts': '$base/Scripts', 'scripts': '$base/Scripts',
'data' : '$base', 'data' : '$base',
}, },
'mac_user': {
'purelib': '$usersite',
'platlib': '$usersite',
'headers': '$userbase/$py_version_short/include/$dist_name',
'scripts': '$userbase/bin',
'data' : '$userbase',
},
'os2': { 'os2': {
'purelib': '$base/Lib/site-packages', 'purelib': '$base/Lib/site-packages',
'platlib': '$base/Lib/site-packages', 'platlib': '$base/Lib/site-packages',
'headers': '$base/Include/$dist_name', 'headers': '$base/Include/$dist_name',
'scripts': '$base/Scripts', 'scripts': '$base/Scripts',
'data' : '$base', 'data' : '$base',
} },
'os2_home': {
'purelib': '$usersite',
'platlib': '$usersite',
'headers': '$userbase/include/python$py_version_short/$dist_name',
'scripts': '$userbase/bin',
'data' : '$userbase',
},
} }
# The keys to an installation scheme; if any new types of files are to be # The keys to an installation scheme; if any new types of files are to be
...@@ -86,6 +117,8 @@ class install (Command): ...@@ -86,6 +117,8 @@ class install (Command):
"(Unix only) prefix for platform-specific files"), "(Unix only) prefix for platform-specific files"),
('home=', None, ('home=', None,
"(Unix only) home directory to install under"), "(Unix only) home directory to install under"),
('user', None,
"install in user site-package '%s'" % USER_SITE),
# Or, just set the base director(y|ies) # Or, just set the base director(y|ies)
('install-base=', None, ('install-base=', None,
...@@ -137,7 +170,7 @@ class install (Command): ...@@ -137,7 +170,7 @@ class install (Command):
"filename in which to record list of installed files"), "filename in which to record list of installed files"),
] ]
boolean_options = ['compile', 'force', 'skip-build'] boolean_options = ['compile', 'force', 'skip-build', 'user']
negative_opt = {'no-compile' : 'compile'} negative_opt = {'no-compile' : 'compile'}
...@@ -148,6 +181,7 @@ class install (Command): ...@@ -148,6 +181,7 @@ class install (Command):
self.prefix = None self.prefix = None
self.exec_prefix = None self.exec_prefix = None
self.home = None self.home = None
self.user = 0
# These select only the installation base; it's up to the user to # These select only the installation base; it's up to the user to
# specify the installation scheme (currently, that means supplying # specify the installation scheme (currently, that means supplying
...@@ -166,6 +200,8 @@ class install (Command): ...@@ -166,6 +200,8 @@ class install (Command):
self.install_lib = None # set to either purelib or platlib self.install_lib = None # set to either purelib or platlib
self.install_scripts = None self.install_scripts = None
self.install_data = None self.install_data = None
self.install_userbase = USER_BASE
self.install_usersite = USER_SITE
self.compile = None self.compile = None
self.optimize = None self.optimize = None
...@@ -241,6 +277,11 @@ class install (Command): ...@@ -241,6 +277,11 @@ class install (Command):
raise DistutilsOptionError, \ raise DistutilsOptionError, \
"must supply either home or prefix/exec-prefix -- not both" "must supply either home or prefix/exec-prefix -- not both"
if self.user and (self.prefix or self.exec_prefix or self.home or
self.install_base or self.install_platbase):
raise DistutilsOptionError("can't combine user with with prefix/"
"exec_prefix/home or install_(plat)base")
# Next, stuff that's wrong (or dubious) only on certain platforms. # Next, stuff that's wrong (or dubious) only on certain platforms.
if os.name != "posix": if os.name != "posix":
if self.exec_prefix: if self.exec_prefix:
...@@ -276,10 +317,13 @@ class install (Command): ...@@ -276,10 +317,13 @@ class install (Command):
'dist_fullname': self.distribution.get_fullname(), 'dist_fullname': self.distribution.get_fullname(),
'py_version': py_version, 'py_version': py_version,
'py_version_short': py_version[0:3], 'py_version_short': py_version[0:3],
'py_version_nodot': py_version[0] + py_version[2],
'sys_prefix': prefix, 'sys_prefix': prefix,
'prefix': prefix, 'prefix': prefix,
'sys_exec_prefix': exec_prefix, 'sys_exec_prefix': exec_prefix,
'exec_prefix': exec_prefix, 'exec_prefix': exec_prefix,
'userbase': self.install_userbase,
'usersite': self.install_usersite,
} }
self.expand_basedirs() self.expand_basedirs()
...@@ -301,6 +345,10 @@ class install (Command): ...@@ -301,6 +345,10 @@ class install (Command):
self.dump_dirs("post-expand_dirs()") self.dump_dirs("post-expand_dirs()")
# Create directories in the home dir:
if self.user:
self.create_home_path()
# Pick the actual directory to install all modules to: either # Pick the actual directory to install all modules to: either
# install_purelib or install_platlib, depending on whether this # install_purelib or install_platlib, depending on whether this
# module distribution is pure or not. Of course, if the user # module distribution is pure or not. Of course, if the user
...@@ -315,7 +363,8 @@ class install (Command): ...@@ -315,7 +363,8 @@ class install (Command):
# Convert directories from Unix /-separated syntax to the local # Convert directories from Unix /-separated syntax to the local
# convention. # convention.
self.convert_paths('lib', 'purelib', 'platlib', self.convert_paths('lib', 'purelib', 'platlib',
'scripts', 'data', 'headers') 'scripts', 'data', 'headers',
'userbase', 'usersite')
# Well, we're not actually fully completely finalized yet: we still # Well, we're not actually fully completely finalized yet: we still
# have to deal with 'extra_path', which is the hack for allowing # have to deal with 'extra_path', which is the hack for allowing
...@@ -376,7 +425,13 @@ class install (Command): ...@@ -376,7 +425,13 @@ class install (Command):
"installation scheme is incomplete") "installation scheme is incomplete")
return return
if self.home is not None: if self.user:
if self.install_userbase is None:
raise DistutilsPlatformError(
"User base directory is not specified")
self.install_base = self.install_platbase = self.install_userbase
self.select_scheme("unix_user")
elif self.home is not None:
self.install_base = self.install_platbase = self.home self.install_base = self.install_platbase = self.home
self.select_scheme("unix_home") self.select_scheme("unix_home")
else: else:
...@@ -401,7 +456,13 @@ class install (Command): ...@@ -401,7 +456,13 @@ class install (Command):
def finalize_other (self): # Windows and Mac OS for now def finalize_other (self): # Windows and Mac OS for now
if self.home is not None: if self.user:
if self.install_userbase is None:
raise DistutilsPlatformError(
"User base directory is not specified")
self.install_base = self.install_platbase = self.install_userbase
self.select_scheme(os.name + "_user")
elif self.home is not None:
self.install_base = self.install_platbase = self.home self.install_base = self.install_platbase = self.home
self.select_scheme("unix_home") self.select_scheme("unix_home")
else: else:
...@@ -431,7 +492,7 @@ class install (Command): ...@@ -431,7 +492,7 @@ class install (Command):
for attr in attrs: for attr in attrs:
val = getattr(self, attr) val = getattr(self, attr)
if val is not None: if val is not None:
if os.name == 'posix': if os.name == 'posix' or os.name == 'nt':
val = os.path.expanduser(val) val = os.path.expanduser(val)
val = subst_vars(val, self.config_vars) val = subst_vars(val, self.config_vars)
setattr(self, attr, val) setattr(self, attr, val)
...@@ -496,6 +557,16 @@ class install (Command): ...@@ -496,6 +557,16 @@ class install (Command):
attr = "install_" + name attr = "install_" + name
setattr(self, attr, change_root(self.root, getattr(self, attr))) setattr(self, attr, change_root(self.root, getattr(self, attr)))
def create_home_path(self):
"""Create directories under ~
"""
if not self.user:
return
home = convert_path(os.path.expanduser("~"))
for name, path in self.config_vars.iteritems():
if path.startswith(home) and not os.path.isdir(path):
self.debug_print("os.makedirs('%s', 0700)" % path)
os.makedirs(path, 0700)
# -- Command execution methods ------------------------------------- # -- Command execution methods -------------------------------------
......
This diff is collapsed.
...@@ -10,6 +10,7 @@ import __builtin__ ...@@ -10,6 +10,7 @@ import __builtin__
import os import os
import sys import sys
import encodings import encodings
import subprocess
# Need to make sure to not import 'site' if someone specified ``-S`` at the # 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 # command-line. Detect this by just making sure 'site' has not been imported
# already. # already.
...@@ -18,6 +19,11 @@ if "site" in sys.modules: ...@@ -18,6 +19,11 @@ if "site" in sys.modules:
else: else:
raise TestSkipped("importation of site.py suppressed") raise TestSkipped("importation of site.py suppressed")
if not os.path.isdir(site.USER_SITE):
# need to add user site directory for tests
os.makedirs(site.USER_SITE)
site.addsitedir(site.USER_SITE)
class HelperFunctionsTests(unittest.TestCase): class HelperFunctionsTests(unittest.TestCase):
"""Tests for helper functions. """Tests for helper functions.
...@@ -30,7 +36,7 @@ class HelperFunctionsTests(unittest.TestCase): ...@@ -30,7 +36,7 @@ class HelperFunctionsTests(unittest.TestCase):
"""Save a copy of sys.path""" """Save a copy of sys.path"""
self.sys_path = sys.path[:] self.sys_path = sys.path[:]
def tearDown(self):
"""Restore sys.path""" """Restore sys.path"""
sys.path = self.sys_path sys.path = self.sys_path
...@@ -90,6 +96,33 @@ class HelperFunctionsTests(unittest.TestCase): ...@@ -90,6 +96,33 @@ class HelperFunctionsTests(unittest.TestCase):
finally: finally:
pth_file.cleanup() pth_file.cleanup()
def test_s_option(self):
usersite = site.USER_SITE
self.assert_(usersite in sys.path)
rc = subprocess.call([sys.executable, '-c',
'import sys; sys.exit("%s" in sys.path)' % usersite])
self.assertEqual(rc, 1)
rc = subprocess.call([sys.executable, '-s', '-c',
'import sys; sys.exit("%s" in sys.path)' % usersite])
self.assertEqual(rc, 0)
env = os.environ.copy()
env["PYTHONNOUSERSITE"] = "1"
rc = subprocess.call([sys.executable, '-c',
'import sys; sys.exit("%s" in sys.path)' % usersite],
env=env)
self.assertEqual(rc, 0)
env = os.environ.copy()
env["PYTHONUSERBASE"] = "/tmp"
rc = subprocess.call([sys.executable, '-c',
'import sys, site; sys.exit(site.USER_BASE.startswith("/tmp"))'],
env=env)
self.assertEqual(rc, 1)
class PthFile(object): class PthFile(object):
"""Helper class for handling testing of .pth files""" """Helper class for handling testing of .pth files"""
......
...@@ -20,6 +20,8 @@ Core and builtins ...@@ -20,6 +20,8 @@ Core and builtins
- Patch #2617: Reserved -J and -X arguments for Jython, IronPython and other - Patch #2617: Reserved -J and -X arguments for Jython, IronPython and other
implementations of Python. implementations of Python.
- Implemented PEP 370: Per user site-packages directory
Extension Modules Extension Modules
----------------- -----------------
......
...@@ -40,7 +40,7 @@ static char **orig_argv; ...@@ -40,7 +40,7 @@ static char **orig_argv;
static int orig_argc; static int orig_argc;
/* command line options */ /* command line options */
#define BASE_OPTS "3bBc:dEhiJm:OQ:StuUvVW:xX?" #define BASE_OPTS "3bBc:dEhiJm:OQ:sStuUvVW:xX?"
#ifndef RISCOS #ifndef RISCOS
#define PROGRAM_OPTS BASE_OPTS #define PROGRAM_OPTS BASE_OPTS
...@@ -72,6 +72,7 @@ static char *usage_2 = "\ ...@@ -72,6 +72,7 @@ static char *usage_2 = "\
-O : optimize generated bytecode slightly; also PYTHONOPTIMIZE=x\n\ -O : optimize generated bytecode slightly; also PYTHONOPTIMIZE=x\n\
-OO : remove doc-strings in addition to the -O optimizations\n\ -OO : remove doc-strings in addition to the -O optimizations\n\
-Q arg : division options: -Qold (default), -Qwarn, -Qwarnall, -Qnew\n\ -Q arg : division options: -Qold (default), -Qwarn, -Qwarnall, -Qnew\n\
-s : don't add user site directory to sys.path; also PYTHONNOUSERSITE\n\
-S : don't imply 'import site' on initialization\n\ -S : don't imply 'import site' on initialization\n\
-t : issue warnings about inconsistent tab usage (-tt: issue errors)\n\ -t : issue warnings about inconsistent tab usage (-tt: issue errors)\n\
"; ";
...@@ -359,6 +360,10 @@ Py_Main(int argc, char **argv) ...@@ -359,6 +360,10 @@ Py_Main(int argc, char **argv)
Py_DontWriteBytecodeFlag++; Py_DontWriteBytecodeFlag++;
break; break;
case 's':
Py_NoUserSiteDirectory++;
break;
case 'S': case 'S':
Py_NoSiteFlag++; Py_NoSiteFlag++;
break; break;
...@@ -431,6 +436,10 @@ Py_Main(int argc, char **argv) ...@@ -431,6 +436,10 @@ Py_Main(int argc, char **argv)
(p = Py_GETENV("PYTHONUNBUFFERED")) && *p != '\0') (p = Py_GETENV("PYTHONUNBUFFERED")) && *p != '\0')
unbuffered = 1; unbuffered = 1;
if (!Py_NoUserSiteDirectory &&
(p = Py_GETENV("PYTHONNOUSERSITE")) && *p != '\0')
Py_NoUserSiteDirectory = 1;
if (command == NULL && module == NULL && _PyOS_optind < argc && if (command == NULL && module == NULL && _PyOS_optind < argc &&
strcmp(argv[_PyOS_optind], "-") != 0) strcmp(argv[_PyOS_optind], "-") != 0)
{ {
......
...@@ -82,6 +82,7 @@ int Py_IgnoreEnvironmentFlag; /* e.g. PYTHONPATH, PYTHONHOME */ ...@@ -82,6 +82,7 @@ int Py_IgnoreEnvironmentFlag; /* e.g. PYTHONPATH, PYTHONHOME */
on the command line, and is used in 2.2 by ceval.c to make all "/" divisions on the command line, and is used in 2.2 by ceval.c to make all "/" divisions
true divisions (which they will be in 2.3). */ true divisions (which they will be in 2.3). */
int _Py_QnewFlag = 0; int _Py_QnewFlag = 0;
int Py_NoUserSiteDirectory = 0; /* for -s and site.py */
/* PyModule_GetWarningsModule is no longer necessary as of 2.6 /* PyModule_GetWarningsModule is no longer necessary as of 2.6
since _warnings is builtin. This API should not be used. */ since _warnings is builtin. This API should not be used. */
......
...@@ -1101,7 +1101,7 @@ svnversion_init(void) ...@@ -1101,7 +1101,7 @@ svnversion_init(void)
shortbranch[len] = '\0'; shortbranch[len] = '\0';
} }
else { else {
Py_FatalError("bad HeadURL"); Py_FatalError("bad HeadURL");
return; return;
} }
...@@ -1158,7 +1158,7 @@ static PyStructSequence_Field flags_fields[] = { ...@@ -1158,7 +1158,7 @@ static PyStructSequence_Field flags_fields[] = {
{"interactive", "-i"}, {"interactive", "-i"},
{"optimize", "-O or -OO"}, {"optimize", "-O or -OO"},
{"dont_write_bytecode", "-B"}, {"dont_write_bytecode", "-B"},
/* {"no_user_site", "-s"}, */ {"no_user_site", "-s"},
{"no_site", "-S"}, {"no_site", "-S"},
{"ignore_environment", "-E"}, {"ignore_environment", "-E"},
{"tabcheck", "-t or -tt"}, {"tabcheck", "-t or -tt"},
...@@ -1178,9 +1178,9 @@ static PyStructSequence_Desc flags_desc = { ...@@ -1178,9 +1178,9 @@ static PyStructSequence_Desc flags_desc = {
flags__doc__, /* doc */ flags__doc__, /* doc */
flags_fields, /* fields */ flags_fields, /* fields */
#ifdef RISCOS #ifdef RISCOS
14 15
#else #else
13 14
#endif #endif
}; };
...@@ -1205,7 +1205,7 @@ make_flags(void) ...@@ -1205,7 +1205,7 @@ make_flags(void)
SetFlag(Py_InteractiveFlag); SetFlag(Py_InteractiveFlag);
SetFlag(Py_OptimizeFlag); SetFlag(Py_OptimizeFlag);
SetFlag(Py_DontWriteBytecodeFlag); SetFlag(Py_DontWriteBytecodeFlag);
/* SetFlag(Py_NoUserSiteDirectory); */ SetFlag(Py_NoUserSiteDirectory);
SetFlag(Py_NoSiteFlag); SetFlag(Py_NoSiteFlag);
SetFlag(Py_IgnoreEnvironmentFlag); SetFlag(Py_IgnoreEnvironmentFlag);
SetFlag(Py_TabcheckFlag); SetFlag(Py_TabcheckFlag);
......
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