blob: 60357b17579b7d89f2f527e402229463adedbad7 [file] [log] [blame]
#!/bin/bash
#
# Copyright 2016 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's experimental UI produces reasonable output.
# 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; }
add_to_bazelrc "build --experimental_build_event_upload_strategy=local"
#### SETUP #############################################################
set -e
function set_up() {
setup_skylib_support
mkdir -p pkg
touch pkg/somesourcefile
cat > pkg/true.sh <<'EOF'
#!/bin/sh
[ -n "${XML_OUTPUT_FILE}" ] && touch "${XML_OUTPUT_FILE}"
exit 0
EOF
chmod 755 pkg/true.sh
cat > pkg/false.sh <<'EOF'
#!/bin/sh
[ -n "${XML_OUTPUT_FILE}" ] && touch "${XML_OUTPUT_FILE}"
exit 1
EOF
chmod 755 pkg/false.sh
cat > pkg/slowtest.sh <<'EOF'
#!/bin/sh
sleep 1
exit 0
EOF
chmod 755 pkg/slowtest.sh
touch pkg/sourcefileA pkg/sourcefileB pkg/sourcefileC
cat > pkg/BUILD <<'EOF'
exports_files(["somesourcefile"])
sh_test(
name = "true",
srcs = ["true.sh"],
size = "small",
)
sh_test(
name = "slow",
srcs = ["slowtest.sh"],
)
test_suite(
name = "suite",
tests = ["true"],
)
sh_test(
name = "flaky",
srcs = ["false.sh"],
flaky = True,
)
genrule(
name = "fails_to_build",
outs = ["fails_to_build.txt"],
cmd = "echo This build will fail && false",
executable = 1,
)
sh_test(
name = "test_that_fails_to_build",
srcs = [":fails_to_build"],
)
genrule(
name = "output_files_and_tags",
outs = ["out1.txt"],
cmd = "echo foo > \"$@\"",
tags = ["tag1", "tag2"]
)
action_listener(
name = "listener",
mnemonics = ["Genrule"],
extra_actions = [":extra"],
visibility = ["//visibility:public"],
)
extra_action(
name = "extra",
cmd = "echo Hello World",
)
filegroup(
name = "innergroup",
srcs = ["sourcefileA", "sourcefileB"],
)
filegroup(
name = "outergroup",
srcs = ["sourcefileC", ":innergroup"],
)
genrule(
name = "not_a_test",
outs = ["not_a_test.txt"],
cmd = "touch $@",
)
EOF
cat > simpleaspect.bzl <<'EOF'
def _simple_aspect_impl(target, ctx):
for orig_out in ctx.rule.attr.outs:
aspect_out = ctx.actions.declare_file(orig_out.name + ".aspect")
ctx.actions.write(
output=aspect_out,
content = "Hello from aspect")
return struct(output_groups={
"aspect-out" : depset([aspect_out]) })
simple_aspect = aspect(implementation=_simple_aspect_impl)
EOF
cat > failingaspect.bzl <<'EOF'
def _failing_aspect_impl(target, ctx):
for orig_out in ctx.rule.attr.outs:
aspect_out = ctx.actions.declare_file(orig_out.name + ".aspect")
ctx.actions.run_shell(
inputs = [],
outputs = [aspect_out],
command = "false",
)
return struct(output_groups={
"aspect-out" : depset([aspect_out]) })
failing_aspect = aspect(implementation=_failing_aspect_impl)
EOF
touch BUILD
cat > sample_workspace_status <<'EOF'
#!/bin/sh
echo SAMPLE_WORKSPACE_STATUS workspace_status_value
EOF
chmod 755 sample_workspace_status
mkdir -p visibility/hidden
cat > visibility/hidden/BUILD <<'EOF'
genrule(
name = "hello",
outs = ["hello.txt"],
cmd = "echo Hello World > $@",
)
EOF
cat > visibility/BUILD <<'EOF'
genrule(
name = "cannotsee",
outs = ["cannotsee.txt"],
srcs = ["//visibility/hidden:hello"],
cmd = "cp $< $@",
)
genrule(
name = "indirect",
outs = ["indirect.txt"],
srcs = [":cannotsee"],
cmd = "cp $< $@",
)
genrule(
name = "cannotsee2",
outs = ["cannotsee2.txt"],
srcs = ["//visibility/hidden:hello"],
cmd = "cp $< $@",
)
genrule(
name = "indirect2",
outs = ["indirect2.txt"],
srcs = [":cannotsee2.txt"],
cmd = "cp $< $@",
)
EOF
mkdir -p failingtool
cat > failingtool/BUILD <<'EOF'
genrule(
name = "tool",
outs = ["tool.sh"],
cmd = "false",
)
genrule(
name = "usestool",
outs = ["out.txt"],
tools = [":tool"],
cmd = "$(location :tool) > $@",
)
EOF
mkdir -p alias/actual
cat > alias/actual/BUILD <<'EOF'
genrule(
name = "it",
outs = ["it.txt"],
cmd = "touch $@",
visibility = ["//:__subpackages__"],
)
EOF
cat > alias/BUILD <<'EOF'
alias(
name = "it",
actual = "//alias/actual:it",
)
EOF
mkdir -p chain
cat > chain/BUILD <<'EOF'
genrule(
name = "entry0",
outs = ["entry0.txt"],
cmd = "echo Hello0; touch $@",
)
EOF
for i in `seq 1 10`
do
cat >> chain/BUILD <<EOF
genrule(
name = "entry$i",
outs = ["entry$i.txt"],
srcs = [ "entry$(( $i - 1)).txt" ],
cmd = "echo Hello$i; cp \$< \$@",
)
EOF
done
}
#### TESTS #############################################################
function test_basic() {
# Basic properties of the event stream
# - a completed target explicitly requested should be reported
# - after success the stream should close naturally, without any
# reports about aborted events
# - the command line is reported in structured and unstructured form
# - the target_kind is reported
# - for single-configuration builds, there is precisely one configuration
# event reported; also make variables are shown
bazel test -k --build_event_text_file=$TEST_log --tool_tag=MyFancyTool pkg:true \
|| fail "bazel test failed"
expect_log 'pkg:true'
# Command line
expect_log_once 'args: "test"'
expect_log_once 'args: "--build_event_text_file='
expect_log_once 'args: "-k"'
expect_log_once 'args: "--tool_tag=MyFancyTool"'
expect_log_once 'args: "pkg:true"'
# Options parsed. Since cmd_line lines are a substring of the equivalent
# explicit_cmd_line lines, we expect 2 instances for these.
expect_log_n 'cmd_line: "--tool_tag=MyFancyTool"' 2
expect_log_n 'cmd_line: "--keep_going"' 2
expect_log_once 'explicit_cmd_line: "--keep_going"'
expect_log_once 'explicit_cmd_line: "--tool_tag=MyFancyTool"'
expect_log_once 'tool_tag: "MyFancyTool"'
# Structured command line. Expect the explicit flags to appear twice,
# in the canonical and original command lines. We did not pass a tool
# command line, but still expect an empty report.
expect_log 'command_line_label: "original"'
expect_log 'command_line_label: "canonical"'
expect_log 'command_line_label: "tool"'
expect_log_n 'combined_form: "-k"' 2
expect_log_n 'option_name: "keep_going"' 2
expect_log 'option_value: "1"' # too vague to count.
expect_log_n 'combined_form: "--tool_tag=MyFancyTool"' 2
expect_log_n 'option_name: "tool_tag"' 2
expect_log_n 'option_value: "MyFancyTool"' 2
expect_log_n "combined_form: \"--build_event_text_file=${TEST_log}\"" 2
expect_log_n 'option_name: "build_event_text_file"' 2
expect_log_n "option_value: \"${TEST_log}\"" 2
expect_log_n 'chunk: "test"' 2
expect_log_n 'chunk: "pkg:true"' 2
# Build Finished
expect_log 'build_finished'
expect_log 'SUCCESS'
expect_log 'finish_time'
expect_log_once 'last_message: true'
expect_not_log 'aborted'
expect_log_once '^build_tool_logs'
# Target kind for the sh_test
expect_log 'target_kind:.*sh'
# Test size should be reported
expect_log 'test_size: SMALL'
# Configuration reported with make variables
expect_log_once '^configuration '
expect_log 'key: "TARGET_CPU"'
}
function test_target_information_early() {
# Verify that certain information is present in the log as part of
# the TargetConfigured event (verifying that it comes at least before
# the first TargetCompleted event, which is fine, if we only ask for
# a single target).
bazel test --build_event_text_file=$TEST_log pkg:true \
|| fail "bazel test failed"
expect_log '^completed'
ed $TEST_log <<'EOF'
1
/^completed/+1,$d
a
...[cut here]
.
w
q
EOF
expect_log 'target_kind:.*sh'
expect_log 'test_size: SMALL'
bazel build --verbose_failures --build_event_text_file=$TEST_log \
pkg:output_files_and_tags || fail "bazel build failed"
expect_log '^completed'
ed $TEST_log <<'EOF'
1
/^completed/+1,$d
a
...[cut here]
.
w
q
EOF
expect_log 'tag1'
expect_log 'tag2'
}
function test_workspace_status() {
bazel test --build_event_text_file=$TEST_log \
--workspace_status_command=sample_workspace_status pkg:true \
|| fail "bazel test failed"
expect_log_once '^workspace_status'
expect_log 'key.*SAMPLE_WORKSPACE_STATUS'
expect_log 'value.*workspace_status_value'
}
function test_suite() {
# ...same true when running a test suite containing that test
bazel test --build_event_text_file=$TEST_log pkg:suite \
|| fail "bazel test failed"
expect_log 'pkg:true'
expect_not_log 'aborted'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_test_summary() {
# Requesting a test, we expect
# - precisely one test summary (for the single test we run)
# - that is properly chained (no additional progress events)
# - the correct overall status being reported
bazel test --build_event_text_file=$TEST_log pkg:true \
|| fail "bazel test failed"
expect_log_once '^test_summary '
expect_not_log 'aborted'
expect_log 'status.*PASSED'
expect_not_log 'status.*FAILED'
expect_not_log 'status.*FLAKY'
}
function test_test_inidivual_results() {
# Requesting a test, we expect
# - precisely one test summary (for the single test we run)
# - that is properly chained (no additional progress events)
bazel test --build_event_text_file=$TEST_log \
--runs_per_test=2 pkg:true \
|| fail "bazel test failed"
expect_log '^test_result'
expect_log 'run.*1'
expect_log 'status.*PASSED'
expect_log_once '^test_summary '
expect_not_log 'aborted'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_test_attempts() {
# Run a failing test declared as flaky.
# We expect to see 3 attempts to happen, and also find the 3 xml files
# mentioned in the stream.
# Moreover, as the test consistently fails, we expect the overall status
# to be reported as failure.
(bazel test --build_event_text_file=$TEST_log pkg:flaky \
&& fail "test failure expected" ) || true
expect_log 'attempt.*1$'
expect_log 'attempt.*2$'
expect_log 'attempt.*3$'
expect_log_once '^test_summary '
expect_log 'status.*FAILED'
expect_not_log 'status.*PASSED'
expect_not_log 'status.*FLAKY'
expect_not_log 'aborted'
expect_log '^test_result'
expect_log 'test_action_output'
expect_log 'flaky/.*_1.xml'
expect_log 'flaky/.*_2.xml'
expect_log 'flaky/.*test.xml'
expect_log 'name:.*test.log'
expect_log 'name:.*test.xml'
expect_log 'name:.*TESTS_FAILED'
expect_not_log 'name:.*SUCCESS'
}
function test_test_runtime() {
bazel test --build_event_text_file=$TEST_log pkg:slow \
|| fail "bazel test failed"
expect_log 'pkg:slow'
expect_log '^test_result'
expect_log 'test_attempt_duration_millis.*[1-9]'
expect_log 'build_finished'
expect_log 'SUCCESS'
expect_log 'finish_time'
expect_not_log 'aborted'
}
function test_test_start_times() {
# Verify that the start time of a test is reported, regardless whether
# it was cached or not.
bazel clean --expunge
bazel test --build_event_text_file=$TEST_log pkg:true \
|| fail "bazel test failed"
expect_log 'test_attempt_start_millis_epoch.*[1-9]'
expect_not_log 'cached_locally'
bazel test --build_event_text_file=$TEST_log pkg:true \
|| fail "bazel test failed"
expect_log 'test_attempt_start_millis_epoch.*[1-9]'
expect_log 'cached_locally.*true'
}
function test_test_attempts_multi_runs() {
# Sanity check on individual test attempts. Even in more complicated
# situations, with some test rerun and some not, all events are properly
# announced by the test actions (and not chained into the progress events).
( bazel test --build_event_text_file=$TEST_log \
--runs_per_test=2 pkg:true pkg:flaky \
&& fail "test failure expected" ) || true
expect_log 'run.*1'
expect_log 'attempt.*2'
expect_not_log 'aborted'
}
function test_test_attempts_multi_runs_flake_detection() {
# Sanity check on individual test attempts. Even in more complicated
# situations, with some test rerun and some not, all events are properly
# announced by the test actions (and not chained into the progress events).
( bazel test --build_event_text_file=$TEST_log \
--runs_per_test=2 --runs_per_test_detects_flakes pkg:true pkg:flaky \
&& fail "test failure expected" ) || true
expect_log 'run.*1'
expect_log 'attempt.*2'
expect_not_log 'aborted'
}
function test_cached_test_results() {
# Verify that both, clean and cached test results are reported correctly,
# including the appropriate reference to log files.
bazel clean --expunge
bazel test --build_event_text_file=$TEST_log pkg:true \
|| fail "Clean testing pkg:true failed"
expect_log '^test_result'
expect_log 'name:.*test.log'
expect_log 'name:.*test.xml'
expect_not_log 'cached_locally'
expect_not_log 'aborted'
bazel test --build_event_text_file=$TEST_log pkg:true \
|| fail "Second testing of pkg:true failed"
expect_log '^test_result'
expect_log 'name:.*test.log'
expect_log 'name:.*test.xml'
expect_log 'cached_locally'
expect_not_log 'aborted'
}
function test_target_complete() {
bazel build --verbose_failures --build_event_text_file=$TEST_log \
pkg:output_files_and_tags || fail "bazel build failed"
expect_log 'output_group'
expect_log 'out1.txt'
expect_log 'tag1'
expect_log 'tag2'
}
function test_test_target_complete() {
bazel test --build_event_text_file="${TEST_log}" pkg:true \
|| tail "expected success"
expect_log_once '^completed'
cp "${TEST_log}" complete_event
ed complete_event <<'EOF'
1
/^complete
1,.-1d
/^}
+1,$d
w
q
EOF
grep -q 'output_group' complete_event \
|| fail "expected reference to output in complete event"
expect_log 'name: *"pkg/true.sh"'
}
function test_extra_action() {
# verify that normal successful actions are not reported, but extra actions
# are
bazel build --build_event_text_file=$TEST_log \
pkg:output_files_and_tags || fail "bazel build failed"
expect_not_log '^action'
bazel build --build_event_text_file=$TEST_log \
--experimental_action_listener=pkg:listener \
pkg:output_files_and_tags || fail "bazel build with listener failed"
expect_log '^action'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_action_ids() {
bazel build --build_event_text_file=$TEST_log \
--experimental_action_listener=pkg:listener \
pkg:output_files_and_tags || fail "bazel build with listener failed"
expect_log '^action'
# Action ids should contain label and configuration if those exist.
# Assumes action_completed id is 6 lines long
for id_line in $(grep -n 'action_completed {' $TEST_log | cut -f 1 -d :)
do
sed -n "$id_line,$((id_line+6))p" $TEST_log > "$TEST_TMPDIR/event_id.txt"
assert_contains '.*primary_output: .*' "$TEST_TMPDIR/event_id.txt"
assert_contains '.*label: .*' "$TEST_TMPDIR/event_id.txt"
assert_contains '.*configuration.*' "$TEST_TMPDIR/event_id.txt"
assert_contains '.*id: .*' "$TEST_TMPDIR/event_id.txt"
done
}
function test_aspect_artifacts() {
bazel build --build_event_text_file=$TEST_log \
--aspects=simpleaspect.bzl%simple_aspect \
--output_groups=aspect-out \
pkg:output_files_and_tags || fail "bazel build failed"
expect_log 'aspect.*simple_aspect'
expect_log 'name.*aspect-out'
expect_log 'name.*out1.txt.aspect'
expect_not_log 'aborted'
count=`grep '^configured' "${TEST_log}" | wc -l`
[ "${count}" -eq 2 ] || fail "Expected 2 configured events, found $count."
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_failing_aspect() {
(bazel build --build_event_text_file=$TEST_log \
--aspects=failingaspect.bzl%failing_aspect \
--output_groups=aspect-out \
pkg:output_files_and_tags && fail "expected failure") || true
expect_log 'aspect.*failing_aspect'
expect_log '^finished'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_build_only() {
# When building but not testing a test, there won't be a test summary
# (as nothing was tested), so it should not be announced.
# Still, no event should only be chained in by progress events.
bazel build --build_event_text_file=$TEST_log pkg:true \
|| fail "bazel build failed"
expect_not_log 'aborted'
expect_not_log 'test_summary '
# Build Finished
expect_log 'build_finished'
expect_log 'finish_time'
expect_log 'SUCCESS'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_query() {
# Verify that at least a minimally meaningful event stream is generated
# for non-build. In particular, we expect bazel not to crash.
bazel query --build_event_text_file=$TEST_log 'tests(//...)' \
|| fail "bazel query failed"
expect_log '^started'
expect_log 'command: "query"'
expect_log 'args: "--build_event_text_file='
expect_log 'build_finished'
expect_not_log 'aborted'
# For query, we also expect the full output to be contained in the protocol,
# as well as a proper finished event.
expect_log '//pkg:true'
expect_log '//pkg:slow'
expect_log '^finished'
expect_log 'name: "SUCCESS"'
expect_log 'last_message: true'
}
function test_command_whitelisting() {
# We expect the "help" command to not generate a build-event stream,
# but the "build" command to do.
bazel shutdown
rm -f bep.txt
bazel help --build_event_text_file=bep.txt || fail "bazel help failed"
( [ -f bep.txt ] && fail "bazel help generated a build-event file" ) || :
bazel version --build_event_text_file=bep.txt || fail "bazel help failed"
( [ -f bep.txt ] && fail "bazel version generated a build-event file" ) || :
bazel build --build_event_text_file=bep.txt //pkg:true \
|| fail "bazel build failed"
[ -f bep.txt ] || fail "build did not generate requested build-event file"
bazel shutdown
}
function test_multiple_transports() {
# Verifies usage of multiple build event transports at the same time
outdir=$(mktemp -d ${TEST_TMPDIR}/bazel.XXXXXXXX)
bazel test \
--build_event_text_file=${outdir}/test_multiple_transports.txt \
--build_event_binary_file=${outdir}/test_multiple_transports.bin \
--build_event_json_file=${outdir}/test_multiple_transports.json \
pkg:suite || fail "bazel test failed"
[ -f ${outdir}/test_multiple_transports.txt ] || fail "Missing expected file test_multiple_transports.txt"
[ -f ${outdir}/test_multiple_transports.bin ] || fail "Missing expected file test_multiple_transports.bin"
[ -f ${outdir}/test_multiple_transports.json ] || fail "Missing expected file test_multiple_transports.bin"
}
function test_basic_json() {
# Verify that the json transport writes json files
bazel test --build_event_json_file=$TEST_log pkg:true \
|| fail "bazel test failed"
# check for some typical fragments that would be encoded differently in the
# proto text format.
expect_log '"started"'
expect_log '"id"'
expect_log '"children" *: *\['
expect_log '"overallSuccess": *true'
}
function test_root_cause_early() {
(bazel build --build_event_text_file=$TEST_log \
pkg:fails_to_build && fail "build failure expected") || true
# We expect precisely one action being reported (the failed one) and
# precisely on report on a completed target; moreover, the action has
# to be reported first.
expect_log_once '^action'
expect_log 'type: "Genrule"'
expect_log_once '^completed'
expect_not_log 'success: true'
local naction=`grep -n '^action' $TEST_log | cut -f 1 -d :`
local ncomplete=`grep -n '^completed' $TEST_log | cut -f 1 -d :`
[ $naction -lt $ncomplete ] \
|| fail "failed action not before completed target"
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_action_conf() {
# Verify that the expected configurations for actions are reported.
# The example contains a configuration transition (from building for
# target to building for host). As the action fails, we expect the
# configuration of the action to be reported as well.
(bazel build --build_event_text_file=$TEST_log \
-k failingtool/... && fail "build failure expected") || true
count=`grep '^configuration' "${TEST_log}" | wc -l`
[ "${count}" -eq 2 ] || fail "Expected 2 configurations, found $count."
}
function test_loading_failure() {
# Verify that if loading fails, this is properly reported as the
# reason for the target expansion event not resulting in targets
# being expanded.
(bazel build --build_event_text_file=$TEST_log \
//does/not/exist && fail "build failure expected") || true
expect_log_once 'aborted'
expect_log_once 'reason: LOADING_FAILURE'
expect_log 'description.*BUILD file not found on package path'
expect_not_log 'expanded'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_visibility_failure() {
bazel shutdown
(bazel build --build_event_text_file=$TEST_log \
//visibility:cannotsee && fail "build failure expected") || true
expect_log 'reason: ANALYSIS_FAILURE'
expect_log '^aborted'
# The same should hold true, if the server has already analyzed the target
(bazel build --build_event_text_file=$TEST_log \
//visibility:cannotsee && fail "build failure expected") || true
expect_log 'reason: ANALYSIS_FAILURE'
expect_log '^aborted'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_visibility_indirect() {
# verify that an indirect visibility error is reported, including the
# target that violates visibility constraints.
bazel shutdown
(bazel build --build_event_text_file=$TEST_log \
//visibility:indirect && fail "build failure expected") || true
expect_log 'reason: ANALYSIS_FAILURE'
expect_log '^aborted'
expect_log '//visibility:cannotsee'
# There should be precisely one events with target_completed as event id type
(echo 'g/^id/+1p'; echo 'q') | ed "${TEST_log}" 2>&1 | tail -n +2 > event_id_types
[ `grep target_completed event_id_types | wc -l` -eq 1 ] \
|| fail "not precisely one target_completed event id"
}
function test_independent_visibility_failures() {
bazel shutdown
(bazel build -k --build_event_text_file=$TEST_log \
//visibility:indirect //visibility:indirect2 \
&& fail "build failure expected") || true
(echo 'g/^aborted/.,+2p'; echo 'q') | ed "${TEST_log}" 2>&1 | tail -n +2 \
> aborted_events
[ `grep '^aborted' aborted_events | wc -l` \
-eq `grep ANALYSIS_FAILURE aborted_events | wc -l` ] \
|| fail "events should only be aborted due to analysis failure"
}
function test_loading_failure_keep_going() {
(bazel build --build_event_text_file=$TEST_log \
-k //does/not/exist && fail "build failure expected") || true
expect_log_once 'aborted'
expect_log_once 'reason: LOADING_FAILURE'
# We don't expect an expanded message in this case, since all patterns failed.
expect_log 'description.*BUILD file not found on package path'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_loading_failure_keep_going_two_targets() {
(bazel build --build_event_text_file=$TEST_log \
-k //does/not/exist //pkg:somesourcefile && fail "build failure expected") || true
expect_log_once 'aborted'
expect_log_once 'reason: LOADING_FAILURE'
expect_log_once '^expanded'
expect_log 'description.*BUILD file not found on package path'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
# TODO(aehlig): readd, once we stop reporting the important artifacts
# for every target completion
#
# function test_artifact_dedup() {
# bazel build --build_event_text_file=$TEST_log \
# pkg:innergroup pkg:outergroup \
# || fail "bazel build failed"
# expect_log_once "name.*sourcefileA"
# expect_log_once "name.*sourcefileB"
# expect_log_once "name.*sourcefileC"
# expect_not_log 'aborted'
# }
function test_stdout_stderr_reported() {
# Verify that bazel's stdout/stderr is included in the build event stream.
# Make sure we generate enough output on stderr
bazel clean --expunge
bazel test --build_event_text_file=$TEST_log --curses=no \
pkg:slow 2>stderr.log || fail "slowtest failed"
# Take a line that is likely not the output of an action (possibly reported
# independently in the stream) and still characteristic enough to not occur
# in the stream by accident. Taking the first line mentioning the test name
# is likely some form of progress report.
sample_line=`cat stderr.log | grep 'slow' | head -n 1 | tr '[]\r' '....'`
echo "Sample regexp of stderr: ${sample_line}"
expect_log "stderr.*${sample_line}"
}
function test_unbuffered_stdout_stderr() {
# Verify that the option --bes_outerr_buffer_size ensures that messages are
# flushed out to the BEP immediately
bazel clean --expunge
bazel build --build_event_text_file="${TEST_log}" \
--bes_outerr_buffer_size=1 chain:entry10
progress_count=$(grep '^progress' "${TEST_log}" | wc -l )
# As we requested no buffereing, each action output has to be reported
# immediately, creating an individual progress event.
[ "${progress_count}" -gt 10 ] || fail "expected at least 10 progress events"
}
function test_srcfiles() {
# Even if the build target is a source file, the stream should be correctly
# and bazel shouldn't crash.
bazel build --build_event_text_file=$TEST_log \
pkg:somesourcefile || fail "build failed"
expect_log 'SUCCESS'
expect_log_once '^configuration'
expect_not_log 'aborted'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_test_fails_to_build() {
(bazel test --build_event_text_file=$TEST_log \
pkg:test_that_fails_to_build && fail "test failure expected") || true
expect_not_log '^test_summary'
expect_log 'last_message: true'
expect_log 'BUILD_FAILURE'
expect_log 'last_message: true'
expect_log 'command_line:.*This build will fail'
expect_log_once '^build_tool_logs'
}
function test_no_tests_found() {
(bazel test --build_event_text_file=$TEST_log \
pkg:not_a_test && fail "failure expected") || true
expect_not_log '^test_summary'
expect_log 'last_message: true'
expect_log 'NO_TESTS_FOUND'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_no_tests_found_build_failure() {
(bazel test -k --build_event_text_file=$TEST_log \
pkg:not_a_test pkg:fails_to_build && fail "failure expected") || true
expect_not_log '^test_summary'
expect_log 'last_message: true'
expect_log 'yet testing was requested'
expect_log 'BUILD_FAILURE'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_alias() {
bazel build --build_event_text_file=$TEST_log alias/... \
|| fail "build failed"
# If alias:it would be reported as the underlying alias/actual:it, then
# there would be no event for alias:it. So we can check the correct reporting
# by checking for aborted events.
expect_not_log 'aborted'
(echo 'g/^completed/?label?p'; echo 'q') | ed "${TEST_log}" 2>&1 | tail -n +2 > completed_labels
cat completed_labels
grep -q '//alias:it' completed_labels || fail "//alias:it not completed"
grep -q '//alias/actual:it' completed_labels \
|| fail "//alias/actual:it not completed"
[ `cat completed_labels | wc -l` -eq 2 ] \
|| fail "more than two targets completed"
rm -f completed_labels
bazel build --build_event_text_file=$TEST_log alias:it \
|| fail "build failed"
expect_log 'label: "//alias:it"'
expect_not_log 'label: "//alias/actual'
}
function test_circular_dep() {
touch test.sh
chmod u+x test.sh
cat > BUILD <<'EOF'
sh_test(
name = "circular",
srcs = ["test.sh"],
deps = ["circular"],
)
EOF
(bazel build --build_event_text_file="${TEST_log}" :circular \
&& fail "Expected failure") || :
expect_log_once 'last_message: true'
expect_log 'name: "PARSING_FAILURE"'
(bazel test --build_event_text_file="${TEST_log}" :circular \
&& fail "Expected failure") || :
expect_log_once 'last_message: true'
expect_log 'name: "PARSING_FAILURE"'
}
function test_missing_file() {
cat > BUILD <<'EOF'
filegroup(
name = "badfilegroup",
srcs = ["doesnotexist"],
)
EOF
(bazel build --build_event_text_file="${TEST_log}" :badfilegroup \
&& fail "Expected failure") || :
# There should be precisely one event with target_completed as event id type
(echo 'g/^id/+1p'; echo 'q') | ed "${TEST_log}" 2>&1 | tail -n +2 > event_id_types
[ `grep target_completed event_id_types | wc -l` -eq 1 ] \
|| fail "not precisely one target_completed event id"
# Moreover, we expect precisely one event identified by an unconfigured label
[ `grep unconfigured_label event_id_types | wc -l` -eq 1 ] \
|| fail "not precisely one unconfigured_label event id"
(bazel build --build_event_text_file="${TEST_log}" :badfilegroup :doesnotexist \
&& fail "Expected failure") || :
# There should be precisely two events with target_completed as event id type
(echo 'g/^id/+1p'; echo 'q') | ed "${TEST_log}" 2>&1 | tail -n +2 > event_id_types
[ `grep target_completed event_id_types | wc -l` -eq 2 ] \
|| fail "not precisely one target_completed event id"
# Moreover, we expect precisely one event identified by an unconfigured label
[ `grep unconfigured_label event_id_types | wc -l` -eq 1 ] \
|| fail "not precisely one unconfigured_label event id"
}
function test_tool_command_line() {
bazel build --experimental_tool_command_line="foo bar" --build_event_text_file=$TEST_log \
|| fail "build failed"
# Sanity check the arglist
expect_log_once 'args: "build"'
expect_log_once 'args: "--experimental_tool_command_line='
# Structured command line. Expect the explicit flags to appear twice,
# in the canonical and original command lines
expect_log 'command_line_label: "original"'
expect_log 'command_line_label: "canonical"'
expect_log 'command_line_label: "tool"'
# Expect the actual tool command line flag to appear twice, because of the two
# bazel command lines that are reported
expect_log_n 'combined_form: "--experimental_tool_command_line=' 2
expect_log_n 'option_name: "experimental_tool_command_line"' 2
expect_log_n 'option_value: "foo bar"' 2
# Check the contents of the tool command line
expect_log_once 'chunk: "foo bar"'
}
function test_noanalyze() {
bazel build --noanalyze --build_event_text_file="${TEST_log}" pkg:true \
|| fail "build failed"
expect_log_once '^aborted'
expect_log 'reason: NO_ANALYZE'
expect_log 'last_message: true'
expect_log_once '^build_tool_logs'
}
function test_nobuild() {
bazel build --nobuild --build_event_text_file="${TEST_log}" pkg:true \
|| fail "build failed"
expect_log_once '^aborted'
expect_log 'reason: NO_BUILD'
}
function test_server_pid() {
bazel build --test_output=all --build_event_text_file=bep.txt \
|| fail "Build failed but should have succeeded"
cat bep.txt | grep server_pid >> "$TEST_log"
expect_log_once "server_pid:.*$(bazel info server_pid)$"
rm bep.txt
}
run_suite "Integration tests for the build event stream"