|
|
|
# Copyright 2023 The gRPC Authors
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
|
|
|
"""
|
|
|
|
Contains macros used for running bazelified tests.
|
|
|
|
"""
|
|
|
|
|
|
|
|
load(":dockerimage_current_versions.bzl", "DOCKERIMAGE_CURRENT_VERSIONS")
|
|
|
|
load("@bazel_toolchains//rules/exec_properties:exec_properties.bzl", "create_rbe_exec_properties_dict")
|
|
|
|
|
|
|
|
def _dockerized_sh_test(name, srcs = [], args = [], data = [], size = "medium", timeout = None, tags = [], exec_compatible_with = [], flaky = None, docker_image_version = None, docker_run_as_root = False, env = {}):
|
|
|
|
"""Runs sh_test under docker either via RBE or via docker sandbox."""
|
|
|
|
if docker_image_version:
|
|
|
|
image_spec = DOCKERIMAGE_CURRENT_VERSIONS.get(docker_image_version, None)
|
|
|
|
if not image_spec:
|
|
|
|
fail("Version info for docker image '%s' not found in dockerimage_current_versions.bzl" % docker_image_version)
|
|
|
|
else:
|
|
|
|
fail("docker_image_version attribute not set for dockerized test '%s'" % name)
|
|
|
|
|
|
|
|
exec_properties = create_rbe_exec_properties_dict(
|
|
|
|
labels = {
|
|
|
|
"workload": "misc",
|
|
|
|
"machine_size": "misc_large",
|
|
|
|
},
|
|
|
|
docker_network = "standard",
|
|
|
|
container_image = image_spec,
|
|
|
|
# TODO(jtattermusch): note that docker sandbox doesn't currently support "docker_run_as_root"
|
|
|
|
docker_run_as_root = docker_run_as_root,
|
|
|
|
)
|
|
|
|
|
|
|
|
# since the tests require special bazel args, only run them when explicitly requested
|
|
|
|
tags = ["manual"] + tags
|
|
|
|
|
|
|
|
# TODO(jtattermusch): find a way to ensure that action can only run under docker sandbox or remotely
|
|
|
|
# to avoid running it outside of a docker container by accident.
|
|
|
|
|
|
|
|
test_args = {
|
|
|
|
"name": name,
|
|
|
|
"srcs": srcs,
|
|
|
|
"tags": tags,
|
|
|
|
"args": args,
|
|
|
|
"flaky": flaky,
|
|
|
|
"data": data,
|
|
|
|
"size": size,
|
|
|
|
"env": env,
|
|
|
|
"timeout": timeout,
|
|
|
|
"exec_compatible_with": exec_compatible_with,
|
|
|
|
"exec_properties": exec_properties,
|
|
|
|
}
|
|
|
|
|
|
|
|
native.sh_test(
|
|
|
|
**test_args
|
|
|
|
)
|
|
|
|
|
|
|
|
def _dockerized_genrule(name, cmd, outs, srcs = [], tags = [], exec_compatible_with = [], docker_image_version = None, docker_run_as_root = False):
|
|
|
|
"""Runs genrule under docker either via RBE or via docker sandbox."""
|
|
|
|
if docker_image_version:
|
|
|
|
image_spec = DOCKERIMAGE_CURRENT_VERSIONS.get(docker_image_version, None)
|
|
|
|
if not image_spec:
|
|
|
|
fail("Version info for docker image '%s' not found in dockerimage_current_versions.bzl" % docker_image_version)
|
|
|
|
else:
|
|
|
|
fail("docker_image_version attribute not set for dockerized test '%s'" % name)
|
|
|
|
|
|
|
|
exec_properties = create_rbe_exec_properties_dict(
|
|
|
|
labels = {
|
|
|
|
"workload": "misc",
|
|
|
|
"machine_size": "misc_large",
|
|
|
|
},
|
|
|
|
docker_network = "standard",
|
|
|
|
container_image = image_spec,
|
|
|
|
# TODO(jtattermusch): note that docker sandbox doesn't currently support "docker_run_as_root"
|
|
|
|
docker_run_as_root = docker_run_as_root,
|
|
|
|
)
|
|
|
|
|
|
|
|
# since the tests require special bazel args, only run them when explicitly requested
|
|
|
|
tags = ["manual"] + tags
|
|
|
|
|
|
|
|
# TODO(jtattermusch): find a way to ensure that action can only run under docker sandbox or remotely
|
|
|
|
# to avoid running it outside of a docker container by accident.
|
|
|
|
|
|
|
|
genrule_args = {
|
|
|
|
"name": name,
|
|
|
|
"cmd": cmd,
|
|
|
|
"srcs": srcs,
|
|
|
|
"tags": tags,
|
|
|
|
"exec_compatible_with": exec_compatible_with,
|
|
|
|
"exec_properties": exec_properties,
|
|
|
|
"outs": outs,
|
|
|
|
}
|
|
|
|
|
|
|
|
native.genrule(
|
|
|
|
**genrule_args
|
|
|
|
)
|
|
|
|
|
|
|
|
def grpc_run_tests_harness_test(name, args = [], data = [], size = "medium", timeout = None, tags = [], exec_compatible_with = [], flaky = None, docker_image_version = None, use_login_shell = None, prepare_script = None, arch = None):
|
|
|
|
"""Execute an run_tests.py-harness style test under bazel.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
name: The name of the test.
|
|
|
|
args: The args to supply to the test binary.
|
|
|
|
data: Data dependencies.
|
|
|
|
size: The size of the test.
|
|
|
|
timeout: The test timeout.
|
|
|
|
tags: The tags for the test.
|
|
|
|
exec_compatible_with: A list of constraint values that must be
|
|
|
|
satisfied for the platform.
|
|
|
|
flaky: Whether this test is flaky.
|
|
|
|
docker_image_version: The docker .current_version file to use for docker containerization.
|
|
|
|
use_login_shell: If True, the run_tests.py command will run under a login shell.
|
|
|
|
prepare_script: Optional script that will be sourced before run_tests.py runs.
|
|
|
|
arch: Optional a command to change reported architecture (such as linux32 for i686) for the test.
|
|
|
|
"""
|
|
|
|
|
|
|
|
data = [
|
|
|
|
"//tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz",
|
|
|
|
] + data
|
|
|
|
|
|
|
|
args = [
|
|
|
|
"$(location //tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz)",
|
|
|
|
] + args
|
|
|
|
|
|
|
|
srcs = [
|
|
|
|
"//tools/bazelify_tests:grpc_run_tests_harness_test.sh",
|
|
|
|
]
|
|
|
|
|
|
|
|
env = {}
|
|
|
|
|
|
|
|
if use_login_shell:
|
|
|
|
env["GRPC_RUNTESTS_USE_LOGIN_SHELL"] = "1"
|
|
|
|
|
|
|
|
if prepare_script:
|
|
|
|
data = data + [prepare_script]
|
|
|
|
env["GRPC_RUNTESTS_PREPARE_SCRIPT"] = "$(location " + prepare_script + ")"
|
|
|
|
|
|
|
|
# Bazel RBE overrides ENTRYPOINT of docker so 32bits test relying on uname returning i686
|
|
|
|
# should have arch = "linux32" even when the docker file has it
|
|
|
|
if arch:
|
|
|
|
env["SETARCH_CMD"] = arch
|
|
|
|
|
|
|
|
# Enable ccache by default. This is important for speeding up the C++ cmake build,
|
|
|
|
# which isn't very efficient and tends to recompile some source files multiple times.
|
|
|
|
# Even though only the local disk cache is enabled (local to the docker container,
|
|
|
|
# so will be thrown away after the bazel actions finishes), ccache still speeds up
|
|
|
|
# the C++ build significantly.
|
|
|
|
# TODO(jtattermusch): find a cleaner way to toggle ccache for builds.
|
|
|
|
env["GRPC_BUILD_ENABLE_CCACHE"] = "true"
|
|
|
|
|
|
|
|
_dockerized_sh_test(name = name, srcs = srcs, args = args, data = data, size = size, timeout = timeout, tags = tags, exec_compatible_with = exec_compatible_with, flaky = flaky, docker_image_version = docker_image_version, env = env)
|
|
|
|
|
|
|
|
def grpc_run_bazel_distribtest_test(name, args = [], data = [], size = "medium", timeout = None, tags = [], exec_compatible_with = [], flaky = None, docker_image_version = None):
|
|
|
|
"""Execute bazel distribtest under bazel (an entire bazel build/test will run in a container as a single bazel action)
|
|
|
|
|
|
|
|
Args:
|
|
|
|
name: The name of the test.
|
|
|
|
args: The args to supply to the test binary.
|
|
|
|
data: Data dependencies.
|
|
|
|
size: The size of the test.
|
|
|
|
timeout: The test timeout.
|
|
|
|
tags: The tags for the test.
|
|
|
|
exec_compatible_with: A list of constraint values that must be
|
|
|
|
satisfied for the platform.
|
|
|
|
flaky: Whether this test is flaky.
|
|
|
|
docker_image_version: The docker .current_version file to use for docker containerization.
|
|
|
|
"""
|
|
|
|
|
|
|
|
data = [
|
|
|
|
"//tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz",
|
|
|
|
] + data
|
|
|
|
|
|
|
|
args = [
|
|
|
|
"$(location //tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz)",
|
|
|
|
] + args
|
|
|
|
|
|
|
|
srcs = [
|
|
|
|
"//tools/bazelify_tests:grpc_run_bazel_distribtest_test.sh",
|
|
|
|
]
|
|
|
|
env = {}
|
|
|
|
_dockerized_sh_test(name = name, srcs = srcs, args = args, data = data, size = size, timeout = timeout, tags = tags, exec_compatible_with = exec_compatible_with, flaky = flaky, docker_image_version = docker_image_version, env = env)
|
|
|
|
|
|
|
|
def grpc_run_cpp_distribtest_test(name, args = [], data = [], size = "medium", timeout = None, tags = [], exec_compatible_with = [], flaky = None, docker_image_version = None):
|
|
|
|
"""Execute an C++ distribtest under bazel.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
name: The name of the test.
|
|
|
|
args: The args to supply to the test binary.
|
|
|
|
data: Data dependencies.
|
|
|
|
size: The size of the test.
|
|
|
|
timeout: The test timeout.
|
|
|
|
tags: The tags for the test.
|
|
|
|
exec_compatible_with: A list of constraint values that must be
|
|
|
|
satisfied for the platform.
|
|
|
|
flaky: Whether this test is flaky.
|
|
|
|
docker_image_version: The docker .current_version file to use for docker containerization.
|
|
|
|
"""
|
|
|
|
|
|
|
|
data = [
|
|
|
|
"//tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz",
|
|
|
|
] + data
|
|
|
|
|
|
|
|
args = [
|
|
|
|
"$(location //tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz)",
|
|
|
|
] + args
|
|
|
|
|
|
|
|
srcs = [
|
|
|
|
"//tools/bazelify_tests:grpc_run_cpp_distribtest_test.sh",
|
|
|
|
]
|
|
|
|
|
|
|
|
# TODO(jtattermusch): revisit running docker as root (but currently some distribtests need to install stuff inside the docker container)
|
|
|
|
env = {}
|
|
|
|
_dockerized_sh_test(name = name, srcs = srcs, args = args, data = data, size = size, timeout = timeout, tags = tags, exec_compatible_with = exec_compatible_with, flaky = flaky, docker_image_version = docker_image_version, env = env, docker_run_as_root = True)
|
|
|
|
|
|
|
|
def grpc_run_simple_command_test(name, args = [], data = [], size = "medium", timeout = None, tags = [], exec_compatible_with = [], flaky = None, docker_image_version = None):
|
|
|
|
"""Execute the specified test command under grpc workspace (and under a docker container)
|
|
|
|
|
|
|
|
Args:
|
|
|
|
name: The name of the test.
|
|
|
|
args: The command to run.
|
|
|
|
data: Data dependencies.
|
|
|
|
size: The size of the test.
|
|
|
|
timeout: The test timeout.
|
|
|
|
tags: The tags for the test.
|
|
|
|
exec_compatible_with: A list of constraint values that must be
|
|
|
|
satisfied for the platform.
|
|
|
|
flaky: Whether this test is flaky.
|
|
|
|
docker_image_version: The docker .current_version file to use for docker containerization.
|
|
|
|
"""
|
|
|
|
|
|
|
|
data = [
|
|
|
|
"//tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz",
|
|
|
|
] + data
|
|
|
|
|
|
|
|
args = [
|
|
|
|
"$(location //tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz)",
|
|
|
|
] + args
|
|
|
|
|
|
|
|
srcs = [
|
|
|
|
"//tools/bazelify_tests:grpc_run_simple_command_test.sh",
|
|
|
|
]
|
|
|
|
|
|
|
|
env = {}
|
|
|
|
_dockerized_sh_test(name = name, srcs = srcs, args = args, data = data, size = size, timeout = timeout, tags = tags, exec_compatible_with = exec_compatible_with, flaky = flaky, docker_image_version = docker_image_version, env = env, docker_run_as_root = False)
|
|
|
|
|
|
|
|
def grpc_build_artifact_task(name, timeout = None, artifact_deps = [], tags = [], exec_compatible_with = [], flaky = None, docker_image_version = None, build_script = None):
|
|
|
|
"""Execute a build artifact task and a corresponding 'build test'.
|
|
|
|
|
|
|
|
The artifact is built by a genrule that always succeeds (Even if the underlying build fails)
|
|
|
|
and an sh_test (with "_build_test" suffix) that presents the result of the artifact build
|
|
|
|
in the result UI (by displaying the the build status, the log, and artifacts produced).
|
|
|
|
Such layout helps to easily build artifacts and run distribtests that depend on other artifacts,
|
|
|
|
while making the test results well structured and easy to interpret.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
name: The name of the target.
|
|
|
|
timeout: The test timeout for the build.
|
|
|
|
artifact_deps: List of dependencies on artifacts built by another grpc_build_artifact_task.
|
|
|
|
tags: The tags for the target.
|
|
|
|
exec_compatible_with: A list of constraint values that must be
|
|
|
|
satisfied for the platform.
|
|
|
|
flaky: Whether this artifact build is flaky.
|
|
|
|
docker_image_version: The docker .current_version file to use for docker containerization.
|
|
|
|
build_script: The script that builds the artifacts.
|
|
|
|
"""
|
|
|
|
|
|
|
|
out_exitcode_file = str(name + "_exit_code")
|
|
|
|
out_build_log = str(name + "_build_log.txt")
|
|
|
|
out_archive_name = str(name + ".tar.gz")
|
|
|
|
|
|
|
|
genrule_outs = [
|
|
|
|
out_exitcode_file,
|
|
|
|
out_build_log,
|
|
|
|
out_archive_name,
|
|
|
|
]
|
|
|
|
|
|
|
|
genrule_srcs = [
|
|
|
|
"//tools/bazelify_tests:grpc_build_artifact_task.sh",
|
|
|
|
"//tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz",
|
|
|
|
build_script,
|
|
|
|
]
|
|
|
|
|
|
|
|
cmd = "$(location //tools/bazelify_tests:grpc_build_artifact_task.sh) $(location //tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz) $(location " + build_script + ") $(location " + out_exitcode_file + ") $(location " + out_build_log + ") $(location " + out_archive_name + ")"
|
|
|
|
|
|
|
|
# for each artifact task we depends on, use the corresponding tar.gz as extra src and pass its location as an extra cmdline arg.
|
|
|
|
for dep in artifact_deps:
|
|
|
|
dep_archive_name = str(dep + ".tar.gz")
|
|
|
|
cmd = cmd + " $(location " + dep_archive_name + ")"
|
|
|
|
genrule_srcs.append(dep_archive_name)
|
|
|
|
|
|
|
|
_dockerized_genrule(name = name, cmd = cmd, outs = genrule_outs, srcs = genrule_srcs, tags = tags, exec_compatible_with = exec_compatible_with, docker_image_version = docker_image_version, docker_run_as_root = False)
|
|
|
|
|
|
|
|
# The genrule above always succeeds (even if the underlying build fails), so that we can create rules that depend
|
|
|
|
# on multiple artifact builds (of which some can fail). The actual build status (exitcode) and the log of the build
|
|
|
|
# will be reported by an associated sh_test (that gets displayed in the UI in a much nicer way than a genrule).
|
|
|
|
# Note that in bazel you cannot declare a test that has declared outputs and you also cannot make other rules
|
|
|
|
# depend on a test - which is the reason why we need a separate genrule to represent the build itself.
|
|
|
|
test_name = str(name + "_build_test")
|
|
|
|
test_srcs = [
|
|
|
|
"//tools/bazelify_tests:grpc_build_artifact_task_build_test.sh",
|
|
|
|
]
|
|
|
|
test_data = [
|
|
|
|
out_exitcode_file,
|
|
|
|
out_build_log,
|
|
|
|
out_archive_name,
|
|
|
|
]
|
|
|
|
test_env = {}
|
|
|
|
test_args = [
|
|
|
|
"$(location " + out_exitcode_file + ")",
|
|
|
|
"$(location " + out_build_log + ")",
|
|
|
|
"$(location " + out_archive_name + ")",
|
|
|
|
]
|
|
|
|
_dockerized_sh_test(name = test_name, srcs = test_srcs, args = test_args, data = test_data, size = "small", tags = tags, exec_compatible_with = exec_compatible_with, flaky = flaky, docker_image_version = docker_image_version, env = test_env, docker_run_as_root = False)
|
|
|
|
|
|
|
|
def grpc_run_distribtest_test(name, artifact_deps = [], size = "medium", timeout = None, tags = [], exec_compatible_with = [], flaky = None, docker_image_version = None, build_script = None, docker_run_as_root = False):
|
|
|
|
"""Run a distribtest for a previously built artifact/package
|
|
|
|
|
|
|
|
Args:
|
|
|
|
name: The name of the test.
|
|
|
|
artifact_deps: List of dependencies on artifacts built by another grpc_build_artifact_task.
|
|
|
|
size: The size of the test.
|
|
|
|
timeout: The test timeout.
|
|
|
|
tags: The tags for the test.
|
|
|
|
exec_compatible_with: A list of constraint values that must be
|
|
|
|
satisfied for the platform.
|
|
|
|
flaky: Whether this test is flaky.
|
|
|
|
docker_image_version: The docker .current_version file to use for docker containerization.
|
|
|
|
build_script: The script that runs the test.
|
|
|
|
docker_run_as_root: If True, the test will run under docker as root.
|
|
|
|
"""
|
|
|
|
|
|
|
|
data = [
|
|
|
|
"//tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz",
|
|
|
|
build_script,
|
|
|
|
]
|
|
|
|
|
|
|
|
args = [
|
|
|
|
"$(location //tools/bazelify_tests:grpc_repo_archive_with_submodules.tar.gz)",
|
|
|
|
"$(location " + build_script + ")",
|
|
|
|
]
|
|
|
|
|
|
|
|
# for each artifact task we depends on, use the corresponding tar.gz as extra data item and pass its location as an extra arg.
|
|
|
|
for dep in artifact_deps:
|
|
|
|
dep_archive_name = str(dep + ".tar.gz")
|
|
|
|
args.append("$(location " + dep_archive_name + ")")
|
|
|
|
data.append(dep_archive_name)
|
|
|
|
|
|
|
|
srcs = [
|
|
|
|
"//tools/bazelify_tests:grpc_run_distribtest_test.sh",
|
|
|
|
]
|
|
|
|
|
|
|
|
env = {}
|
|
|
|
_dockerized_sh_test(name = name, srcs = srcs, args = args, data = data, size = size, timeout = timeout, tags = tags, exec_compatible_with = exec_compatible_with, flaky = flaky, docker_image_version = docker_image_version, env = env, docker_run_as_root = docker_run_as_root)
|