|  | #!/bin/bash | 
|  | # | 
|  | # Copyright 2019 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. | 
|  | # | 
|  | # Tests Starlark API pertaining to action inspection via aspect. | 
|  |  | 
|  | # --- 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 | 
|  |  | 
|  | add_to_bazelrc "build --package_path=%workspace%" | 
|  |  | 
|  | function test_directory_args_inspection() { | 
|  | mkdir -p package | 
|  | cat > package/makes_tree_artifacts.sh <<EOF | 
|  | #!/bin/bash | 
|  | my_dir=\$1 | 
|  |  | 
|  | touch \$my_dir/a.txt | 
|  | touch \$my_dir/b.txt | 
|  | touch \$my_dir/c.txt | 
|  | EOF | 
|  | chmod 755 package/makes_tree_artifacts.sh | 
|  |  | 
|  | cat > package/write.sh <<EOF | 
|  | #!/bin/bash | 
|  | output_file=\$1 | 
|  | shift; | 
|  |  | 
|  | echo "\$@" > \$output_file | 
|  | EOF | 
|  | chmod 755 package/write.sh | 
|  |  | 
|  | cat > package/lib.bzl <<EOF | 
|  | def _tree_art_impl(ctx): | 
|  | my_dir = ctx.actions.declare_directory('dir') | 
|  | ctx.actions.run( | 
|  | executable = ctx.executable._makes_tree, | 
|  | outputs = [my_dir], | 
|  | arguments = [my_dir.path]) | 
|  |  | 
|  | digest = ctx.actions.declare_file('digest') | 
|  | digest_args = ctx.actions.args() | 
|  | digest_args.add(digest.path) | 
|  | digest_args.add_all([my_dir]) | 
|  |  | 
|  | ctx.actions.run(executable = ctx.executable._write, | 
|  | inputs = [my_dir], | 
|  | outputs = [digest], | 
|  | arguments = [digest_args]) | 
|  | return [DefaultInfo(files=depset([digest]))] | 
|  |  | 
|  | def _actions_test_impl(target, ctx): | 
|  | action = target.actions[1] # digest action | 
|  | aspect_out = ctx.actions.declare_file('aspect_out') | 
|  | ctx.actions.run_shell(inputs = action.inputs, | 
|  | outputs = [aspect_out], | 
|  | command = "echo \$@ > " + aspect_out.path, | 
|  | arguments = action.args) | 
|  | return [OutputGroupInfo(out=[aspect_out])] | 
|  |  | 
|  | tree_art_rule = rule(implementation = _tree_art_impl, | 
|  | attrs = { | 
|  | "_makes_tree" : attr.label(allow_single_file = True, | 
|  | cfg = "host", | 
|  | executable = True, | 
|  | default = "//package:makes_tree_artifacts.sh"), | 
|  | "_write" : attr.label(allow_single_file = True, | 
|  | cfg = "host", | 
|  | executable = True, | 
|  | default = "//package:write.sh")}) | 
|  |  | 
|  | actions_test_aspect = aspect(implementation = _actions_test_impl) | 
|  | EOF | 
|  |  | 
|  | cat > package/BUILD <<EOF | 
|  | load(":lib.bzl", "tree_art_rule") | 
|  |  | 
|  | tree_art_rule(name = "x") | 
|  | EOF | 
|  |  | 
|  | bazel build package:x || fail "Unexpected build failure" | 
|  |  | 
|  | cat "${PRODUCT_NAME}-bin/package/digest" | grep "a.txt.*b.txt.*c.txt" \ | 
|  | || fail "rule digest does not contain tree artifact args" | 
|  |  | 
|  | bazel build package:x --aspects=//package:lib.bzl%actions_test_aspect \ | 
|  | --output_groups=out | 
|  |  | 
|  | cat "${PRODUCT_NAME}-bin/package/aspect_out" | grep "a.txt.*b.txt.*c.txt" \ | 
|  | || fail "aspect Args do not contain tree artifact args" | 
|  | } | 
|  |  | 
|  | function test_directory_args_inspection_param_file() { | 
|  | mkdir -p package | 
|  | cat > package/makes_tree_artifacts.sh <<EOF | 
|  | #!/bin/bash | 
|  | my_dir=\$1 | 
|  |  | 
|  | touch \$my_dir/a.txt | 
|  | touch \$my_dir/b.txt | 
|  | touch \$my_dir/c.txt | 
|  | EOF | 
|  | chmod 755 package/makes_tree_artifacts.sh | 
|  |  | 
|  | cat > package/write.sh <<EOF | 
|  | #!/bin/bash | 
|  |  | 
|  | param_file=\$1 | 
|  | shift; | 
|  |  | 
|  | output_file="\$(head -n 1 \$param_file)" | 
|  |  | 
|  | tail \$param_file -n +2 | tr '\n' ' ' > "\$output_file" | 
|  | EOF | 
|  | chmod 755 package/write.sh | 
|  |  | 
|  | cat > package/lib.bzl <<EOF | 
|  | def _tree_art_impl(ctx): | 
|  | my_dir = ctx.actions.declare_directory('dir') | 
|  | ctx.actions.run( | 
|  | executable = ctx.executable._makes_tree, | 
|  | outputs = [my_dir], | 
|  | arguments = [my_dir.path]) | 
|  |  | 
|  | digest = ctx.actions.declare_file('digest') | 
|  | digest_args = ctx.actions.args() | 
|  | digest_args.add(digest.path) | 
|  | digest_args.add_all([my_dir]) | 
|  | digest_args.use_param_file("%s", use_always=True) | 
|  |  | 
|  | ctx.actions.run(executable = ctx.executable._write, | 
|  | inputs = [my_dir], | 
|  | outputs = [digest], | 
|  | arguments = [digest_args]) | 
|  | return [DefaultInfo(files=depset([digest]))] | 
|  |  | 
|  | def _actions_test_impl(target, ctx): | 
|  | action = target.actions[1] # digest action | 
|  | raw_args_out = ctx.actions.declare_file('raw_args_out') | 
|  | param_file_out = ctx.actions.declare_file('param_file_out') | 
|  | ctx.actions.run_shell(inputs = action.inputs, | 
|  | outputs = [raw_args_out], | 
|  | command = "echo \$@ > " + raw_args_out.path, | 
|  | arguments = action.args) | 
|  | ctx.actions.run_shell(inputs = action.inputs, | 
|  | outputs = [param_file_out], | 
|  | command = "cat \$2 > " + param_file_out.path, | 
|  | arguments = action.args) | 
|  | return [OutputGroupInfo(out=[raw_args_out, param_file_out])] | 
|  |  | 
|  | tree_art_rule = rule(implementation = _tree_art_impl, | 
|  | attrs = { | 
|  | "_makes_tree" : attr.label(allow_single_file = True, | 
|  | cfg = "host", | 
|  | executable = True, | 
|  | default = "//package:makes_tree_artifacts.sh"), | 
|  | "_write" : attr.label(allow_single_file = True, | 
|  | cfg = "host", | 
|  | executable = True, | 
|  | default = "//package:write.sh")}) | 
|  |  | 
|  | actions_test_aspect = aspect(implementation = _actions_test_impl) | 
|  | EOF | 
|  |  | 
|  | cat > package/BUILD <<EOF | 
|  | load(":lib.bzl", "tree_art_rule") | 
|  |  | 
|  | tree_art_rule(name = "x") | 
|  | EOF | 
|  |  | 
|  | bazel build package:x || fail "Unexpected build failure" | 
|  |  | 
|  | cat "${PRODUCT_NAME}-bin/package/digest" | grep "a.txt.*b.txt.*c.txt" \ | 
|  | || fail "rule digest does not contain tree artifact args" | 
|  |  | 
|  | bazel build package:x --aspects=//package:lib.bzl%actions_test_aspect \ | 
|  | --output_groups=out | 
|  |  | 
|  | cat "${PRODUCT_NAME}-bin/package/raw_args_out" | grep ".params" \ | 
|  | || fail "aspect Args does not contain a params file" | 
|  |  | 
|  | cat "${PRODUCT_NAME}-bin/package/param_file_out" | tr '\n' ' ' \ | 
|  | | grep "a.txt.*b.txt.*c.txt" \ | 
|  | || fail "aspect params file does not contain tree artifact args" | 
|  | } | 
|  |  | 
|  | # Test that a starlark action shadowing another action is not rerun after blaze | 
|  | # shutdown. | 
|  | function test_starlark_action_with_shadowed_action_not_rerun_after_shutdown() { | 
|  | local package="a1" | 
|  |  | 
|  | create_starlark_action_with_shadowed_action_cache_test_files "${package}" | 
|  |  | 
|  | bazel build "${package}:a" \ | 
|  | --aspects="//${package}:lib.bzl%actions_test_aspect" \ | 
|  | --output_groups=out || \ | 
|  | fail "bazel build should've succeeded" | 
|  |  | 
|  | cp "bazel-bin/${package}/run_timestamp" "${package}/run_1_timestamp" | 
|  |  | 
|  | # Test that the Starlark action is not rerun after bazel shutdown if | 
|  | # the inputs did not change | 
|  | bazel shutdown | 
|  |  | 
|  | bazel build "${package}:a" \ | 
|  | --aspects="//${package}:lib.bzl%actions_test_aspect" \ | 
|  | --output_groups=out || \ | 
|  | fail "bazel build should've succeeded" | 
|  |  | 
|  | cp "bazel-bin/${package}/run_timestamp" "${package}/run_2_timestamp" | 
|  |  | 
|  | diff "${package}/run_1_timestamp" "${package}/run_2_timestamp" \ | 
|  | || fail "Starlark action should not rerun after bazel shutdown" | 
|  | } | 
|  |  | 
|  | # Test that a starlark action shadowing another action is rerun if the inputs | 
|  | # of the shadowed_action change. | 
|  | function test_starlark_action_rerun_after_shadowed_action_inputs_change() { | 
|  | local package="a2" | 
|  |  | 
|  | create_starlark_action_with_shadowed_action_cache_test_files "${package}" | 
|  |  | 
|  | bazel build "${package}:a" \ | 
|  | --aspects="//${package}:lib.bzl%actions_test_aspect" \ | 
|  | --output_groups=out || \ | 
|  | fail "bazel build should've succeeded" | 
|  |  | 
|  | cp "bazel-bin/${package}/run_timestamp" "${package}/run_1_timestamp" | 
|  |  | 
|  | # Test that the Starlark action would rerun if the inputs of the | 
|  | # shadowed action changed | 
|  | rm -f "${package}/x.h" | 
|  | echo "inline int x() { return 0; }" > "${package}/x.h" | 
|  |  | 
|  | bazel build "${package}:a" \ | 
|  | --aspects="//${package}:lib.bzl%actions_test_aspect" \ | 
|  | --output_groups=out || \ | 
|  | fail "bazel build should've succeeded" | 
|  |  | 
|  | cp "bazel-bin/${package}/run_timestamp" "${package}/run_2_timestamp" | 
|  |  | 
|  | diff "${package}/run_1_timestamp" "${package}/run_2_timestamp" \ | 
|  | && fail "Starlark action should rerun after shadowed action inputs change" \ | 
|  | || : | 
|  | } | 
|  |  | 
|  | function create_starlark_action_with_shadowed_action_cache_test_files() { | 
|  | local package="$1" | 
|  |  | 
|  | mkdir -p "${package}" | 
|  |  | 
|  | cat > "${package}/lib.bzl" <<EOF | 
|  | def _actions_test_impl(target, ctx): | 
|  | compile_action = None | 
|  |  | 
|  | for action in target.actions: | 
|  | if action.mnemonic == "CppCompile": | 
|  | compile_action = action | 
|  |  | 
|  | if not compile_action: | 
|  | fail("Couln't find compile action") | 
|  |  | 
|  | aspect_out = ctx.actions.declare_file("run_timestamp") | 
|  | ctx.actions.run_shell( | 
|  | shadowed_action = compile_action, | 
|  | mnemonic = "AspectAction", | 
|  | outputs = [aspect_out], | 
|  | # record the timestamp in output file to validate action rerun | 
|  | command = "cat ${package}/x.h > %s && date '+%%s' >> %s" % ( | 
|  | aspect_out.path, | 
|  | aspect_out.path, | 
|  | ), | 
|  | ) | 
|  |  | 
|  | return [OutputGroupInfo(out = [aspect_out])] | 
|  |  | 
|  | actions_test_aspect = aspect(implementation = _actions_test_impl) | 
|  | EOF | 
|  |  | 
|  | echo "inline int x() { return 42; }" > "${package}/x.h" | 
|  | cat > "${package}/a.cc" <<EOF | 
|  | #include "${package}/x.h" | 
|  |  | 
|  | int a() { return x(); } | 
|  | EOF | 
|  | cat > "${package}/BUILD" <<EOF | 
|  | cc_library( | 
|  | name = "x", | 
|  | hdrs  = ["x.h"], | 
|  | ) | 
|  |  | 
|  | cc_library( | 
|  | name = "a", | 
|  | srcs = ["a.cc"], | 
|  | deps = [":x"], | 
|  | ) | 
|  | EOF | 
|  | } | 
|  |  | 
|  | function test_aspect_requires_aspect_no_action_conflict() { | 
|  | local package="test" | 
|  | mkdir -p "${package}" | 
|  |  | 
|  | cat > "${package}/write.sh" <<EOF | 
|  | #!/bin/bash | 
|  | output_file=\$1 | 
|  | unused_file=\$2 | 
|  |  | 
|  | echo 'output' > \$output_file | 
|  | echo 'unused' > \$unused_file | 
|  | EOF | 
|  |  | 
|  | chmod 755 "${package}/write.sh" | 
|  |  | 
|  | cat > "${package}/defs.bzl" <<EOF | 
|  | def _aspect_b_impl(target, ctx): | 
|  | file = ctx.actions.declare_file('{}_aspect_b_file.txt'.format(target.label.name)) | 
|  | unused_file = ctx.actions.declare_file('{}.unused'.format(target.label.name)) | 
|  | args = ctx.actions.args() | 
|  | args.add(file.path) | 
|  | args.add(unused_file.path) | 
|  | ctx.actions.run( | 
|  | executable = ctx.executable._write, | 
|  | outputs = [file, unused_file], | 
|  | arguments = [args], | 
|  | # Adding unused_inputs_list just to make this action not shareable | 
|  | unused_inputs_list = unused_file, | 
|  | ) | 
|  | return [OutputGroupInfo(aspect_b_out = [file])] | 
|  |  | 
|  | aspect_b = aspect( | 
|  | implementation = _aspect_b_impl, | 
|  | attrs = { | 
|  | "_write": attr.label( | 
|  | allow_single_file = True, | 
|  | cfg = "host", | 
|  | executable = True, | 
|  | default = "//${package}:write.sh") | 
|  | } | 
|  | ) | 
|  |  | 
|  | def _aspect_a_impl(target, ctx): | 
|  | print(['aspect_a can see file ' + | 
|  | f.path.split('/')[-1] for f in target[OutputGroupInfo].aspect_b_out.to_list()]) | 
|  | files = target[OutputGroupInfo].aspect_b_out.to_list() | 
|  | return [OutputGroupInfo(aspect_a_out = files)] | 
|  |  | 
|  | aspect_a = aspect( | 
|  | implementation = _aspect_a_impl, | 
|  | requires = [aspect_b], | 
|  | attr_aspects = ['dep_2'], | 
|  | ) | 
|  |  | 
|  | def _rule_1_impl(ctx): | 
|  | files = [] | 
|  | if ctx.attr.dep_1: | 
|  | files += ctx.attr.dep_1[OutputGroupInfo].rule_2_out.to_list() | 
|  | if ctx.attr.dep_2: | 
|  | files += ctx.attr.dep_2[OutputGroupInfo].aspect_a_out.to_list() | 
|  | return [DefaultInfo(files=depset(files))] | 
|  |  | 
|  | rule_1 = rule( | 
|  | implementation = _rule_1_impl, | 
|  | attrs = { | 
|  | 'dep_1': attr.label(), | 
|  | 'dep_2': attr.label(aspects = [aspect_a]), | 
|  | } | 
|  | ) | 
|  |  | 
|  | def _rule_2_impl(ctx): | 
|  | files = [] | 
|  | if ctx.attr.dep: | 
|  | print([ctx.label.name + | 
|  | ' can see file ' + | 
|  | f.path.split('/')[-1] for f in ctx.attr.dep[OutputGroupInfo].aspect_b_out.to_list()]) | 
|  | files += ctx.attr.dep[OutputGroupInfo].aspect_b_out.to_list() | 
|  | return [OutputGroupInfo(rule_2_out = files)] | 
|  |  | 
|  | rule_2 = rule( | 
|  | implementation = _rule_2_impl, | 
|  | attrs = { | 
|  | 'dep': attr.label(aspects = [aspect_b]), | 
|  | } | 
|  | ) | 
|  | EOF | 
|  |  | 
|  | cat > "${package}/BUILD" <<EOF | 
|  | load('//${package}:defs.bzl', 'rule_1', 'rule_2') | 
|  | exports_files(["write.sh"]) | 
|  | rule_1( | 
|  | name = 'main_target', | 
|  | dep_1 = ':dep_target_1', | 
|  | dep_2 = ':dep_target_2', | 
|  | ) | 
|  | rule_2( | 
|  | name = 'dep_target_1', | 
|  | dep = ':dep_target_2', | 
|  | ) | 
|  | rule_2(name = 'dep_target_2') | 
|  | EOF | 
|  |  | 
|  | bazel build "//${package}:main_target" &> $TEST_log || fail "Build failed" | 
|  |  | 
|  | expect_log "aspect_a can see file dep_target_2_aspect_b_file.txt" | 
|  | expect_log "dep_target_1 can see file dep_target_2_aspect_b_file.txt" | 
|  | } | 
|  |  | 
|  | run_suite "Tests Starlark API pertaining to action inspection via aspect" |