如何将 Python 脚本转换为命令行应用程序

如何将 Python 脚本转换为命令行应用程序

2022-09-20 作者: xuzhiping 浏览: 112 次

摘要: 利用 Python 中的 scaffold 和 click,可以将一个简单的实用工具升级为一个成熟的命令行界面工具。 在我们的职业生涯中,一定写过、使用过以及看到过很多松散的脚本,也经常会希望在这些脚本中能够体会到更像命令行工具的感受。然而,将质量水平从一次...

利用 Python 中的 scaffold 和 click,可以将一个简单的实用工具升级为一个成熟的命令行界面工具。

Python 编程语言徽标和 Tux,Linux 的企鹅徽标

在我们的职业生涯中,一定写过、使用过以及看到过很多松散的脚本,也经常会希望在这些脚本中能够体会到更像命令行工具的感受。然而,将质量水平从一次性脚本提升到适当的工具到底有多难呢?事实证明,这在 Python 中其实并不难。

Scaffold

在本文中,将从 Python 的片段开始,将其放到 scaffold 模块中,并通过 click 以接受命令行参数来进行扩展。

#!/usr/bin/python

from glob import glob
from os.path import join, basename
from shutil import move
from datetime import datetime
from os import link, unlink

LATEST = 'latest.txt'
ARCHIVE = '/Users/mark/archive'
INCOMING = '/Users/mark/incoming'
TPATTERN = '%Y-%m-%d'

def transmogrify_filename(fname):
    bname = basename(fname)
    ts = datetime.now().strftime(TPATTERN)
    return '-'.join([ts, bname])

def set_current_latest(file):
    latest = join(ARCHIVE, LATEST)
    try:
        unlink(latest)
    except:
        pass
    link(file, latest)

def rotate_file(source):
    target = join(ARCHIVE, transmogrify_filename(source))
    move(source, target)
    set_current_latest(target)

def rotoscope():
    file_no = 0
    folder = join(INCOMING, '*.txt')
    print(f'Looking in {INCOMING}')
    for file in glob(folder):
        rotate_file(file)
        print(f'Rotated: {file}')
        file_no = file_no + 1
    print(f'Total files rotated: {file_no}')

if __name__ == '__main__':
    print('This is rotoscope 0.4.1. Bleep, bloop.')
    rotoscope()

本文中的所有非内联代码示例都引用了代码的特定版本,可在 https://codeberg.org/ofosos/rotoscope 找到特定版本的代码。该 repo 中的每一次提交都描述了本操作指南文章中的一些有意义步骤。

此代码段可执行以下操作:

  • 检查传入文件中指定的路径中是否有任何文本文件;
  • 如果存在,它将创建一个具有当前时间戳的新文件名,移动并存档;
  • 删除当前存档/最新存档,txt 链接并创建一个指向刚刚添加新文件。

利用 pyscaffold 创建应用程序

首先,需要安装 scaffoldclicktox Python 模块。

$ python3 -m pip install scaffold click tox

安装 scaffold 后,切换到示例 rotoscope 项目所在的目录,并执行以下命令:

$ putup rotoscope -p rotoscope \
--force --no-skeleton -n rotoscope \
-d 'Move some files around.' -l GLWT \
-u http://codeberg.org/ofosos/rotoscope \
--save-config --pre-commit --markdown

Pyscaffold 覆盖了 README.md,可在 Git 中进行恢复:

$ git checkout README.md

Pyscaffold 在文档层次结构中设置了一个完整的示例项目,除此外,Pyscaffold 还可以在项目中提供持续集成(CI)模板,具体如下:

  • 打包:项目现在已启用 PyPi,因此可将其下载到 repo 并安装;
  • 文档:目前有一个完整的文档文件夹层次结构,基于 Sphinx 并包括 readthedocs.org 构建器;
  • 测试:可与 tox 测试运行器一起使用,tests 文件夹包含运行基于 pytest 测试所需的所有样板文件;
  • 依赖关系管理:打包和测试基础设施都需要一种管理依赖关系的方法,setup.cfg 文件解决了此问题,并包含依赖项;
  • 预提交 hook:包括 Python 源格式化程序 “black” 和 “flake8” Python 样式检查器。

创建 Git 标记(例如,v0.2),该工具将其识别为可安装版本。在提交更改之前,先浏览自动生成的 setup.cfg,并根据用例对其进行编辑。对于此示例,可以修改 LICENSE 和项目描述。如果要将这些更改添加到 Git 的暂存区域,需要禁用预提交 hook 来提交,否则会遇到错误。

$ PRE_COMMIT_ALLOW_NO_CONFIG=1 git commit

如能有一个入口点可以进入此脚本,用户就可以从命令行调用,而现在只能通过查找 .py 文件并手动执行来运行。幸运的是,Python 的打包基础设施有一个很好的 "canned" 方式,可以使配置更改变得很容易,同时可将以下内容添加到 options.entry_points 中的 setup.cfg 部分 :

console_scripts =
    roto = rotoscope.rotoscope:rotoscope

此更改创建了名为 roto 的 shell 命令,能够使用该命令调用 rotoscope 脚本。使用 pip 安装 rotoscope 后,执行roto 命令。从 Pyscaffold 中免费获得所有的打包、测试和文档设置。

命令行工具

现在,有一些值硬编码到脚本中,作为命令参数会更方便。例如,INCOMING 常数作为命令行参数会更好。

首先,导入 Click 库。利用 Click 提供的命令对 rotoscope() 方法进行注释,并添加 Click 传递给 rotoscope 函数的参数。Click 提供了一组验证器,因此在参数中添加一个路径验证器。Click 还可以方便地使用函数的 here 字符串作为命令行文档的一部分。因此最终会得到以下方法签名:

@click.command()
@click.argument('incoming', type=click.Path(exists=True))
def rotoscope(incoming):
    """
    Rotoscope 0.4 - Bleep, blooop.
    Simple sample that move files.
    """

主要部分调用 rotoscope(),现在是一个 Click 命令, 因此不需要传递任何参数,选项也可以由 环境变量 自动填充。例如,将 ARCHIVE 常量更改为选项:

@click.option('archive', '--archive', default='/Users/mark/archive', envvar='ROTO_ARCHIVE', type=click.Path())

相同的路径验证程序再次应用。这一次,允许 Click 填充环境变量,如果环境没有提供任何内容,则默认为旧常量的值,同时具有彩色控制台输出、提示和子命令的功能,允许构建复杂的 CLI 工具,浏览 Click 文档会发现其更强大的功能。并能添加一些测试。

测试

Click 提供了一些关于使用 CLI runner 运行程序端到端测试 的建议,可以利用它来实现一个完整的测试(在 示例项目 中,测试位于 tests 文件夹中),测试位于测试类的方法中。大多数都非常接近其他 Python 项目中使用的约定,但也有一些细节,因为 rotoscope 使用 click。在该 test 方法中,创建了 CliRunner。测试使用此命令在隔离文件系统中运行命令, 测试创建传入和归档目录以及一个虚拟传入/测试, incoming/test.txt 文件调用 CliRunner,就如调用命令行应用程序一样。运行完成后,测试将检查隔离的文件系统,并验证 incoming 文件是否为空,以及 archive 包含两个文件(最新链接和归档文件)。

from os import listdir, mkdir
from click.testing import CliRunner
from rotoscope.rotoscope import rotoscope

class TestRotoscope:
    def test_roto_good(self, tmp_path):
        runner = CliRunner()

        with runner.isolated_filesystem(temp_dir=tmp_path) as td:
            mkdir("incoming")
            mkdir("archive")
            with open("incoming/test.txt", "w") as f:
                f.write("hello")

            result = runner.invoke(rotoscope, ["incoming", "--archive", "archive"])
            assert result.exit_code == 0

            print(td)
            incoming_f = listdir("incoming")
            archive_f = listdir("archive")
            assert len(incoming_f) == 0
            assert len(archive_f) == 2

要在控制台上执行这些测试,请在项目的根目录中运行 tox

在实现测试的过程中,在代码中发现了一个 bug。当进行 Click 转换时,rotoscope 只是断开了最新文件的链接,无论它是否存在。测试从一个新的文件系统(而不是主文件夹)开始,不过很快以失败告终。我们可以通过在一个完全隔离和自动化的测试环境中运行来防止这种错误,这将避免很多 “在我的机器上运行没有问题” 的情况。

Scaffolding 和模块

这样就完成了对可以使用 Scaffold 和进行高级操作的介绍 click。有很多可能性可以升级一个随意的 Python 脚本,甚至可以将简单实用程序变为成熟的 CLI 工具。

以上完成了对可以使用 Scaffoldclick 执行的高级操作的介绍。有很多可能性可以升级一个随意的Python脚本,甚至让你的简单实用程序成为成熟的CLI工具。

相关推荐

关注公众号
获取免费资源

随机推荐


Copyright © Since 2014. 开源地理空间基金会中文分会 吉ICP备05002032号

Powered by TorCMS

OSGeo 中国中心 邮件列表

问题讨论 : 要订阅或者退订列表,请点击 订阅

发言 : 请写信给: osgeo-china@lists.osgeo.org