首页 > 解决方案 > 如何在保留用户站点包的同时避免在 Python 的 sys.path 中有当前目录?

问题描述

用户可以在包含他们自己的 Python 脚本的目录中执行我的 Python 应用程序。如果这些脚本中的任何一个的名称与我的程序使用的库模块冲突(甚至与我的模块的名称冲突),那么一切都会因无用的错误消息而中断。

我知道 Python(自 3.4 起)有-I开关,这使得导入机器忽略当前目录(即""不会被添加到sys.path)。这几乎可以解决我的问题,但它也有忽略用户站点包的副作用。

我知道我可以sys.path在我的程序加载后进行调整,但如果用户(意外)隐藏了我的程序的主模块,这将无济于事。

我可以以某种方式让 Python 忽略当前目录但同时给我用户站点包吗?

标签: pythoncommand-line

解决方案


您可以通过调用重新启用用户站点路径site.addusersitepackages(None)None简单的意思是:检查现有sys.path定义以避免添加重复路径):

import site
import sys

if sys.flags.no_user_site and sys.prefix == sys.base_prefix:
    # python -s or python -I has disabled user sites, and this is not a virtualenv
    site.ENABLE_USER_SITE = True
    site.addusersitepackages(None)

快速演示:

>>> import site, sys, os.path
>>> user_site = os.path.expanduser(f"~/.local/lib/python{sys.version_info.major}.{sys.version_info.minor}/site-packages")
>>> user_site in sys.path
False
>>> sys.flags.no_user_site
1
>>> site.ENABLE_USER_SITE = True
>>> site.addusersitepackages(None)
>>> user_site in sys.path
True

虽然site.addusersitepackages() 没有记录,但它是一个关键组件,site.main()而且非常简单。如果你觉得你不能依赖它的持续存在,请像这样重新实现它:

import os.path
import site

def addusersitepackages():
    user_site = site.getusersitepackages()

    if site.ENABLE_USER_SITE and os.path.isdir(user_site):
        site.addsitedir(user_site)

这个版本完全省略了known_paths参数,只使用记录的site属性和函数site.getusersitepackages()site.ENABLE_USER_SITEsite.addsitedir().

或者,将上面的代码折叠到代码中以启用用户站点目录,包装为一个函数:

import os.path
import site
import sys

def ensure_usersitepackages():
    if sys.flags.no_user_site and sys.prefix == sys.base_prefix:
        # if python -s or python -I has disabled user sites, and this 
        # is not a virtualenv, explicitly enable user sites anyway.
        site.ENABLE_USER_SITE = True
        user_site = site.getusersitepackages()
        if os.path.isdir(user_site):
            site.addsitedir(user_site)


推荐阅读