| #!/usr/bin/env bash | 
 | # | 
 | # Copyright 2015 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. | 
 | # | 
 | # An end-to-end test that Bazel produces runfiles trees as expected. | 
 |  | 
 | # --- 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; } | 
 | source "$(rlocation "io_bazel/src/test/shell/integration/runfiles_test_utils.sh")" \ | 
 |   || { echo "runfiles_test_utils.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 | 
 |  | 
 | # We disable Python toolchains in EXTRA_BUILD_FLAGS because it throws off the | 
 | # counts and manifest checks in test_foo_runfiles. | 
 | # TODO(#8169): Update this test and remove the toolchain opt-out. | 
 | if "$is_windows"; then | 
 |   export EXT=".exe" | 
 |   export EXTRA_STARTUP_FLAGS="--windows_enable_symlinks" | 
 |   export EXTRA_BUILD_FLAGS="--incompatible_use_python_toolchains=false \ | 
 | --enable_runfiles --build_python_zip=0" | 
 | else | 
 |   export EXT="" | 
 |   export EXTRA_STARTUP_FLAGS="" | 
 |   export EXTRA_BUILD_FLAGS="--incompatible_use_python_toolchains=false" | 
 | fi | 
 |  | 
 | #### SETUP ############################################################# | 
 |  | 
 | set -e | 
 |  | 
 | function create_pkg() { | 
 |   local -r pkg=$1 | 
 |   mkdir -p $pkg | 
 |   cd $pkg | 
 |  | 
 |   mkdir -p a/b c/d e/f/g x/y | 
 |   touch py.py a/b/no_module.py c/d/one_module.py c/__init__.py e/f/g/ignored.txt x/y/z.sh | 
 |   chmod +x x/y/z.sh | 
 |  | 
 |   cd .. | 
 |   touch __init__.py | 
 | } | 
 |  | 
 | # This is basically a cross-platform version of `find -printf '%n %y %Y'`. | 
 | # i.e. recursively print paths, their raw file type, and for symbolic links, | 
 | # the type of file the link points to. | 
 | # Macs don't support `find -printf`, and stat, readlink etc all have different | 
 | # args and format specifiers. Basic bash works fine, though. | 
 | function recursive_path_info() { | 
 |   for path in $(find "$1" | sort); do | 
 |     if [[ -L "$path" ]]; then | 
 |       actual_type=symlink | 
 |     else | 
 |       actual_type=regular | 
 |     fi | 
 |     if [[ -f "$path" ]]; then | 
 |       effective_type=file | 
 |     elif [[ -d "$path" ]]; then | 
 |       effective_type="$actual_type dir" | 
 |     else | 
 |       # The various special file types shouldn't occur in practice, so just | 
 |       # call them unknown | 
 |       effective_type=unknown | 
 |     fi | 
 |     echo "$path $effective_type" | 
 |   done | 
 | } | 
 |  | 
 | #### TESTS ############################################################# | 
 |  | 
 | function test_hidden() { | 
 |   local -r pkg=$FUNCNAME | 
 |  | 
 |   mkdir -p "$pkg/e/f/g" | 
 |   touch "$pkg/e/f/g/hidden.txt" | 
 |   cat > "$pkg/defs.bzl" << EOF | 
 | def _obscured_impl(ctx): | 
 |     executable = ctx.actions.declare_file(ctx.label.name) | 
 |     ctx.actions.write(executable, "# nop") | 
 |     return [DefaultInfo( | 
 |         executable = executable, | 
 |         runfiles = ctx.runfiles(files = ctx.files.data), | 
 |     )] | 
 |  | 
 | obscured = rule( | 
 |     implementation = _obscured_impl, | 
 |     attrs = {"data": attr.label_list(allow_files = True)}, | 
 |     # Must be executable to trigger the obscured runfile check | 
 |     executable = True, | 
 | ) | 
 | EOF | 
 |  | 
 |   cat > $pkg/BUILD << EOF | 
 | load(":defs.bzl", "obscured") | 
 | obscured(name="bin", data=["e/f", "e/f/g/hidden.txt"]) | 
 | genrule(name = "hidden", | 
 |         outs = [ "e/f/g/hidden.txt" ], | 
 |         cmd = "touch \$@") | 
 | EOF | 
 |   bazel $EXTRA_STARTUP_FLAGS build $pkg:bin $EXTRA_BUILD_FLAGS >&$TEST_log 2>&1 || fail "build failed" | 
 |  | 
 |   # we get a warning that hidden.txt is inaccessible | 
 |   expect_log_once "${pkg}/e/f/g/hidden.txt obscured by ${pkg}/e/f " | 
 | } | 
 |  | 
 | function test_foo_runfiles() { | 
 |   add_rules_python "MODULE.bazel" | 
 |   add_rules_shell "MODULE.bazel" | 
 |   local WORKSPACE_NAME=$TEST_WORKSPACE | 
 |   local -r pkg=$FUNCNAME | 
 |   create_pkg $pkg | 
 | cat > BUILD << EOF | 
 | load("@rules_python//python:py_library.bzl", "py_library") | 
 |  | 
 | py_library(name = "root", | 
 |            srcs = ["__init__.py"], | 
 |            visibility = ["//visibility:public"]) | 
 | EOF | 
 | cat > $pkg/BUILD << EOF | 
 | load("@rules_python//python:py_binary.bzl", "py_binary") | 
 | load("@rules_shell//shell:sh_binary.bzl", "sh_binary") | 
 |  | 
 | sh_binary(name = "foo", | 
 |           srcs = [ "x/y/z.sh" ], | 
 |           data = [ ":py", | 
 |                    "e/f" ]) | 
 | py_binary(name = "py", | 
 |           srcs = [ "py.py", | 
 |                    "a/b/no_module.py", | 
 |                    "c/d/one_module.py", | 
 |                    "c/__init__.py", | 
 |                  ], | 
 |           data = ["e/f/g/ignored.txt"], | 
 |           deps = ["//:root"]) | 
 | EOF | 
 |   bazel $EXTRA_STARTUP_FLAGS build $pkg:foo $EXTRA_BUILD_FLAGS >&$TEST_log || fail "build failed" | 
 |   workspace_root=$PWD | 
 |  | 
 |   cd ${PRODUCT_NAME}-bin/$pkg/foo${EXT}.runfiles | 
 |  | 
 |   # workaround until we use assert/fail macros in the tests below | 
 |   touch $TEST_TMPDIR/__fail | 
 |  | 
 |   # output manifest exists and is non-empty | 
 |   test    -f MANIFEST | 
 |   test    -s MANIFEST | 
 |  | 
 |   cd ${WORKSPACE_NAME} | 
 |  | 
 |  | 
 |   cd $pkg | 
 |  | 
 |   # these are real empty files | 
 |   test \! -s a/__init__.py | 
 |   test \! -s a/b/__init__.py | 
 |   test \! -s c/d/__init__.py | 
 |   test \! -s __init__.py | 
 |   cd .. | 
 |  | 
 |   # These are basically tuples of (path filetype) | 
 |   expected=" | 
 | . regular dir | 
 | ./__init__.py file | 
 | ./test_foo_runfiles regular dir | 
 | ./test_foo_runfiles/__init__.py file | 
 | ./test_foo_runfiles/a regular dir | 
 | ./test_foo_runfiles/a/__init__.py file | 
 | ./test_foo_runfiles/a/b regular dir | 
 | ./test_foo_runfiles/a/b/__init__.py file | 
 | ./test_foo_runfiles/a/b/no_module.py file | 
 | ./test_foo_runfiles/c regular dir | 
 | ./test_foo_runfiles/c/__init__.py file | 
 | ./test_foo_runfiles/c/d regular dir | 
 | ./test_foo_runfiles/c/d/__init__.py file | 
 | ./test_foo_runfiles/c/d/one_module.py file | 
 | ./test_foo_runfiles/e regular dir | 
 | ./test_foo_runfiles/e/f symlink dir | 
 | ./test_foo_runfiles/foo file | 
 | ./test_foo_runfiles/py file | 
 | ./test_foo_runfiles/py.py file | 
 | ./test_foo_runfiles/x regular dir | 
 | ./test_foo_runfiles/x/y regular dir | 
 | ./test_foo_runfiles/x/y/z.sh file | 
 | " | 
 |   expected="$expected$(get_python_runtime_runfiles)" | 
 |  | 
 |   # For shell binary and python binary, we build both `bin` and `bin.exe`, | 
 |   # but on Linux we only build `bin`. | 
 |   if "$is_windows"; then | 
 |     expected="${expected} | 
 | ./test_foo_runfiles/py.exe file | 
 | ./test_foo_runfiles/foo.exe file | 
 | " | 
 |   fi | 
 |  | 
 |   # Sort and delete empty lines. This makes it easier to append to the | 
 |   # expected string and not have to worry about stray newlines from shell | 
 |   # commands and quoting. | 
 |   expected=$(sort <<<"$expected" | sed '/^$/d') | 
 |   actual=$(recursive_path_info .) | 
 |   assert_equals "$expected" "$actual" | 
 |  | 
 |   # The manifest only records files and symlinks, not real directories | 
 |   expected="$expected$(get_repo_mapping_manifest_file)" | 
 |   expected_manifest_size=$(echo "$expected" | grep -v ' regular dir' | wc -l) | 
 |   actual_manifest_size=$(wc -l < ../MANIFEST) | 
 |   assert_equals $expected_manifest_size $actual_manifest_size | 
 |  | 
 |   # that accounts for everything | 
 |   cd .. | 
 |  | 
 |   for i in $(find ${WORKSPACE_NAME} \! -type d); do | 
 |     target="$(readlink "$i" || true)" | 
 |     if [[ -z "$target" ]]; then | 
 |       echo "$i " >> ${TEST_TMPDIR}/MANIFEST2 | 
 |     else | 
 |       if "$is_windows"; then | 
 |         echo "$i $(cygpath -m $target)" >> ${TEST_TMPDIR}/MANIFEST2 | 
 |       else | 
 |         echo "$i $target" >> ${TEST_TMPDIR}/MANIFEST2 | 
 |       fi | 
 |     fi | 
 |   done | 
 |  | 
 |   # Add the repo mapping manifest entry for Bazel. | 
 |   if [[ "$PRODUCT_NAME" == "bazel" ]]; then | 
 |     repo_mapping="_repo_mapping" | 
 |     repo_mapping_target="$(readlink "$repo_mapping")" | 
 |     if "$is_windows"; then | 
 |       repo_mapping_target="$(cygpath -m $repo_mapping_target)" | 
 |     fi | 
 |     echo "$repo_mapping $repo_mapping_target" >> ${TEST_TMPDIR}/MANIFEST2 | 
 |   fi | 
 |  | 
 |   sort MANIFEST > ${TEST_TMPDIR}/MANIFEST_sorted | 
 |   sort ${TEST_TMPDIR}/MANIFEST2 > ${TEST_TMPDIR}/MANIFEST2_sorted | 
 |   diff -u ${TEST_TMPDIR}/MANIFEST_sorted ${TEST_TMPDIR}/MANIFEST2_sorted | 
 |  | 
 |   # Rebuild the same target with a new dependency. | 
 |   cd "$workspace_root" | 
 | cat > $pkg/BUILD << EOF | 
 | load("@rules_shell//shell:sh_binary.bzl", "sh_binary") | 
 | sh_binary(name = "foo", | 
 |           srcs = [ "x/y/z.sh" ], | 
 |           data = [ "e/f" ]) | 
 | EOF | 
 |   bazel $EXTRA_STARTUP_FLAGS build $pkg:foo $EXTRA_BUILD_FLAGS >&$TEST_log || fail "build failed" | 
 |  | 
 |   cd ${PRODUCT_NAME}-bin/$pkg/foo${EXT}.runfiles | 
 |  | 
 |   # workaround until we use assert/fail macros in the tests below | 
 |   touch $TEST_TMPDIR/__fail | 
 |  | 
 |   # output manifest exists and is non-empty | 
 |   test    -f MANIFEST | 
 |   test    -s MANIFEST | 
 |  | 
 |   cd ${WORKSPACE_NAME} | 
 |  | 
 |   # these are real directories | 
 |   test \! -L $pkg | 
 |   test    -d $pkg | 
 |  | 
 |   # these directory should not exist anymore | 
 |   test \! -e a | 
 |   test \! -e c | 
 |  | 
 |   cd $pkg | 
 |   test \! -L e | 
 |   test    -d e | 
 |   test \! -L x | 
 |   test    -d x | 
 |   test \! -L x/y | 
 |   test    -d x/y | 
 |  | 
 |   # these are symlinks to the source tree | 
 |   test    -L foo | 
 |   test    -L x/y/z.sh | 
 |   test    -L e/f | 
 |   test    -d e/f | 
 |  | 
 |   # that accounts for everything | 
 |   cd ../.. | 
 |   # For shell binary, we build both `bin` and `bin.exe`, but on Linux we only build `bin` | 
 |   # That's why we have one more symlink on Windows. | 
 |   if "$is_windows"; then | 
 |     assert_equals  4 $(find ${WORKSPACE_NAME} -type l | wc -l) | 
 |     assert_equals  0 $(find ${WORKSPACE_NAME} -type f | wc -l) | 
 |     assert_equals  5 $(find ${WORKSPACE_NAME} -type d | wc -l) | 
 |     assert_equals  9 $(find ${WORKSPACE_NAME} | wc -l) | 
 |     if [[ "$PRODUCT_NAME" == "bazel" ]]; then | 
 |       assert_equals  5 $(wc -l < MANIFEST) | 
 |     else | 
 |       assert_equals  4 $(wc -l < MANIFEST) | 
 |     fi | 
 |   else | 
 |     assert_equals  3 $(find ${WORKSPACE_NAME} -type l | wc -l) | 
 |     assert_equals  0 $(find ${WORKSPACE_NAME} -type f | wc -l) | 
 |     assert_equals  5 $(find ${WORKSPACE_NAME} -type d | wc -l) | 
 |     assert_equals  8 $(find ${WORKSPACE_NAME} | wc -l) | 
 |     if [[ "$PRODUCT_NAME" == "bazel" ]]; then | 
 |       assert_equals  4 $(wc -l < MANIFEST) | 
 |     else | 
 |       assert_equals  3 $(wc -l < MANIFEST) | 
 |     fi | 
 |   fi | 
 |  | 
 |   rm -f ${TEST_TMPDIR}/MANIFEST | 
 |   rm -f ${TEST_TMPDIR}/MANIFEST2 | 
 |   for i in $(find ${WORKSPACE_NAME} \! -type d); do | 
 |     target="$(readlink "$i" || true)" | 
 |     if [[ -z "$target" ]]; then | 
 |       echo "$i " >> ${TEST_TMPDIR}/MANIFEST2 | 
 |     else | 
 |       if "$is_windows"; then | 
 |         echo "$i $(cygpath -m $target)" >> ${TEST_TMPDIR}/MANIFEST2 | 
 |       else | 
 |         echo "$i $target" >> ${TEST_TMPDIR}/MANIFEST2 | 
 |       fi | 
 |     fi | 
 |   done | 
 |  | 
 |   # Add the repo mapping manifest entry for Bazel. | 
 |   if [[ "$PRODUCT_NAME" == "bazel" ]]; then | 
 |     repo_mapping="_repo_mapping" | 
 |     repo_mapping_target="$(readlink "$repo_mapping")" | 
 |     if "$is_windows"; then | 
 |       repo_mapping_target="$(cygpath -m $repo_mapping_target)" | 
 |     fi | 
 |     echo "$repo_mapping $repo_mapping_target" >> ${TEST_TMPDIR}/MANIFEST2 | 
 |   fi | 
 |  | 
 |   sort MANIFEST > ${TEST_TMPDIR}/MANIFEST_sorted | 
 |   sort ${TEST_TMPDIR}/MANIFEST2 > ${TEST_TMPDIR}/MANIFEST2_sorted | 
 |   diff -u ${TEST_TMPDIR}/MANIFEST_sorted ${TEST_TMPDIR}/MANIFEST2_sorted | 
 | } | 
 |  | 
 | # regression test for b/237547165 | 
 | function test_fail_on_runfiles_tree_in_transitive_runfiles_for_executable() { | 
 |   local exit_code | 
 |  | 
 |   cat > rule.bzl <<EOF | 
 | def _impl(ctx): | 
 |     exe = ctx.actions.declare_file(ctx.label.name + '.out') | 
 |     ctx.actions.write(exe, "") | 
 |     internal_outputs = ctx.attr.bin[OutputGroupInfo]._hidden_top_level_INTERNAL_ | 
 |     runfiles = ctx.runfiles(transitive_files = internal_outputs) | 
 |     return DefaultInfo(runfiles = runfiles, executable = exe) | 
 | bad_runfiles = rule( | 
 |   implementation = _impl, | 
 |   attrs = {"bin" : attr.label()}, | 
 |   executable = True, | 
 | ) | 
 | EOF | 
 |   cat > BUILD <<EOF | 
 | load(":rule.bzl", "bad_runfiles"); | 
 | cc_binary( | 
 |     name = "thing", | 
 |     srcs = ["thing.cc"], | 
 | ) | 
 | bad_runfiles(name = "test", bin = ":thing") | 
 | EOF | 
 |   cat > thing.cc <<EOF | 
 | int main() { return 0; } | 
 | EOF | 
 |   bazel build //:test &> $TEST_log || exit_code=$? | 
 |   if [[ $exit_code -ne 1 ]]; then | 
 |     fail "Expected regular build failure but instead got exit code $exit_code" | 
 |   fi | 
 |   expect_log_once "Runfiles must not contain runfiles tree artifacts" | 
 | } | 
 |  | 
 | function test_manifest_action_reruns_on_output_base_change() { | 
 |   add_rules_shell "MODULE.bazel" | 
 |   CURRENT_DIRECTORY=$(pwd) | 
 |   if $is_windows; then | 
 |     CURRENT_DIRECTORY=$(cygpath -m "${CURRENT_DIRECTORY}") | 
 |   fi | 
 |  | 
 |   if $is_windows; then | 
 |     MANIFEST_PATH=bazel-bin/hello_world.exe.runfiles_manifest | 
 |   else | 
 |     MANIFEST_PATH=bazel-bin/hello_world.runfiles_manifest | 
 |   fi | 
 |  | 
 |   OUTPUT_BASE="${CURRENT_DIRECTORY}/test/outputs/__main__" | 
 |   TEST_FOLDER_1="${CURRENT_DIRECTORY}/test/test1/$(basename ${CURRENT_DIRECTORY})" | 
 |   TEST_FOLDER_2="${CURRENT_DIRECTORY}/test/test2/$(basename ${CURRENT_DIRECTORY})" | 
 |  | 
 |   mkdir -p "${OUTPUT_BASE}" | 
 |   mkdir -p "${TEST_FOLDER_1}" | 
 |   mkdir -p "${TEST_FOLDER_2}" | 
 |  | 
 |   cat > BUILD <<EOF | 
 | load("@rules_shell//shell:sh_binary.bzl", "sh_binary") | 
 | sh_binary( | 
 |     name = "hello_world", | 
 |     srcs = ["hello_world.sh"], | 
 | ) | 
 | EOF | 
 |   cat > hello_world.sh <<EOF | 
 | echo "Hello World" | 
 | EOF | 
 |   chmod +x hello_world.sh | 
 |  | 
 |   for d in $(ls -a | grep -v '^test$' | grep -v '^\.*$'); do | 
 |     cp -R "${CURRENT_DIRECTORY}/${d}" "${TEST_FOLDER_1}" | 
 |     cp -R "${CURRENT_DIRECTORY}/${d}" "${TEST_FOLDER_2}" | 
 |   done | 
 |  | 
 |   cd "${TEST_FOLDER_1}" | 
 |   bazel --output_base="${OUTPUT_BASE}" build //:hello_world | 
 |   assert_contains "${TEST_FOLDER_1}" "${MANIFEST_PATH}" | 
 |   assert_not_contains "${TEST_FOLDER_2}" "${MANIFEST_PATH}" | 
 |  | 
 |   cd "${TEST_FOLDER_2}" | 
 |   bazel --output_base="${OUTPUT_BASE}" build //:hello_world | 
 |   assert_not_contains "${TEST_FOLDER_1}" "${MANIFEST_PATH}" | 
 |   assert_contains "${TEST_FOLDER_2}" "${MANIFEST_PATH}" | 
 | } | 
 |  | 
 | function test_removal_of_old_tempfiles() { | 
 |   add_rules_shell "MODULE.bazel" | 
 |   cat > BUILD << EOF | 
 | load("@rules_shell//shell:sh_binary.bzl", "sh_binary") | 
 | sh_binary( | 
 |     name = "foo", | 
 |     srcs = ["foo.sh"], | 
 | ) | 
 | EOF | 
 |   touch foo.sh | 
 |   chmod +x foo.sh | 
 |  | 
 |   # Build once to create a runfiles directory. | 
 |   bazel $EXTRA_STARTUP_FLAGS build //:foo $EXTRA_BUILD_FLAGS >&$TEST_log || fail "build failed" | 
 |  | 
 |   # Remove the MANIFEST file that was created by the previous build. | 
 |   # Create an inaccessible file in the place where build-runfiles writes | 
 |   # its temporary results. | 
 |   # | 
 |   # This simulates the case where the runfiles creation process is | 
 |   # interrupted and leaves the temporary file behind. The temporary file | 
 |   # may become read-only if it was stored in a snapshot. | 
 |   rm ${PRODUCT_NAME}-bin/foo${EXT}.runfiles/MANIFEST | 
 |   touch ${PRODUCT_NAME}-bin/foo${EXT}.runfiles/MANIFEST.tmp | 
 |   chmod 0 ${PRODUCT_NAME}-bin/foo${EXT}.runfiles/MANIFEST.tmp | 
 |  | 
 |   # Even with the inaccessible temporary file in place, build-runfiles | 
 |   # should complete successfully. The MANIFEST file should be recreated. | 
 |   bazel $EXTRA_STARTUP_FLAGS build //:foo $EXTRA_BUILD_FLAGS >&$TEST_log || fail "build failed" | 
 |   [[ -f ${PRODUCT_NAME}-bin/foo${EXT}.runfiles/MANIFEST ]] \ | 
 |     || fail "MANIFEST file not recreated" | 
 | } | 
 |  | 
 | function test_rebuilt_when_mapping_changes { | 
 |   if "$is_windows"; then | 
 |     # Can't do 'ln -s' on Windows | 
 |     return | 
 |   fi | 
 |  | 
 |   mkdir -p a | 
 |   cat > a/a.bzl <<'EOF' | 
 | def _a_impl(ctx): | 
 |     ex = ctx.actions.declare_file("a.sh") | 
 |     r = ctx.runfiles( | 
 |         files = [ex], | 
 |         symlinks = {ctx.attr.link: ctx.file.target}, | 
 |     ) | 
 |     ctx.actions.write(ex, "#!/bin/bash", True) | 
 |     return DefaultInfo( | 
 |         files = depset([ex]), | 
 |         default_runfiles = r, | 
 |         executable = ex, | 
 |     ) | 
 |  | 
 | a = rule( | 
 |     implementation = _a_impl, | 
 |     executable = True, | 
 |     attrs = { | 
 |         "link": attr.string(), | 
 |         "target": attr.label(allow_single_file = True), | 
 |     }, | 
 | ) | 
 | EOF | 
 |  | 
 |   cat >a/BUILD <<'EOF' | 
 | load(":a.bzl", "a") | 
 | a( | 
 |   name = "a", | 
 |   link = "link_one", | 
 |   target = ":f", | 
 | ) | 
 |  | 
 | genrule( | 
 |   name = "g", | 
 |   srcs = [], | 
 |   outs = ["go"], | 
 |   output_to_bindir = 1, | 
 |   tools = [":a"], | 
 |   cmd = "echo $(location :a).runfiles/*/link_* > $@", | 
 | ) | 
 | EOF | 
 |  | 
 |   touch a/f | 
 |  | 
 |   bazel build //a:g || fail "first build failed" | 
 |   assert_contains '/link_one$' *-bin/a/go | 
 |  | 
 |   inplace-sed 's/link_one/link_two/' a/BUILD | 
 |   bazel build //a:g || fail "first build failed" | 
 |   assert_contains '/link_two$' *-bin/a/go | 
 | } | 
 |  | 
 | function test_special_chars_in_runfiles_source_paths() { | 
 |   add_rules_shell "MODULE.bazel" | 
 |  | 
 |   mkdir -p pkg | 
 |   if "$is_windows"; then | 
 |     cat > pkg/constants.bzl <<'EOF' | 
 | NAME = "pkg/a b .txt" | 
 | EOF | 
 |   else | 
 |     cat > pkg/constants.bzl <<'EOF' | 
 | NAME = "pkg/a \n \\ b .txt" | 
 | EOF | 
 |   fi | 
 |   cat > pkg/defs.bzl <<'EOF' | 
 | load(":constants.bzl", "NAME") | 
 | def _special_chars_impl(ctx): | 
 |     out = ctx.actions.declare_file("data.txt") | 
 |     ctx.actions.write(out, "my content") | 
 |     runfiles = ctx.runfiles( | 
 |         symlinks = { | 
 |             NAME: out, | 
 |         }, | 
 |     ) | 
 |     return [DefaultInfo(files = depset([out]), runfiles = runfiles)] | 
 |  | 
 | spaces = rule( | 
 |     implementation = _special_chars_impl, | 
 | ) | 
 | EOF | 
 |   cat > pkg/BUILD <<'EOF' | 
 | load(":defs.bzl", "spaces") | 
 | load("@rules_shell//shell:sh_test.bzl", "sh_test") | 
 |  | 
 | spaces(name = "spaces") | 
 | sh_test( | 
 |     name = "foo", | 
 |     srcs = ["foo.sh"], | 
 |     data = [":spaces"], | 
 | ) | 
 | EOF | 
 |   if "$is_windows"; then | 
 |     cat > pkg/foo.sh <<'EOF' | 
 | #!/usr/bin/env bash | 
 | if [[ "$(cat $'pkg/a b .txt')" != "my content" ]]; then | 
 |   echo "unexpected content or not found" | 
 |   exit 1 | 
 | fi | 
 | EOF | 
 |   else | 
 |     cat > pkg/foo.sh <<'EOF' | 
 | #!/usr/bin/env bash | 
 | if [[ "$(cat $'pkg/a \n \\ b .txt')" != "my content" ]]; then | 
 |   echo "unexpected content or not found" | 
 |   exit 1 | 
 | fi | 
 | EOF | 
 |   fi | 
 |   chmod +x pkg/foo.sh | 
 |  | 
 |   bazel test --test_output=errors \ | 
 |     //pkg:foo $EXTRA_BUILD_FLAGS >&$TEST_log || fail "test failed" | 
 | } | 
 |  | 
 | function test_special_chars_in_runfiles_target_paths() { | 
 |   add_rules_shell "MODULE.bazel" | 
 |   mkdir -p pkg | 
 |   if "$is_windows"; then | 
 |     cat > pkg/constants.bzl <<'EOF' | 
 | NAME = "pkg/a b .txt" | 
 | EOF | 
 |   else | 
 |     cat > pkg/constants.bzl <<'EOF' | 
 | NAME = "pkg/a \n \\ b .txt" | 
 | EOF | 
 |   fi | 
 |   cat > pkg/defs.bzl <<'EOF' | 
 | load(":constants.bzl", "NAME") | 
 | def _special_chars_impl(ctx): | 
 |     out = ctx.actions.declare_file(NAME) | 
 |     ctx.actions.write(out, "my content") | 
 |     runfiles = ctx.runfiles( | 
 |         symlinks = { | 
 |             "pkg/data.txt": out, | 
 |         }, | 
 |     ) | 
 |     return [DefaultInfo(files = depset([out]), runfiles = runfiles)] | 
 |  | 
 | spaces = rule( | 
 |     implementation = _special_chars_impl, | 
 | ) | 
 | EOF | 
 |   cat > pkg/BUILD <<'EOF' | 
 | load(":defs.bzl", "spaces") | 
 | load("@rules_shell//shell:sh_test.bzl", "sh_test") | 
 |  | 
 | spaces(name = "spaces") | 
 | sh_test( | 
 |     name = "foo", | 
 |     srcs = ["foo.sh"], | 
 |     data = [":spaces"], | 
 | ) | 
 | EOF | 
 |   cat > pkg/foo.sh <<'EOF' | 
 | #!/usr/bin/env bash | 
 | if [[ "$(cat pkg/data.txt)" != "my content" ]]; then | 
 |   echo "unexpected content or not found" | 
 |   exit 1 | 
 | fi | 
 | EOF | 
 |   chmod +x pkg/foo.sh | 
 |  | 
 |   bazel test --test_output=errors \ | 
 |     //pkg:foo $EXTRA_BUILD_FLAGS >&$TEST_log || fail "test failed" | 
 | } | 
 |  | 
 | function test_special_chars_in_runfiles_source_and_target_paths() { | 
 |   add_rules_shell "MODULE.bazel" | 
 |   mkdir -p pkg | 
 |   if "$is_windows"; then | 
 |     cat > pkg/constants.bzl <<'EOF' | 
 | NAME = "a b .txt" | 
 | EOF | 
 |   else | 
 |     cat > pkg/constants.bzl <<'EOF' | 
 | NAME = "a \n \\ b .txt" | 
 | EOF | 
 |   fi | 
 |   cat > pkg/defs.bzl <<'EOF' | 
 | load(":constants.bzl", "NAME") | 
 | def _special_chars_impl(ctx): | 
 |     out = ctx.actions.declare_file(NAME) | 
 |     ctx.actions.write(out, "my content") | 
 |     return [DefaultInfo(files = depset([out]))] | 
 |  | 
 | spaces = rule( | 
 |     implementation = _special_chars_impl, | 
 | ) | 
 | EOF | 
 |   cat > pkg/BUILD <<'EOF' | 
 | load("@rules_shell//shell:sh_test.bzl", "sh_test") | 
 | load(":defs.bzl", "spaces") | 
 |  | 
 | spaces(name = "spaces") | 
 | sh_test( | 
 |     name = "foo", | 
 |     srcs = ["foo.sh"], | 
 |     data = [":spaces"], | 
 | ) | 
 | EOF | 
 |   if "$is_windows"; then | 
 |     cat > pkg/foo.sh <<'EOF' | 
 | #!/usr/bin/env bash | 
 | if [[ "$(cat $'pkg/a b .txt')" != "my content" ]]; then | 
 |   echo "unexpected content or not found" | 
 |   exit 1 | 
 | fi | 
 | EOF | 
 |   else | 
 |     cat > pkg/foo.sh <<'EOF' | 
 | #!/usr/bin/env bash | 
 | if [[ "$(cat $'pkg/a \n \\ b .txt')" != "my content" ]]; then | 
 |   echo "unexpected content or not found" | 
 |   exit 1 | 
 | fi | 
 | EOF | 
 |   fi | 
 |   chmod +x pkg/foo.sh | 
 |  | 
 |   bazel test --test_output=errors \ | 
 |     //pkg:foo $EXTRA_BUILD_FLAGS >&$TEST_log || fail "test failed" | 
 | } | 
 |  | 
 | # Verify that Bazel's runfiles manifest is compatible with v3 of the Bash | 
 | # runfiles library snippet, even if the workspace path contains a space and | 
 | # a backslash. | 
 | function test_compatibility_with_bash_runfiles_library_snippet() { | 
 |   if [[ "${PRODUCT_NAME}" != "bazel" ]]; then | 
 |     # This test is only relevant for Bazel. | 
 |     return | 
 |   fi | 
 |   # Create a workspace path with a space. | 
 |   add_rules_shell "MODULE.bazel" | 
 |   WORKSPACE="$(mktemp -d jar_manifest.XXXXXXXX)/my w\orkspace" | 
 |   trap "rm -fr '$WORKSPACE'" EXIT | 
 |   mkdir -p "$WORKSPACE" | 
 |   cd "$WORKSPACE" || fail "failed to cd to $WORKSPACE" | 
 |   cat > MODULE.bazel <<'EOF' | 
 | module(name = "my_module") | 
 | EOF | 
 |   add_rules_shell "MODULE.bazel" | 
 |   mkdir pkg | 
 |   cat > pkg/BUILD <<'EOF' | 
 | load("@rules_shell//shell:sh_binary.bzl", "sh_binary") | 
 |  | 
 | sh_binary( | 
 |     name = "tool", | 
 |     srcs = ["tool.sh"], | 
 |     deps = ["@bazel_tools//tools/bash/runfiles"], | 
 | ) | 
 |  | 
 | genrule( | 
 |     name = "gen", | 
 |     outs = ["out"], | 
 |     tools = [":tool"], | 
 |     cmd = "$(execpath :tool) $@", | 
 | ) | 
 | EOF | 
 |   cat > pkg/tool.sh <<'EOF' | 
 | #!/usr/bin/env bash | 
 | # --- begin runfiles.bash initialization v3 --- | 
 | # Copy-pasted from the Bazel Bash runfiles library v3. | 
 | set -uo pipefail; set +e; f=bazel_tools/tools/bash/runfiles/runfiles.bash | 
 | # shellcheck disable=SC1090 | 
 | source "${RUNFILES_DIR:-/dev/null}/$f" 2>/dev/null || \ | 
 |   source "$(grep -sm1 "^$f " "${RUNFILES_MANIFEST_FILE:-/dev/null}" | cut -f2- -d' ')" 2>/dev/null || \ | 
 |   source "$0.runfiles/$f" 2>/dev/null || \ | 
 |   source "$(grep -sm1 "^$f " "$0.runfiles_manifest" | cut -f2- -d' ')" 2>/dev/null || \ | 
 |   source "$(grep -sm1 "^$f " "$0.exe.runfiles_manifest" | cut -f2- -d' ')" 2>/dev/null || \ | 
 |   { echo>&2 "ERROR: cannot find $f"; exit 1; }; f=; set -e | 
 | # --- end runfiles.bash initialization v3 --- | 
 |  | 
 | if [[ ! -z "${RUNFILES_DIR+x}" ]]; then | 
 |   echo "RUNFILES_DIR is set" | 
 |   exit 1 | 
 | fi | 
 |  | 
 | if [[ -z "${RUNFILES_MANIFEST_FILE+x}" ]]; then | 
 |   echo "RUNFILES_MANIFEST_FILE is not set" | 
 |   exit 1 | 
 | fi | 
 |  | 
 | if [[ -z "$(rlocation "my_module/pkg/tool.sh")" ]]; then | 
 |   echo "rlocation failed" | 
 |   exit 1 | 
 | fi | 
 |  | 
 | touch $1 | 
 | EOF | 
 |   chmod +x pkg/tool.sh | 
 |  | 
 |   bazel build --noenable_runfiles \ | 
 |     --spawn_strategy=local \ | 
 |     --action_env=RUNFILES_LIB_DEBUG=1 \ | 
 |     //pkg:gen >&$TEST_log || fail "build failed" | 
 | } | 
 |  | 
 | run_suite "runfiles" |