python - 无法显示带有点击入口点的 Python 二进制 PEX 版本

标签 python python-3.x python-packaging python-click python-pex

Python Click CLI 应用程序

当您使用 Click 库构建 Python CLI 应用程序时,您可以这样做:

@click.version_option()
def cli():
    '''
    Main Entry Point to Click Interface
    '''

能够做到这一点:

[user@host]$ clickapp --version

点击Packaged in pex

但是当我将其打包为 pex 文件时,我的 click 应用程序的所有其他参数、选项、命令、子命令都有效,--version 除外。

当我运行时(clickapp 现在是一个二进制可执行文件 pex 文件):

[user@host]$ ./clickapp --version

我收到以下错误:

Traceback (most recent call last):
  File "/~/clickapp/.bootstrap/pex/pex.py", line 446, in execute
  File "/~/clickapp/.bootstrap/pex/pex.py", line 378, in _wrap_coverage
  File "/~/clickapp/.bootstrap/pex/pex.py", line 409, in _wrap_profiling
  File "/~/clickapp/.bootstrap/pex/pex.py", line 508, in _execute
  File "/~/clickapp/.bootstrap/pex/pex.py", line 610, in execute_entry
  File "/~/clickapp/.bootstrap/pex/pex.py", line 626, in execute_pkg_resources
  File "/~/.pex/installed_wheels/7bcb1b5bf49ca3f89c348c338d33c04e3d59dfc1/click-7.1.2-py2.py3-none-any.whl/click/core.py", line 829, in __call__
    return self.main(*args, **kwargs)
  File "/~/.pex/installed_wheels/7bcb1b5bf49ca3f89c348c338d33c04e3d59dfc1/click-7.1.2-py2.py3-none-any.whl/click/core.py", line 781, in main
    with self.make_context(prog_name, args, **extra) as ctx:
  File "/~/.pex/installed_wheels/7bcb1b5bf49ca3f89c348c338d33c04e3d59dfc1/click-7.1.2-py2.py3-none-any.whl/click/core.py", line 700, in make_context
    self.parse_args(ctx, args)
  File "/~/.pex/installed_wheels/7bcb1b5bf49ca3f89c348c338d33c04e3d59dfc1/click-7.1.2-py2.py3-none-any.whl/click/core.py", line 1212, in parse_args
    rest = Command.parse_args(self, ctx, args)
  File "/~/.pex/installed_wheels/7bcb1b5bf49ca3f89c348c338d33c04e3d59dfc1/click-7.1.2-py2.py3-none-any.whl/click/core.py", line 1048, in parse_args
    value, args = param.handle_parse_result(ctx, opts, args)
  File "/~/.pex/installed_wheels/7bcb1b5bf49ca3f89c348c338d33c04e3d59dfc1/click-7.1.2-py2.py3-none-any.whl/click/core.py", line 1630, in handle_parse_result
    value = invoke_param_callback(self.callback, ctx, self, value)
  File "/~/.pex/installed_wheels/7bcb1b5bf49ca3f89c348c338d33c04e3d59dfc1/click-7.1.2-py2.py3-none-any.whl/click/core.py", line 123, in invoke_param_callback
    return callback(ctx, param, value)
  File "/~/.pex/installed_wheels/7bcb1b5bf49ca3f89c348c338d33c04e3d59dfc1/click-7.1.2-py2.py3-none-any.whl/click/decorators.py", line 295, in callback
    raise RuntimeError("Could not determine version")
RuntimeError: Could not determine version

详情

setup.py 文件:

from setuptools import setup, find_namespace_packages

setup(
    name='clickapp',
    version='0.1.3',
    author='Hamza M Zubair',
    packages=find_namespace_packages(),
    include_package_data=True,
    package_data={
        '': ['*.yaml'],
    },
    classifiers=[
        'Programming Language :: Python :: 3',
        'Operating System :: OS Independent',
        'Natural Language :: English',
        'License :: Other/Proprietary License',
    ],
    python_requires='>=3.6',
    install_requires=[
        'click',
        'pandas',
        'sqlalchemy',
        'jinjasql',
        'pyyaml',
        'joblib',
        'python-dateutil',
        'loguru',
        'pymysql',
        'xgboost',
        'sklearn',
        'wheel',
        'importlib-resources'
    ],
    entry_points='''
        [console_scripts]
        clickapp=clickapp.cli:cli
    ''',
)

用于创建pex 文件的命令:

[user@host]$ python setup.py bdist_pex --bdist-all

工具规范

我正在使用以下版本的库/包在不同的系统中构建和运行 pex 文件。目标机器只有 Python 没有库,因为 pex 文件不需要库/virtualenv 等。

Build Machine OS: CentOS Linux release 7.8.2003 (Core)  
Build Machine Python: 3.6.8  
setuptools: 51.0.0  
pex: 2.1.21  
click: 7.1.2  
Target Machine OS: CentOS Linus release 7.4.1708 (Core)
Target Machine Python: 3.6.8

我尝试了什么

  1. 我已经测试了我的 clickapp 的全部功能,并且所有其他参数和命令都运行良好。

即使这样也能正确显示我的 clickapp 的帮助。

[user@host]$ ./clickapp --help
  1. 我多次尝试重新构建包
  2. 我只在 Python3.6 中测试过这个。我没有尝试过不同的 python 版本,在源系统和目标系统中设置它有点困难。
  3. 当我删除 @click.version_option() 时,我收到错误:--version not implemented,这是预期的
  4. 我还没有在第二个目标系统上测试它,以防我当前目标服务器的某些特性导致错误

更多信息

为了帮助 SO 用户,我还应该提供哪些其他信息?

最佳答案

简答: setuptools 不见了。


看起来您点击了 v7.1.2。在该版本中,自动计算版本号的代码路径之一使用 pkg_resources 这是 setuptools 的顶级包:

                try:
                    import pkg_resources
                except ImportError:
                    pass
                else:
                    for dist in pkg_resources.working_set:
                        scripts = dist.get_entry_map().get("console_scripts") or {}
                        for _, entry_point in iteritems(scripts):
                            if entry_point.module_name == module:
                                ver = dist.version
                                break

-- https://github.com/pallets/click/blob/7.1.2/src/click/decorators.py#L283-L293

所以在某种程度上,点击 取决于setuptools。但是还有其他代码路径不需要 pkg_resources ,例如版本号可以在装饰器的参数中明确设置(如果我没记错的话):@click.version_option(version='1.2.3') ( doc ),在这种情况下 setuptools 不是强制依赖项。

在大多数情况下,它之所以有效是因为(出于巧合)setuptools 几乎总是预先安装的(例如在虚拟环境中)。但是“almost always”与“always”是不一样的,例如,绝对有可能拥有没有 setuptools 安装。这看起来就像是 pex 打包应用程序的情况。我相信 pex 可以创建干净的虚拟环境(即没有 pipsetuptoolswheel 或其他任何东西)。

由于 setuptools 未声明为依赖项,因此未安装,导入 pkg_resources 失败,查找版本字符串失败。

我会说这是 click 方面的失败。他们应该将 setuptools 声明为强制依赖项,或者至少声明为可选依赖项(在 extra 中)并正确记录。

一个可能的解决方法是将setuptools 添加为应用程序的直接依赖项:添加setuptoolsinstall_requires 的列表在你的setup.py .

请注意,据我所知,click v8 中的情况会发生变化。找出版本字符串将依赖于 importlib.metadata 来自标准库(作为 importlib_metadata 向后移植到 Python < 3.8)而不是 pkg_resources :

看样子,又要出问题了,估计还要出问题。因为点击仍然没有声明 importlib-metadata 作为旧 Python 版本的依赖项,其中 importlib.metadata 不在标准库中(<3.8),版本字符串查找将失败。

关于python - 无法显示带有点击入口点的 Python 二进制 PEX 版本,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65195692/

相关文章:

python - 使用 openpyxl 保存更改

python - 在 Python 中创建阈值编码的 ROC 图

python - 如何从 python 列表中只删除零位?

python - 为什么我无法使用python使用单个输入函数获得两个由空格分隔的整数

python - 如何使用 if __name__ ='__main__' block 在 Python3 中使用相对导入?

Python弱依赖项目的打包

python - 使用 Flask 匿名密码保护没有用户名的页面

python - 当没有更多元素可生成时,Python 生成器是否应该引发异常?

python - 在 Linux 上 pip 安装 UnicodeDecodeError

带有 entry_point 控制台脚本的 Python 包分发