3

Python 单击 CLI 应用程序

当您使用 Click 库构建 Python CLI 应用程序时,您可以执行以下操作:

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

能够做到这一点:

[user@host]$ clickapp --version

点击打包在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 用户,我还应该提供哪些其他信息?

4

1 回答 1

3

简短setuptools的回答:失踪。


看起来您已经单击 v7.1.2。在该版本中,自动计算版本号的代码路径之一是setuptoolspkg_resources的顶级包:

                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几乎总是预先安装的(例如在虚拟环境中)。但是“几乎总是”与“总是”不同,例如,绝对有可能拥有没有安装setuptools的环境(虚拟或非虚拟环境) 。这看起来正是在pex打包应用程序的情况下发生的情况。我相信pex创建干净的虚拟环境(即没有pipsetuptoolswheel或其他任何东西)。

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

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

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

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

从表面上看,它会再次引起问题,甚至可能更多。由于对于不在标准库 ( < 3.8 ) 中的旧 Python 版本, click仍然没有声明importlib-metadata为依赖项,因此版本字符串查找将失败。importlib.metadata

于 2020-12-08T10:43:17.267 回答