This repository contains Bazel macros that execute integration tests that use Bazel (e.g. execute tests in a child workspace). The macros support running integration tests with multiple versions of Bazel.
The following provides a quick introduction on how to use the rules in this repository. Also, check out the documentation, the integration test defined in this repo, and the custom test runner example for more information.
Add the following to your WORKSPACE
file to add this repository and its dependencies.
http_archive(
name = "contrib_rules_bazel_integration_test",
sha256 = "24e5e8f388bec2da0975cfda6073ed0174a4f62cb874b5dc8037c98faa6acdfd",
strip_prefix = "rules_bazel_integration_test-0.7.0",
urls = [
"http://github.com/bazel-contrib/rules_bazel_integration_test/archive/v0.7.0.tar.gz",
],
)
load("@contrib_rules_bazel_integration_test//bazel_integration_test:deps.bzl", "bazel_integration_test_rules_dependencies")
bazel_integration_test_rules_dependencies()
load("@cgrindel_bazel_starlib//:deps.bzl", "bazel_starlib_dependencies")
bazel_starlib_dependencies()
load("@bazel_skylib//:workspace.bzl", "bazel_skylib_workspace")
bazel_skylib_workspace()
Add the following to a file called bazel_versions.bzl
at the root of your repository. Replace the
Bazel version values with the values that you would like to test against for your integration tests.
CURRENT_BAZEL_VERSION = "5.1.0"
OTHER_BAZEL_VERSIONS = [
"4.2.2",
"6.0.0-pre.20220328.1",
]
SUPPORTED_BAZEL_VERSIONS = [
CURRENT_BAZEL_VERSION,
] + OTHER_BAZEL_VERSIONS
NOTE: The above code designates a current version and other versions. This can be useful if you have a large number of integration tests where you want to execute all of them against the current version and execute a subset of them against other Bazel versions.
Add the following to the Bazel build file in the same package as the bazel_versions.bzl
file.
load("@bazel_skylib//:bzl_library.bzl", "bzl_library")
bzl_library(
name = "bazel_versions",
srcs = ["bazel_versions.bzl"],
visibility = ["//:__subpackages__"],
)
Back in the WORKSPACE
file, add the following to download and prepare the Bazel binaries for
testing.
load("//:bazel_versions.bzl", "SUPPORTED_BAZEL_VERSIONS")
load("@contrib_rules_bazel_integration_test//bazel_integration_test:defs.bzl", "bazel_binaries")
bazel_binaries(versions = SUPPORTED_BAZEL_VERSIONS)
If you have child workspaces under your parent workspace, you need to tell Bazel to ignore the child workspaces. This can be done in one of two ways:
- Add entries to the
.bazelignore
file in the parent workspace. - Specify a list of deleted packages using the
--deleted_packages
flag.
For glob()
in the parent workspace to find files in the child workspaces, we need to use the
deleted packages mechanism.
Add the following to the .bazelrc
in the parent workspace. Leave the values blank for now.
# To update these lines, execute
# `bazel run @contrib_rules_bazel_integration_test//tools:update_deleted_packages`
build --deleted_packages=
query --deleted_packages=
To populate the values, we will run a utility that looks for child workspaces (i.e., WORKSPACE
files) and find all of the directories that have Bazel build files (i.e., BUILD
, BUILD.bazel
).
Execute the following in a parent workspace directory.
# Populate the --deleted_packages flags in the .bazelrc
$ bazel run @contrib_rules_bazel_integration_test//tools:update_deleted_packages
After running the utility, the --deleted_packages
entries in the .bazelrc
should have a
comma-separated list of packages under the child workspaces.
For the purposes of this example, lets assume that we have an examples
directory which contains a
subdirectory called simple
. The simple
directory contains another Bazel workspace (i.e. has its
own WORKSPACE
file) that does not have a dependency on the parent workspace. We want to
execute tests in the simple
workspace using different versions of Bazel.
Add the following to the BUILD.bazel
file in the examples
directory.
load("//:bazel_versions.bzl", "CURRENT_BAZEL_VERSION", "OTHER_BAZEL_VERSIONS")
load(
"@contrib_rules_bazel_integration_test//bazel_integration_test:defs.bzl",
"bazel_integration_test",
"bazel_integration_tests",
"default_test_runner",
"integration_test_utils",
)
# Declare a test runner to drive the integration tests.
default_test_runner(
name = "simple_test_runner",
)
# If you only want to execute against a single version of Bazel, use
# bazel_integration_test.
bazel_integration_test(
name = "simple_test",
bazel_version = CURRENT_BAZEL_VERSION,
test_runner = ":simple_test_runner",
workspace_path = "simple",
)
# If you want to execute an integration test using multiple versions of Bazel,
# use bazel_integration_tests. It will generate multiple integration test
# targets with names derived from the base name and the bazel version.
bazel_integration_tests(
name = "simple_test",
bazel_versions = OTHER_BAZEL_VERSIONS,
test_runner = ":simple_test_runner",
workspace_path = "simple",
)
# By default, the integration test targets are tagged as `manual`. This
# prevents the targets from being found from most target expansion (e.g.
# `//...`, `:all`). To easily execute a group of integration tests, you may
# want to define a test suite which includes the desired test targets.
test_suite(
name = "all_integration_tests",
# If you don't apply the test tags to the test suite, the test suite will
# be found when `bazel test //...` is executed.
tags = integration_test_utils.DEFAULT_INTEGRATION_TEST_TAGS,
tests = [":simple_test"] +
integration_test_utils.bazel_integration_test_names(
"simple_test",
OTHER_BAZEL_VERSIONS,
),
visibility = ["//:__subpackages__"],
)
The above code defines three test targets: //examples:simple_test
,
//examples:simple_test_bazel_5_0_0-pre_20211011_2
, and
//examples:simple_test_bazel_6_0_0-pre_20211101_2
. The all_integration_tests
target is a test
suite that executes all of the integration tests with a single command:
# Execute all of the integration tests
$ bazel test //examples:all_integration_tests
In the quickstart example, the child workspace does not reference the parent workspace. In many cases, a child workspace will reference the parent workspace using a local_repository declaration to demonstrate functionality from the parent workspace.
# Child WORKSPACE at examples/simple/WORKSPACE
# Reference the parent workspace
local_repository(
name = "my_parent_workspace",
path = "../..",
)
This section explains how to use rules_bazel_integration_test
in this situation. To review working
examples, check out bazel-starlib and
rules_spm.
The child workspace needs to access parent workspace files. To easily reference the files, declare a
filegroup
at the root of the parent workspace to collect all of the files that the child workspace
needs. The values listed in the srcs
should include every file and/or package that the child
workspaces require.
# This target collects all of the parent workspace files needed by the child workspaces.
filegroup(
name = "local_repository_files",
# Include every package that is required by the child workspaces.
srcs = [
"BUILD.bazel",
"WORKSPACE",
"//spm:all_files",
"//spm/internal:all_files",
"//spm/internal/modulemap_parser:all_files",
],
visibility = ["//:__subpackages__"],
)
In every parent workspace package that is listed in the srcs
above, create a filegroup globbing
all of the files in the package.
# Add to every package that is required by a child workspace.
filegroup(
name = "all_files",
srcs = glob(["*"]),
visibility = ["//:__subpackages__"],
)
The bazel_integration_test
and
bazel_integration_tests
declarations
include a workspace_files
attribute. If not specified, it defaults to a custom glob expression
selecting files under the child workspace directory. To include the parent workspace files, add the
attribute with an expression that globs the workspace files and the //:local_repository_files
target.
bazel_integration_test(
name = "simple_test",
bazel_version = CURRENT_BAZEL_VERSION,
test_runner = ":simple_test_runner",
workspace_files = integration_test_utils.glob_workspace_files("simple") + [
"//:local_repository_files",
],
workspace_path = "simple",
)
Execute the integration test.
# Execute the integration test called simple_test
$ bazel test //examples:simple_test
For information on implementing a custom test runner, please see the documentation.