| #!/bin/bash |
| # |
| # Copyright 2018 The Bazel Authors. All rights reserved. |
| # |
| # 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. |
| # |
| # modify_execution_info_test.sh: tests of the --modify_execution_info flag. |
| |
| # --- begin runfiles.bash initialization --- |
| # Copy-pasted from Bazel's Bash runfiles library (tools/bash/runfiles/runfiles.bash). |
| set -euo pipefail |
| if [[ ! -d "${RUNFILES_DIR:-/dev/null}" && ! -f "${RUNFILES_MANIFEST_FILE:-/dev/null}" ]]; then |
| if [[ -f "$0.runfiles_manifest" ]]; then |
| export RUNFILES_MANIFEST_FILE="$0.runfiles_manifest" |
| elif [[ -f "$0.runfiles/MANIFEST" ]]; then |
| export RUNFILES_MANIFEST_FILE="$0.runfiles/MANIFEST" |
| elif [[ -f "$0.runfiles/bazel_tools/tools/bash/runfiles/runfiles.bash" ]]; then |
| export RUNFILES_DIR="$0.runfiles" |
| fi |
| fi |
| if [[ -f "${RUNFILES_DIR:-/dev/null}/bazel_tools/tools/bash/runfiles/runfiles.bash" ]]; then |
| source "${RUNFILES_DIR}/bazel_tools/tools/bash/runfiles/runfiles.bash" |
| elif [[ -f "${RUNFILES_MANIFEST_FILE:-/dev/null}" ]]; then |
| source "$(grep -m1 "^bazel_tools/tools/bash/runfiles/runfiles.bash " \ |
| "$RUNFILES_MANIFEST_FILE" | cut -d ' ' -f 2-)" |
| else |
| echo >&2 "ERROR: cannot find @bazel_tools//tools/bash/runfiles:runfiles.bash" |
| exit 1 |
| fi |
| # --- end runfiles.bash initialization --- |
| |
| source "$(rlocation "io_bazel/src/test/shell/integration_test_setup.sh")" \ |
| || { echo "integration_test_setup.sh not found!" >&2; exit 1; } |
| |
| case "$(uname -s | tr [:upper:] [:lower:])" in |
| msys*|mingw*|cygwin*) |
| declare -r is_windows=true |
| ;; |
| *) |
| declare -r is_windows=false |
| ;; |
| esac |
| |
| if "$is_windows"; then |
| export MSYS_NO_PATHCONV=1 |
| export MSYS2_ARG_CONV_EXCL="*" |
| fi |
| |
| #### HELPER FUNCTIONS ################################################## |
| |
| if ! type try_with_timeout >&/dev/null; then |
| # Bazel's testenv.sh defines try_with_timeout but the Google-internal version |
| # uses a different testenv.sh. |
| function try_with_timeout() { $* ; } |
| fi |
| |
| function set_up() { |
| cd ${WORKSPACE_DIR} |
| } |
| |
| function tear_down() { |
| try_with_timeout bazel shutdown |
| } |
| |
| #### TESTS ############################################################# |
| |
| function test_aquery_respects_modify_execution_info_changes { |
| local pkg="${FUNCNAME[0]}" |
| mkdir -p "$pkg" || fail "mkdir -p $pkg" |
| cat > "$pkg/BUILD" <<'EOF' |
| genrule(name = "bar", outs = ["bar_out.txt"], cmd = "touch $(OUTS)") |
| EOF |
| bazel aquery --output=text "//$pkg:bar" \ |
| --modify_execution_info=Genrule=+requires-x \ |
| > output1 2> "$TEST_log" || fail "Expected success" |
| assert_contains "ExecutionInfo: {requires-x: ''}" output1 |
| |
| bazel aquery --output=text "//$pkg:bar" \ |
| --modify_execution_info=Genrule=+requires-y \ |
| > output2 2> "$TEST_log" || fail "Expected success" |
| assert_contains "ExecutionInfo: {requires-y: ''}" output2 |
| } |
| |
| function test_modify_execution_info_multiple { |
| local pkg="${FUNCNAME[0]}" |
| mkdir -p "$pkg" || fail "mkdir -p $pkg" |
| cat > "$pkg/BUILD" <<'EOF' |
| genrule( |
| name = "bar", |
| outs = ["bar_out.txt"], |
| cmd = "touch $(OUTS)", |
| tags = ["requires-x"], |
| ) |
| cc_binary(name="zero", srcs=["zero.cc"]) |
| EOF |
| echo "int main(void) {}" > "$pkg/zero.cc" |
| |
| # multiple elements in the value list that match the same mnemonic. |
| bazel aquery --output=text "//$pkg:bar" \ |
| --modify_execution_info=Genrule=+requires-y,Genrule=+requires-z \ |
| > output 2> "$TEST_log" || fail "Expected success" |
| assert_contains "ExecutionInfo: {requires-x: '', requires-y: '', "\ |
| "requires-z: ''}" output |
| |
| # multiple elements in the value list, the first of which adds an |
| # ExecutionInfo and the second of which removes it. |
| bazel aquery --output=text "//$pkg:bar" \ |
| --modify_execution_info=Genrule=+requires-z,.*=-requires-z \ |
| > output 2> "$TEST_log" || fail "Expected success" |
| assert_contains "ExecutionInfo: {requires-x: ''}" output |
| |
| # multiple elements in the value list, the first of which removes an |
| # ExecutionInfo (previously absent) and the second of which adds it. |
| bazel aquery --output=text "//$pkg:bar" \ |
| --modify_execution_info=Genrule=-requires-z,.*=+requires-z \ |
| > output 2> "$TEST_log" || fail "Expected success" |
| assert_contains "ExecutionInfo: {requires-x: '', requires-z: ''}" output |
| |
| # multiple elements in the value list, the first of which removes an |
| # ExecutionInfo (previously present) and the second of which adds it back. |
| bazel aquery --output=text "//$pkg:bar" \ |
| --modify_execution_info=Genrule=-requires-x,.*=+requires-x \ |
| > output 2> "$TEST_log" || fail "Expected success" |
| assert_contains "ExecutionInfo: {requires-x: ''}" output |
| |
| # multiple elements with multiple values |
| bazel aquery --output=text "//$pkg:all" \ |
| --modify_execution_info=Genrule=-requires-x,Genrule=+requires-z,\ |
| Genrule=+requires-a,CppCompile=+requires-b,CppCompile=+requires-c \ |
| > output 2> "$TEST_log" || fail "Expected success" |
| assert_contains "ExecutionInfo: {requires-a: '', requires-z: ''}" output |
| assert_contains "ExecutionInfo: {requires-b: '', requires-c: ''" output |
| |
| # negative lookahead |
| bazel aquery --output=text "//$pkg:all" \ |
| --modify_execution_info='(?!Genrule).*=+requires-a,(?!CppCompile).*=+requires-z' \ |
| > output 2> "$TEST_log" || fail "Expected success" |
| assert_contains "ExecutionInfo: {requires-x: '', requires-z: ''}" output |
| assert_contains "ExecutionInfo: {requires-a: ''" output |
| } |
| |
| function test_modify_execution_info_various_types() { |
| if [[ "$PRODUCT_NAME" = "bazel" ]]; then |
| # proto_library requires this external workspace. |
| cat >> WORKSPACE << EOF |
| http_archive( |
| name = "rules_proto", |
| strip_prefix = "rules_proto-97d8af4dc474595af3900dd85cb3a29ad28cc313", |
| sha256 = "602e7161d9195e50246177e7c55b2f39950a9cf7366f74ed5f22fd45750cd208", |
| urls = [ |
| "https://mirror.bazel.build/github.com/bazelbuild/rules_proto/archive/97d8af4dc474595af3900dd85cb3a29ad28cc313.tar.gz", |
| "https://github.com/bazelbuild/rules_proto/archive/97d8af4dc474595af3900dd85cb3a29ad28cc313.tar.gz", |
| ], |
| ) |
| load("@rules_proto//proto:repositories.bzl", "rules_proto_dependencies", "rules_proto_toolchains") |
| rules_proto_dependencies() |
| rules_proto_toolchains() |
| |
| # @com_google_protobuf//:protoc depends on @io_bazel//third_party/zlib. |
| new_local_repository( |
| name = "io_bazel", |
| path = "$(dirname $(rlocation io_bazel/third_party/rules_python/rules_python.WORKSPACE))/../..", |
| build_file_content = "# Intentionally left empty.", |
| workspace_file_content = "workspace(name = 'io_bazel')", |
| ) |
| |
| # TODO(#9029): May require some adjustment if/when we depend on the real |
| # @rules_python in the real source tree, since this third_party/ package won't |
| # be available. |
| new_local_repository( |
| name = "rules_python", |
| path = "$(dirname $(rlocation io_bazel/third_party/rules_python/rules_python.WORKSPACE))", |
| build_file = "$(rlocation io_bazel/third_party/rules_python/BUILD)", |
| workspace_file = "$(rlocation io_bazel/third_party/rules_python/rules_python.WORKSPACE)", |
| ) |
| EOF |
| fi |
| local pkg="${FUNCNAME[0]}" |
| mkdir -p "$pkg" || fail "mkdir -p $pkg" |
| echo "load('//$pkg:shell.bzl', 'skylark_shell')" > "$pkg/BUILD" |
| cat >> "$pkg/BUILD" <<'EOF' |
| skylark_shell( |
| name = "shelly", |
| output = "ok.txt", |
| ) |
| |
| cc_binary(name="zero", srcs=["zero.cc"]) |
| |
| sh_test(name="test_a", srcs=["a.sh"]) |
| |
| java_library(name = "javalib", srcs = ["HelloWorld.java"]) |
| |
| action_listener( |
| name = "al", |
| extra_actions = [":echo-filename"], |
| mnemonics = ["Javac"], |
| visibility = ["//visibility:public"], |
| ) |
| |
| extra_action(name = "echo-filename", cmd = "echo Hi \$(EXTRA_ACTION_FILE)") |
| |
| py_binary(name = "pybar", srcs=["pybar.py"],) |
| |
| proto_library(name = "proto", srcs=["foo.proto"]) |
| EOF |
| cat > "$pkg/shell.bzl" <<'EOF' |
| def _impl(ctx): |
| ctx.actions.run_shell( |
| outputs = [ ctx.outputs.output ], |
| command = "touch %s" % ctx.outputs.output.path, |
| ) |
| |
| skylark_shell = rule( |
| _impl, |
| attrs = { |
| "output": attr.output(mandatory=True), |
| } |
| ) |
| EOF |
| cat > "$pkg/a.sh" <<'EOF' |
| #!/bin/sh |
| exit 0 |
| EOF |
| chmod 755 "$pkg/a.sh" |
| echo "int main(void) {}" > "$pkg/zero.cc" |
| echo "public class HelloWorld {}" > "$pkg/HelloWorld.java" |
| echo 'print("Hi")' > "$pkg/pybar.py" |
| echo 'syntax="proto2"; package foo;' > "$pkg/foo.proto" |
| |
| bazel aquery --output=text "//$pkg:all" \ |
| --experimental_action_listener=$pkg:al \ |
| --modify_execution_info=\ |
| echo.*=+requires-extra-action,\ |
| .*Proto.*=+requires-proto,\ |
| CppCompile=+requires-cpp-compile,\ |
| CppLink=+requires-cpp-link,\ |
| TestRunner=+requires-test-runner,\ |
| Turbine=+requires-turbine,\ |
| JavaSourceJar=+requires-java-source-jar,\ |
| Javac=+requires-javac,\ |
| PyTinypar=+requires-py-tinypar,\ |
| Action=+requires-action \ |
| > output 2> "$TEST_log" || fail "Expected success" |
| |
| # There are sometimes other elements in ExecutionInfo, e.g. requires-darwin |
| # for obj-c, supports-workers for java. Since testing for these combinations |
| # would be brittle, irrelevant to the operation of the flag, and in some |
| # cases platform-dependent, we just search for the key itself, not the whole |
| # ExecutionInfo: {...} line. |
| assert_contains "requires-action: ''" output |
| assert_contains "requires-cpp-compile: ''" output |
| assert_contains "requires-cpp-link: ''" output |
| assert_contains "requires-extra-action: ''" output |
| assert_contains "requires-test-runner: ''" output |
| assert_contains "requires-javac: ''" output |
| assert_contains "requires-turbine: ''" output |
| assert_contains "requires-java-source-jar: ''" output |
| assert_contains "requires-proto: ''" output # GenProtoDescriptorSet should match |
| if [[ "$PRODUCT_NAME" != "bazel" ]]; then |
| # Python rules generate some cpp actions and local actions, but py-tinypar |
| # is the main unique-to-python rule which runs remotely for a py_binary. |
| assert_contains "requires-py-tinypar: ''" output |
| fi |
| } |
| |
| # Regression test for b/127874955. We use --output=textproto since --output=text |
| # sorts the execution info. |
| function test_modify_execution_info_deterministic_order() { |
| local pkg="${FUNCNAME[0]}" |
| mkdir -p "$pkg/x" "$pkg/y" || fail "mkdir failed" |
| touch "$pkg/BUILD" |
| cat > "$pkg/build_defs.bzl" <<'EOF' || fail "Couldn't cat" |
| def _rule_x_impl(ctx): |
| output = ctx.outputs.out |
| ctx.actions.run_shell( |
| outputs = [output], |
| command = "touch %s" % output.path, |
| mnemonic = "RuleX", |
| execution_requirements = {"requires-x": ""}, |
| ) |
| |
| rule_x = rule(outputs = {"out": "%{name}.out"}, implementation = _rule_x_impl) |
| |
| def _rule_y_impl(ctx): |
| output = ctx.outputs.out |
| ctx.actions.run_shell( |
| outputs = [output], |
| command = "touch %s" % output.path, |
| mnemonic = "RuleY", |
| execution_requirements = {"requires-y": ""}, |
| ) |
| |
| rule_y = rule(outputs = {"out": "%{name}.out"}, implementation = _rule_y_impl) |
| EOF |
| echo "load('//$pkg:build_defs.bzl', 'rule_x')" > "$pkg/x/BUILD" |
| echo 'rule_x(name = "x")' >> "$pkg/x/BUILD" |
| echo "load('//$pkg:build_defs.bzl', 'rule_y')" > "$pkg/y/BUILD" |
| echo 'rule_y(name = "y")' >> "$pkg/y/BUILD" |
| |
| mod='Rule(X|Y)=+requires-x,Rule(X|Y)=+requires-y' |
| |
| bazel aquery "//$pkg/x" --output=textproto --modify_execution_info="$mod" \ |
| > output1 2> "$TEST_log" || fail "Expected success" |
| |
| bazel shutdown >& "$TEST_log" || fail "Couldn't shutdown" |
| |
| bazel aquery "//$pkg/y" --modify_execution_info="$mod" \ |
| >& "$TEST_log" || fail "Expected success" |
| |
| bazel aquery "//$pkg/x" --output=textproto --modify_execution_info="$mod" \ |
| > output2 2> "$TEST_log" || fail "Expected success" |
| |
| assert_equals "$(cat output1)" "$(cat output2)" |
| } |
| |
| # Regression test for b/130762259. |
| function test_modify_execution_info_changes_test_runner_cache_key() { |
| local pkg="${FUNCNAME[0]}" |
| mkdir -p "$pkg" |
| echo "sh_test(name = 'test', srcs = ['test.sh'])" > "$pkg/BUILD" |
| touch "$pkg/test.sh" |
| |
| bazel aquery "mnemonic(TestRunner,//$pkg:test)" --output=text \ |
| --modify_execution_info= \ |
| 2> "$TEST_log" | grep ActionKey > key1 || fail "Expected success" |
| |
| bazel aquery "mnemonic(TestRunner,//$pkg:test)" --output=text \ |
| --modify_execution_info=TestRunner=+requires-x \ |
| 2> "$TEST_log" | grep ActionKey > key2 || fail "Expected success" |
| |
| assert_not_equals "$(cat key1)" "$(cat key2)" |
| } |
| |
| run_suite "Integration tests of the --modify_execution_info option." |