|  | #!/bin/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. | 
|  |  | 
|  | set -euo pipefail | 
|  |  | 
|  | # Load the test setup defined in the parent directory | 
|  | CURRENT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" | 
|  | source "${CURRENT_DIR}/../integration_test_setup.sh" \ | 
|  | || { echo "integration_test_setup.sh not found!" >&2; exit 1; } | 
|  | source "${CURRENT_DIR}/execution_statistics_utils.sh" \ | 
|  | || { echo "execution_statistics_utils.sh not found!" >&2; exit 1; } | 
|  |  | 
|  | readonly CPU_TIME_SPENDER="${CURRENT_DIR}/../../../test/shell/integration/spend_cpu_time" | 
|  |  | 
|  | readonly OUT_DIR="${TEST_TMPDIR}/out" | 
|  | readonly OUT="${OUT_DIR}/outfile" | 
|  | readonly ERR="${OUT_DIR}/errfile" | 
|  |  | 
|  | readonly EXIT_STATUS_SIGABRT=$((128 + 6)) | 
|  | readonly EXIT_STATUS_SIGKILL=$((128 + 9)) | 
|  | readonly EXIT_STATUS_SIGALRM=$((128 + 14)) | 
|  | readonly EXIT_STATUS_SIGTERM=$((128 + 15)) | 
|  |  | 
|  | function set_up() { | 
|  | rm -rf $OUT_DIR | 
|  | mkdir -p $OUT_DIR | 
|  | } | 
|  |  | 
|  | function assert_stdout() { | 
|  | assert_equals "$1" "$(cat $OUT)" | 
|  | } | 
|  |  | 
|  | function assert_output() { | 
|  | assert_equals "$1" "$(cat $OUT)" | 
|  | assert_equals "$2" "$(cat $ERR)" | 
|  | } | 
|  |  | 
|  | function test_basic_functionality() { | 
|  | $process_wrapper --stdout=$OUT --stderr=$ERR /bin/echo hi there &> $TEST_log || fail | 
|  | assert_output "hi there" "" | 
|  | } | 
|  |  | 
|  | function test_to_stderr() { | 
|  | $process_wrapper --stdout=$OUT --stderr=$ERR /bin/sh -c "/bin/echo hi there >&2" &> $TEST_log || fail | 
|  | assert_output "" "hi there" | 
|  | } | 
|  |  | 
|  | function test_exit_code() { | 
|  | local code=0 | 
|  | $process_wrapper --stdout=$OUT --stderr=$ERR /bin/sh -c "exit 71" &> $TEST_log || code=$? | 
|  | assert_equals 71 "$code" | 
|  | } | 
|  |  | 
|  | function test_signal_death() { | 
|  | local code=0 | 
|  | $process_wrapper --stdout=$OUT --stderr=$ERR /bin/sh -c 'kill -ABRT $$' &> $TEST_log || code=$? | 
|  | assert_equals "${EXIT_STATUS_SIGABRT}" "$code" | 
|  | } | 
|  |  | 
|  | function test_signal_catcher() { | 
|  | local code=0 | 
|  | $process_wrapper --timeout=1 --kill_delay=10 --stdout=$OUT --stderr=$ERR /bin/sh -c \ | 
|  | 'trap "echo later; exit 0" INT TERM ALRM; sleep 10' &> $TEST_log || code=$? | 
|  | assert_equals "${EXIT_STATUS_SIGALRM}" "$code" | 
|  | assert_stdout "later" | 
|  | } | 
|  |  | 
|  | function test_basic_timeout() { | 
|  | $process_wrapper --timeout=1 --kill_delay=2 --stdout=$OUT --stderr=$ERR /bin/sh -c \ | 
|  | "echo before; sleep 10; echo after" &> $TEST_log || code=$? | 
|  | assert_equals "${EXIT_STATUS_SIGALRM}" "$code" | 
|  | assert_stdout "before" | 
|  | } | 
|  |  | 
|  | # Tests that the timeout causes the process to receive a SIGTERM, but with the | 
|  | # process exiting on its own without the need for a SIGKILL. To make sure that | 
|  | # this is the case, we pass a very large kill delay to cause the outer test to | 
|  | # fail if we violate this expectation. | 
|  | function test_timeout_grace() { | 
|  | local code=0 | 
|  | $process_wrapper --timeout=1 --kill_delay=100000 --stdout=$OUT --stderr=$ERR \ | 
|  | /bin/sh -c \ | 
|  | 'trap "echo ignoring signal" INT TERM ARLM; \ | 
|  | for i in $(seq 5); do sleep 1; done; echo after' \ | 
|  | &> $TEST_log || code=$? | 
|  | assert_equals "${EXIT_STATUS_SIGALRM}" "$code" | 
|  | assert_stdout 'ignoring signal | 
|  | after' | 
|  | } | 
|  |  | 
|  | # Tests that the timeout causes the process to receive a SIGTERM and waits until | 
|  | # the process has exited on its own, even if that takes a little bit of time. To | 
|  | # make sure that this is the case, we pass a very large kill delay to cause the | 
|  | # outer test to fail if we violate this expectation. | 
|  | # | 
|  | # In the past, even though we would terminate the process quickly, we would | 
|  | # get stuck until the kill delay passed (because we'd be stuck waiting for a | 
|  | # zombie process without us actually collecting it). So this is a regression | 
|  | # test for that subtle bug. | 
|  | function test_timeout_exits_as_soon_as_process_terminates() { | 
|  | local code=0 | 
|  | $process_wrapper --timeout=1 --kill_delay=100000 --stdout=$OUT --stderr=$ERR \ | 
|  | /bin/sh -c \ | 
|  | 'trap "" INT TERM ARLM; \ | 
|  | for i in $(seq 5); do echo sleeping $i; sleep 1; done' \ | 
|  | &> $TEST_log || code=$? | 
|  | assert_equals "${EXIT_STATUS_SIGALRM}" "$code" | 
|  | assert_stdout 'sleeping 1 | 
|  | sleeping 2 | 
|  | sleeping 3 | 
|  | sleeping 4 | 
|  | sleeping 5' | 
|  | } | 
|  |  | 
|  | # Tests that the timeout causes the process to receive a SIGTERM first and a | 
|  | # SIGKILL later once a kill delay has passed without the process exiting on | 
|  | # its own. We make the process get stuck indefinitely until killed, and we do | 
|  | # this with individual calls to sleep instead of a single one to ensure that a | 
|  | # single termination of the sleep subprocess doesn't cause us to spuriously | 
|  | # exit and thus pass this test. | 
|  | function test_timeout_kill() { | 
|  | local code=0 | 
|  | $process_wrapper --timeout=1 --kill_delay=5 --stdout=$OUT --stderr=$ERR \ | 
|  | /bin/sh -c \ | 
|  | 'trap "echo ignoring signal" INT TERM ARLM; \ | 
|  | while :; do sleep 1; done; echo after' \ | 
|  | &> $TEST_log || code=$? | 
|  | assert_equals "${EXIT_STATUS_SIGALRM}" "$code" | 
|  | assert_stdout 'ignoring signal' | 
|  | } | 
|  |  | 
|  | # Tests that sending a SIGTERM causes the process to receive such SIGTERM if | 
|  | # graceful SIGTERM handling is enabled, but with the process exiting on its own | 
|  | # without the need for a SIGKILL. To make sure that this is the case, we pass a | 
|  | # very large kill delay to cause the outer test to fail if we violate this | 
|  | # expectation. | 
|  | function test_sigterm_grace() { | 
|  | $process_wrapper --graceful_sigterm --kill_delay=100000 \ | 
|  | --stdout=$OUT --stderr=$ERR \ | 
|  | /bin/sh -c \ | 
|  | 'trap "echo ignoring signal" INT TERM ARLM; \ | 
|  | for i in $(seq 5); do sleep 1; done; echo after' \ | 
|  | &> $TEST_log & | 
|  | local pid=$! | 
|  | sleep 1 | 
|  | kill -TERM "${pid}" | 
|  | local code=0 | 
|  | wait "${pid}" || code=$? | 
|  | assert_equals "${EXIT_STATUS_SIGTERM}" "$code" | 
|  | assert_stdout 'ignoring signal | 
|  | after' | 
|  | } | 
|  |  | 
|  | # Tests that sending a SIGTERM causes the process to receive such SIGTERM if | 
|  | # graceful SIGTERM handling is enabled and waits until the process has exited | 
|  | # on its own, even if that takes a little bit of time. To make sure that this is | 
|  | # the case, we pass a very large kill delay to cause the outer test to fail if | 
|  | # we violate this expectation. | 
|  | # | 
|  | # In the past, even though we would terminate the process quickly, we would | 
|  | # get stuck until the kill delay passed (because we'd be stuck waiting for a | 
|  | # zombie process without us actually collecting it). So this is a regression | 
|  | # test for that subtle bug. | 
|  | function test_sigterm_exits_as_soon_as_process_terminates() { | 
|  | $process_wrapper --graceful_sigterm --kill_delay=100000 \ | 
|  | --stdout=$OUT --stderr=$ERR \ | 
|  | /bin/sh -c \ | 
|  | 'trap "" INT TERM ARLM; \ | 
|  | for i in $(seq 5); do echo sleeping $i; sleep 1; done' \ | 
|  | &> $TEST_log & | 
|  | local pid=$! | 
|  | sleep 1 | 
|  | kill -TERM "${pid}" | 
|  | local code=0 | 
|  | wait "${pid}" || code=$? | 
|  | assert_equals "${EXIT_STATUS_SIGTERM}" "$code" | 
|  | assert_stdout 'sleeping 1 | 
|  | sleeping 2 | 
|  | sleeping 3 | 
|  | sleeping 4 | 
|  | sleeping 5' | 
|  | } | 
|  |  | 
|  | # Tests that sending a SIGTERM causes the process to receive such SIGTERM if | 
|  | # graceful SIGTERM handling is enabled and a SIGKILL later once a kill delay has | 
|  | # passed without the process exiting on its own. We make the process get stuck | 
|  | # indefinitely until killed, and we do this with individual calls to sleep | 
|  | # instead of a single one to ensure that a single termination of the sleep | 
|  | # subprocess doesn't cause us to spuriously exit and thus pass this test. | 
|  | function test_sigterm_kill() { | 
|  | $process_wrapper --graceful_sigterm --kill_delay=5 \ | 
|  | --stdout=$OUT --stderr=$ERR \ | 
|  | /bin/sh -c \ | 
|  | 'trap "echo ignoring signal" INT TERM ARLM; \ | 
|  | while :; do sleep 1; done; echo after' \ | 
|  | &> $TEST_log & | 
|  | local pid=$! | 
|  | sleep 1 | 
|  | kill -TERM "${pid}" | 
|  | local code=0 | 
|  | wait "${pid}" || code=$? | 
|  | assert_equals "${EXIT_STATUS_SIGTERM}" "$code" | 
|  | assert_stdout 'ignoring signal' | 
|  | } | 
|  |  | 
|  | function test_execvp_error_message() { | 
|  | local code=0 | 
|  | $process_wrapper --stdout=$OUT --stderr=$ERR /bin/notexisting &> $TEST_log || code=$? | 
|  | assert_equals 1 "$code" | 
|  | assert_contains "\"execvp(/bin/notexisting, ...)\": No such file or directory" "$ERR" | 
|  | } | 
|  |  | 
|  | function assert_process_wrapper_exec_time() { | 
|  | local user_time_low="$1"; shift | 
|  | local user_time_high="$1"; shift | 
|  | local sys_time_low="$1"; shift | 
|  | local sys_time_high="$1"; shift | 
|  |  | 
|  | local local_tmp="$(mktemp -d "${OUT_DIR}/assert_process_wrapper_timeXXXXXX")" | 
|  | local stdout_path="${local_tmp}/stdout" | 
|  | local stderr_path="${local_tmp}/stderr" | 
|  | local stats_out_path="${local_tmp}/statsfile" | 
|  | local stats_out_decoded_path="${local_tmp}/statsfile.decoded" | 
|  |  | 
|  | # Wrapped process will be terminated after 100 seconds if not self terminated. | 
|  | local code=0 | 
|  | "${process_wrapper}" \ | 
|  | --timeout=100 \ | 
|  | --kill_delay=2 \ | 
|  | --stdout="${stdout_path}" \ | 
|  | --stderr="${stderr_path}" \ | 
|  | --stats="${stats_out_path}" \ | 
|  | "${CPU_TIME_SPENDER}" "${user_time_low}" "${sys_time_low}" \ | 
|  | &> "${TEST_log}" || code="$?" | 
|  | sed -e 's,^subprocess stdout: ,,' "${stdout_path}" >>"${TEST_log}" | 
|  | sed -e 's,^subprocess stderr: ,,' "${stderr_path}" >>"${TEST_log}" | 
|  | assert_equals 0 "${code}" | 
|  |  | 
|  | assert_execution_time_in_range \ | 
|  | "${user_time_low}" \ | 
|  | "${user_time_high}" \ | 
|  | "${sys_time_low}" \ | 
|  | "${sys_time_high}" \ | 
|  | "${stats_out_path}" | 
|  | } | 
|  |  | 
|  | function test_stats_high_user_time() { | 
|  | assert_process_wrapper_exec_time 10 19 0 9 | 
|  | } | 
|  |  | 
|  | function test_stats_high_system_time() { | 
|  | assert_process_wrapper_exec_time 0 9 10 19 | 
|  | } | 
|  |  | 
|  | function test_stats_high_user_time_and_high_system_time() { | 
|  | assert_process_wrapper_exec_time 10 25 10 25 | 
|  | } | 
|  |  | 
|  | run_suite "process-wrapper" |