自定义命令¶
借助 Python 和 Conan 公共 API 的强大功能,可以创建自己的 Conan 命令来满足自身需求。
位置和命名¶
所有自定义命令必须位于 [YOUR_CONAN_HOME]/extensions/commands/
文件夹中。如果您不知道 [YOUR_CONAN_HOME]
位于何处,可以运行 conan config home 来检查。
如果尚未创建 _commands_ 子目录,则必须创建它。这些自定义命令文件必须是 Python 文件,并以 cmd_[your_command_name].py
前缀开头。调用自定义命令的方式与任何其他现有的 Conan 命令相同:conan your_command_name。
作用域¶
可以有另一个文件夹层,以便将一些命令分组在同一主题下。
例如
| - [YOUR_CONAN_HOME]/extensions/commands/greet/
| - cmd_hello.py
| - cmd_bye.py
调用这些命令的方式略有不同:conan [topic_name]:your_command_name。按照前面的示例
$ conan greet:hello
$ conan greet:bye
注意
只能有一个文件夹层,因此使用类似 [YOUR_CONAN_HOME]/extensions/commands/topic1/topic2/cmd_command.py
的结构将不起作用
装饰器¶
conan_command(group=None, formatters=None)¶
将函数声明为新的 Conan 命令的主要装饰器。参数为
group
是在相同名称下声明的命令组的名称。此分组将出现在执行 conan -h 命令时。formatters
是一个类似 dict 的 Python 对象,其中key
是格式化程序名称,值是函数实例,其中将处理命令返回的信息。
import json
from conan.api.conan_api import ConanAPI
from conan.api.output import ConanOutput
from conan.cli.command import conan_command
def output_json(msg):
return json.dumps({"greet": msg})
@conan_command(group="Custom commands", formatters={"json": output_json})
def hello(conan_api: ConanAPI, parser, *args):
"""
Simple command to print "Hello World!" line
"""
msg = "Hello World!"
ConanOutput().info(msg)
return msg
重要
由 @conan_command(....)
装饰的函数必须与 Python 文件使用的后缀名称相同。例如,在前面的示例中,文件名是 cmd_hello.py
,并且被装饰的命令函数是 def hello(....)
。
conan_subcommand(formatters=None)¶
与 conan_command
类似,但此装饰器声明现有自定义命令的子命令。例如
from conan.api.conan_api import ConanAPI
from conan.api.output import ConanOutput
from conan.cli.command import conan_command, conan_subcommand
@conan_subcommand()
def hello_moon(conan_api, parser, subparser, *args):
"""
Sub-command of "hello" that prints "Hello Moon!" line
"""
ConanOutput().info("Hello Moon!")
@conan_command(group="Custom commands")
def hello(conan_api: ConanAPI, parser, *args):
"""
Simple command "hello"
"""
命令调用看起来像 conan hello moon。
注意
请注意,要声明子命令,需要一个空的 Python 函数充当主命令。
参数定义和解析¶
命令可以使用 argparse
Python 库定义自己的参数。
@conan_command(group='Creator')
def build(conan_api, parser, *args):
"""
Command help
"""
parser.add_argument("path", nargs="?", help='help for command')
...
args = parser.parse_args(*args)
# Use args.path
当存在子命令时,基本命令不能定义参数,只有子命令可以定义参数。如果所有子命令都有一组通用参数,则可以定义一个添加它们的函数。
@conan_command(group="MyGroup")
def mycommand(conan_api, parser, *args):
"""
Command help
"""
# Do not define arguments in the base command
pass
@conan_subcommand()
def mycommand_mysubcommand(conan_api: ConanAPI, parser, subparser, *args):
"""
Subcommand help
"""
# Arguments are added to "subparser"
subparser.add_argument("reference", help="Recipe reference or Package reference")
# You can add common args with your helper
# add_my_common_args(subparser)
# But parsing all of them happens to "parser"
args = parser.parse_args(*args)
# use args.reference
格式化程序¶
命令的返回值将作为参数传递给格式化程序。如果不同的格式化程序需要不同的参数,则方法是返回一个字典,并让格式化程序选择他们需要的参数。例如,graph info
命令使用多个格式化程序,例如
def format_graph_html(result):
graph = result["graph"]
conan_api = result["conan_api"]
...
def format_graph_info(result):
graph = result["graph"]
field_filter = result["field_filter"]
package_filter = result["package_filter"]
...
@conan_subcommand(formatters={"text": format_graph_info,
"html": format_graph_html,
"json": format_graph_json,
"dot": format_graph_dot})
def graph_info(conan_api, parser, subparser, *args):
...
return {"graph": deps_graph,
"field_filter": args.filter,
"package_filter": args.package_filter,
"conan_api": conan_api}
命令参数¶
这些是传递给任何自定义命令及其子命令函数的参数
from conan.cli.command import conan_command, conan_subcommand
@conan_subcommand()
def command_subcommand(conan_api, parser, subparser, *args):
"""
subcommand information. This info will appear on ``conan command subcommand -h``.
:param conan_api: <object conan.api.conan_api.ConanAPI> instance
:param parser: root <object argparse.ArgumentParser> instance (coming from main command)
:param subparser: <object argparse.ArgumentParser> instance for sub-command
:param args: ``list`` of all the arguments passed after sub-command call
:return: (optional) whatever is returned will be passed to formatters functions (if declared)
"""
# ...
@conan_command(group="Custom commands")
def command(conan_api, parser, *args):
"""
command information. This info will appear on ``conan command -h``.
:param conan_api: <object conan.api.conan_api.ConanAPI> instance
:param parser: root <object argparse.ArgumentParser> instance
:param args: ``list`` of all the arguments passed after command call
:return: (optional) whatever is returned will be passed to formatters functions (if declared)
"""
# ...
conan_api
:ConanAPI
类的实例。有关更多信息,请参阅 conan.api.conan_api.ConanAPI 部分parser
:Pythonargparse.ArgumentParser
类的根实例,供主命令函数使用。有关更多信息,请参阅 argparse 官方网站。subparser
(仅用于子命令):每个子命令函数的 Pythonargparse.ArgumentParser
类的子实例。*args
:通过命令行传递的所有参数的列表,以便在命令函数内部进行解析和使用。通常,它们会被解析为args = parser.parse_args(*args)
。例如,运行 conan mycommand arg1 arg2 arg3,命令函数将把它们作为 Python 列表接收,例如["arg1", "arg2", "arg3"]
。
另请参阅
您可以在 conan-extensions 存储库中查看更多 Conan 自定义命令示例 https://github.com/conan-io/conan-extensions