自定义命令:清理旧的配方和包修订版本

注意

这主要是一个示例命令。内置的 conan remove *#!latest 语法(意思是“除了最新的所有修订版本”)可能足以满足此用例,而无需此自定义命令。

请首先克隆源代码以重新创建此项目。你可以在 GitHub 上的 examples2 存储库中找到它们

$ git clone https://github.com/conan-io/examples2.git
$ cd examples2/examples/extensions/commands/clean

在本示例中,我们将了解如何创建/使用自定义命令:conan clean。它从本地缓存或远程仓库中删除每个配方及其包修订版本,但最新的配方版本的最新包修订版本除外。

注意

为了更好地理解此示例,强烈建议先阅读自定义命令参考

找到命令

将命令文件 cmd_clean.py 复制到你的 [YOUR_CONAN_HOME]/extensions/commands/ 文件夹中(如果不存在,则创建它)。如果你不知道 [YOUR_CONAN_HOME] 的位置,可以运行 conan config home 来检查它。

运行它

现在,你应该能够在命令提示符中看到新命令

$ conan -h
...
Custom commands
clean        Deletes (from local cache or remotes) all recipe and package revisions but the
               latest package revision from the latest recipe revision.

$ conan clean -h
usage: conan clean [-h] [-r REMOTE] [--force]

Deletes (from local cache or remotes) all recipe and package revisions but
the latest package revision from the latest recipe revision.

optional arguments:
  -h, --help            show this help message and exit
  -r REMOTE, --remote REMOTE
                        Will remove from the specified remote
  --force               Remove without requesting a confirmation

最后,如果执行 conan clean

$ conan clean
Do you want to remove all the recipes revisions and their packages ones, except the latest package revision from the latest recipe one? (yes/no): yes
other/1.0
Removed package revision: other/1.0#31da245c3399e4124e39bd4f77b5261f:da39a3ee5e6b4b0d3255bfef95601890afd80709#a16985deb2e1aa73a8480faad22b722c [Local cache]
Removed recipe revision: other/1.0#721995a35b1a8d840ce634ea1ac71161 and all its package revisions [Local cache]
hello/1.0
Removed package revision: hello/1.0#9a77cdcff3a539b5b077dd811b2ae3b0:da39a3ee5e6b4b0d3255bfef95601890afd80709#cee90a74944125e7e9b4f74210bfec3f [Local cache]
Removed package revision: hello/1.0#9a77cdcff3a539b5b077dd811b2ae3b0:da39a3ee5e6b4b0d3255bfef95601890afd80709#7cddd50952de9935d6c3b5b676a34c48 [Local cache]
libcxx/0.1

如果再次运行它,则不应发生任何事情

$ conan clean
Do you want to remove all the recipes revisions and their packages ones, except the latest package revision from the latest recipe one? (yes/no): yes
other/1.0
hello/1.0
libcxx/0.1

代码导览

conan clean 命令具有以下代码

cmd_clean.py
from conan.api.conan_api import ConanAPI
from conan.api.output import ConanOutput, Color
from conan.cli.command import OnceArgument, conan_command


recipe_color = Color.BRIGHT_BLUE
removed_color = Color.BRIGHT_YELLOW


@conan_command(group="Custom commands")
def clean(conan_api: ConanAPI, parser, *args):
    """
    Deletes (from local cache or remotes) all recipe and package revisions but
    the latest package revision from the latest recipe revision.
    """
    parser.add_argument('-r', '--remote', action=OnceArgument,
                        help='Will remove from the specified remote')
    args = parser.parse_args(*args)

    out = ConanOutput()
    remote = conan_api.remotes.get(args.remote) if args.remote else None
    output_remote = remote or "Local cache"

    # Getting all the recipes
    recipes = conan_api.search.recipes("*/*", remote=remote)
    for recipe in recipes:
        out.writeln(f"{str(recipe)}", fg=recipe_color)
        all_rrevs = conan_api.list.recipe_revisions(recipe, remote=remote)
        latest_rrev = all_rrevs[0] if all_rrevs else None
        for rrev in all_rrevs:
            if rrev != latest_rrev:
                conan_api.remove.recipe(rrev, remote=remote)
                out.writeln(f"Removed recipe revision: {rrev.repr_notime()} "
                            f"and all its package revisions [{output_remote}]", fg=removed_color)
            else:
                packages = conan_api.list.packages_configurations(rrev, remote=remote)
                for package_ref in packages:
                    all_prevs = conan_api.list.package_revisions(package_ref, remote=remote)
                    latest_prev = all_prevs[0] if all_prevs else None
                    for prev in all_prevs:
                    if prev != latest_prev:
                        conan_api.remove.package(prev, remote=remote)
                        out.writeln(f"Removed package revision: {prev.repr_notime()} [{output_remote}]", fg=removed_color)

让我们分析最重要的部分。

parser

parser 参数是 Python 命令行解析 argparse.ArgumentParser 的实例,因此如果你想了解有关其 API 的更多信息,请访问 其官方网站

用户输出

ConanOutput():用于管理用户输出的类。在此示例中,我们仅使用 out.writeln(message, fg=None, bg=None),其中 fg 是字体前景色,bg 是字体背景色。除此之外,你还有一些预定义的方法,如 out.info()out.success()out.error() 等。

Conan 公共 API

警告

此功能是实验性的,可能会有重大更改。有关更多信息,请参见 Conan 稳定性 部分。

本示例最重要的部分是通过 conan_api 参数使用 Conan API。这些是在此自定义命令中使用的一些示例

conan_api.remotes.get(args.remote)
conan_api.search.recipes("*/*", remote=remote)
conan_api.list.recipe_revisions(recipe, remote=remote)
conan_api.remove.recipe(rrev, remote=remote)
conan_api.list.packages_configurations(rrev, remote=remote)
conan_api.list.package_revisions(package_ref, remote=remote)
conan_api.remove.package(prev, remote=remote)
  • conan_api.remotes.get(...)[RemotesAPI] 返回给定远程名称的 RemoteRegistry。

  • conan_api.search.recipes(...)[SearchAPI] 返回与给定模式匹配的所有配方的列表。

  • conan_api.list.recipe_revisions(...)[ListAPI] 返回给定配方参考的所有配方修订版本的列表。

  • conan_api.list.packages_configurations(...)[ListAPI] 返回配方修订版本的不同配置(package_id)的列表。

  • conan_api.list.package_revisions(...)[ListAPI] 返回给定配方修订版本的包修订版本的列表。

  • conan_api.remove.recipe(...)[RemoveAPI] 删除给定的配方修订版本。

  • conan_api.remove.package(...)[RemoveAPI] 删除给定的包修订版本。

除此之外,以下几行值得特别关注

all_rrevs = conan_api.list.recipe_revisions(recipe, remote=remote)
latest_rrev = all_rrevs[0] if all_rrevs else None

...

packages = conan_api.list.packages_configurations(rrev, remote=remote)

...

all_prevs = conan_api.list.package_revisions(package_ref, remote=remote)
latest_prev = all_prevs[0] if all_prevs else None

基本上,这些 API 调用分别返回配方修订版本和包修订版本的列表,但我们将第一个元素保存为最新版本,因为这些调用始终获取排序的列表。

如果你想了解有关 Conan API 的更多信息,请访问 ConanAPI 部分