切面

报告问题 查看源代码 每夜 build · 7.4 .

本页面介绍了使用方面的基础知识和好处,并提供了简单和高级示例。

通过方面,您可以使用其他信息和操作来增强构建依赖项图。切面可能有用的一些典型场景:

  • 集成了 Bazel 的 IDE 可以使用 aspect 收集有关项目的信息。
  • 代码生成工具可以利用方面以与目标无关的方式在其输入上执行操作。例如,BUILD 文件可以指定 protobuf 库定义的层次结构,而特定于语言的规则可以使用切面来附加为特定语言生成 protobuf 支持代码的操作。

宽高比基础知识

BUILD 文件提供项目源代码的说明:哪些源文件属于项目、应根据这些文件构建哪些工件(目标)、这些文件之间的依赖关系等。Bazel 会使用这些信息执行构建,也就是说,它会确定生成工件(例如运行编译器或链接器)并执行这些操作所需的一系列操作。为此,Bazel 会在目标之间构建一个依赖关系图,并访问此图表来收集这些操作。

请参考以下 BUILD 文件:

java_library(name = 'W', ...)
java_library(name = 'Y', deps = [':W'], ...)
java_library(name = 'Z', deps = [':W'], ...)
java_library(name = 'Q', ...)
java_library(name = 'T', deps = [':Q'], ...)
java_library(name = 'X', deps = [':Y',':Z'], runtime_deps = [':T'], ...)

BUILD 文件定义了下图所示的依赖项图:

构建图表

图 1. BUILD 文件依赖关系图。

Bazel 会通过为上述示例中的每个目标调用相应规则(在本例中为“java_library”)的实现函数来分析此依赖项图。规则实现函数会生成用于构建工件(例如 .jar 文件)的操作,并将信息(例如这些工件的位置和名称)传递给 provider 中这些目标的反向依赖项。

与规则类似,方面具有用于生成操作并返回提供程序的实现函数。不过,它们的强大功能来自为它们构建依赖关系图的方式。一个方面具有实现以及它沿途传播的所有属性的列表。假设有一个沿着名为“deps”的属性传播的方面 A。此方面可以应用于目标 X,从而生成方面应用节点 A(X)。在应用过程中,A 会递归应用于 X 在其“deps”属性中引用的所有目标(A 传播列表中的所有属性)。

因此,将 A 方面应用于目标 X 这一单个操作会产生目标原始依赖项图的“阴影图”,如下图所示:

使用 Aspect 构建图表

图 2. 使用方面构建图表。

只有沿着传播集的属性的边会被阴影,因此在本例中,runtime_deps 边不会被阴影。然后,系统会对阴影图中的所有节点调用一个方面实现函数,这与对原始图的节点调用规则实现类似。

简单示例

此示例演示了如何递归输出规则及其具有 deps 属性的所有依赖项的源文件。该图展示了切面实现、切面定义以及如何从 Bazel 命令行调用切面。

def _print_aspect_impl(target, ctx):
    # Make sure the rule has a srcs attribute.
    if hasattr(ctx.rule.attr, 'srcs'):
        # Iterate through the files that make up the sources and
        # print their paths.
        for src in ctx.rule.attr.srcs:
            for f in src.files.to_list():
                print(f.path)
    return []

print_aspect = aspect(
    implementation = _print_aspect_impl,
    attr_aspects = ['deps'],
)

我们将该示例拆分为各个部分,并分别对其进行检查。

方面定义

print_aspect = aspect(
    implementation = _print_aspect_impl,
    attr_aspects = ['deps'],
)

切面定义与规则定义类似,是使用 aspect 函数定义的。

与规则一样,一个方面也有一个实现函数,在本例中为 _print_aspect_impl

attr_aspects 是切面传播的规则属性列表。在这种情况下,该方面将沿其应用到的规则的 deps 属性传播。

attr_aspects 的另一个常见参数是 ['*'],它会将相应方面传播到规则的所有属性。

方面实现

def _print_aspect_impl(target, ctx):
    # Make sure the rule has a srcs attribute.
    if hasattr(ctx.rule.attr, 'srcs'):
        # Iterate through the files that make up the sources and
        # print their paths.
        for src in ctx.rule.attr.srcs:
            for f in src.files.to_list():
                print(f.path)
    return []

方面实现函数类似于规则实现函数。它们会返回提供程序,可以生成操作,并接受两个参数:

  • target:要应用该方面到哪个目标
  • ctxctx 对象,可用于访问属性并生成输出和操作。

实现函数可以通过 ctx.rule.attr 访问目标规则的属性。它可以检查应用于其的目标(通过 target 参数)提供的提供程序。

必须提供方面才能返回提供商列表。在此示例中,此维度未提供任何内容,因此会返回一个空列表。

使用命令行调用切面

应用方面最简单的方法是在命令行中使用 --aspects 参数。假设上述切面在名为 print.bzl 的文件中定义,如下所示:

bazel build //MyExample:example --aspects print.bzl%print_aspect

print_aspect 应用于目标 example 以及可通过 deps 属性递归访问的所有目标规则。

--aspects 标志接受一个参数,该参数是采用 <extension file label>%<aspect top-level name> 格式的宽高比规范。

高级示例

以下示例演示了如何使用目标规则中的切面来统计目标中的文件数量,并可能按扩展名对文件进行过滤。该示例展示了如何使用提供程序返回值、如何使用参数将实参传递到切面实现,以及如何从规则调用切面。

file_count.bzl 文件:

FileCountInfo = provider(
    fields = {
        'count' : 'number of files'
    }
)

def _file_count_aspect_impl(target, ctx):
    count = 0
    # Make sure the rule has a srcs attribute.
    if hasattr(ctx.rule.attr, 'srcs'):
        # Iterate through the sources counting files
        for src in ctx.rule.attr.srcs:
            for f in src.files.to_list():
                if ctx.attr.extension == '*' or ctx.attr.extension == f.extension:
                    count = count + 1
    # Get the counts from our dependencies.
    for dep in ctx.rule.attr.deps:
        count = count + dep[FileCountInfo].count
    return [FileCountInfo(count = count)]

file_count_aspect = aspect(
    implementation = _file_count_aspect_impl,
    attr_aspects = ['deps'],
    attrs = {
        'extension' : attr.string(values = ['*', 'h', 'cc']),
    }
)

def _file_count_rule_impl(ctx):
    for dep in ctx.attr.deps:
        print(dep[FileCountInfo].count)

file_count_rule = rule(
    implementation = _file_count_rule_impl,
    attrs = {
        'deps' : attr.label_list(aspects = [file_count_aspect]),
        'extension' : attr.string(default = '*'),
    },
)

BUILD.bazel 文件:

load('//:file_count.bzl', 'file_count_rule')

cc_library(
    name = 'lib',
    srcs = [
        'lib.h',
        'lib.cc',
    ],
)

cc_binary(
    name = 'app',
    srcs = [
        'app.h',
        'app.cc',
        'main.cc',
    ],
    deps = ['lib'],
)

file_count_rule(
    name = 'file_count',
    deps = ['app'],
    extension = 'h',
)

方面定义

file_count_aspect = aspect(
    implementation = _file_count_aspect_impl,
    attr_aspects = ['deps'],
    attrs = {
        'extension' : attr.string(values = ['*', 'h', 'cc']),
    }
)

此示例展示了宽高比如何通过 deps 属性传播。

attrs 用于定义某个方面(aspect)的一组属性。公开的 aspect 属性用于定义参数,并且只能是 boolintstring 类型。对于通过规则传播的方面,intstring 参数必须指定 values。此示例包含一个名为 extension 的参数,其值可以是“*”“h”或“cc”。

对于规则传播的方面,系统会使用具有相同名称和类型的规则属性,从请求该方面的规则中获取参数值。(请参阅 file_count_rule 的定义)。

对于命令行方面,可以使用 --aspects_parameters 标志传递参数值。可以省略 intstring 参数的 values 限制。

方面还可以具有类型为 labellabel_list 的私有属性。自有品牌属性可用于指定由方面生成的操作所需的工具或库的依赖项。此示例中未定义私有属性,但以下代码段演示了如何将工具传入到某个方面:

...
    attrs = {
        '_protoc' : attr.label(
            default = Label('//tools:protoc'),
            executable = True,
            cfg = "exec"
        )
    }
...

方面实现

FileCountInfo = provider(
    fields = {
        'count' : 'number of files'
    }
)

def _file_count_aspect_impl(target, ctx):
    count = 0
    # Make sure the rule has a srcs attribute.
    if hasattr(ctx.rule.attr, 'srcs'):
        # Iterate through the sources counting files
        for src in ctx.rule.attr.srcs:
            for f in src.files.to_list():
                if ctx.attr.extension == '*' or ctx.attr.extension == f.extension:
                    count = count + 1
    # Get the counts from our dependencies.
    for dep in ctx.rule.attr.deps:
        count = count + dep[FileCountInfo].count
    return [FileCountInfo(count = count)]

与规则实现函数一样,一个方面实现函数会返回可供其依赖项访问的提供程序结构体。

在此示例中,FileCountInfo 定义为具有一个字段 count 的提供程序。最佳实践是使用 fields 属性明确定义提供程序的字段。

某个方面应用 A(X) 的一组提供程序是来自目标 X 的规则实现和方面 A 的实现的提供程序的联合。规则实现传播的提供程序会在应用方面之前创建并冻结,并且无法通过方面进行修改。如果目标和应用于它的方面各自提供相同类型的提供程序,则会发生错误,OutputGroupInfo(只要规则和方面指定不同的输出组,就会合并)和 InstrumentedFilesInfo(从方面获取)除外。这意味着,aspect 实现绝不能返回 DefaultInfo

参数和私有属性会在 ctx 的属性中传递。此示例引用 extension 参数,并确定要统计哪些文件。

对于返回的提供程序,沿着哪些属性传播了此方面(从 attr_aspects 列表中)的值会被替换为将此方面应用于这些属性的结果。例如,如果目标 X 的依赖项中包含 Y 和 Z,则 A(X) 的 ctx.rule.attr.deps 将为 [A(Y), A(Z)]。在此示例中,ctx.rule.attr.deps 是 Target 对象,是将此 aspect 应用于已应用此 aspect 的原始 target 的“依赖项”的结果。

在该示例中,该方面从目标的依赖项访问 FileCountInfo 提供程序,以累积文件的传递总数。

从规则调用方面

def _file_count_rule_impl(ctx):
    for dep in ctx.attr.deps:
        print(dep[FileCountInfo].count)

file_count_rule = rule(
    implementation = _file_count_rule_impl,
    attrs = {
        'deps' : attr.label_list(aspects = [file_count_aspect]),
        'extension' : attr.string(default = '*'),
    },
)

规则实现演示了如何通过 ctx.attr.deps 访问 FileCountInfo

规则定义演示了如何定义参数 (extension) 并为其指定默认值 (*)。请注意,由于在方面定义中对参数施加了限制,因此如果默认值不是“cc”“h”或“*”中的一种,则会出错。

通过目标规则调用切面

load('//:file_count.bzl', 'file_count_rule')

cc_binary(
    name = 'app',
...
)

file_count_rule(
    name = 'file_count',
    deps = ['app'],
    extension = 'h',
)

这演示了如何通过规则将 extension 参数传递到 aspect。由于 extension 参数在规则实现中具有默认值,因此 extension 会被视为可选参数。

构建 file_count 目标后,系统会评估我们的 aspect 本身,以及通过 deps 递归访问的所有目标。

参考