blob: c68450235ada17ea1f55c3a890ce9bae5276b63f [file] [log] [blame]
#!/bin/bash
#
# Copyright 2017 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.
#
# --- begin runfiles.bash initialization ---
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="*"
declare -r EXE_EXT=".exe"
else
declare -r EXE_EXT=""
fi
#### TESTS #############################################################
# Tests in this file cannot invoke a real Python 3 runtime. This is because this
# file is shared by both Bazel's public test suite and Google's internal tests,
# and the internal tests do not have a Python 3 environment.
#
# - If you only need a real Python 2 environment and do not use Python 3 at
# all, you can place your test in this file.
#
# - If you need to check Bazel's behavior concerning the *selection* of a
# Python 2 or 3 runtime, but do not actually need the runtime itself, then
# you may put your test in this file and call `use_fake_python_runtimes`
# before your test logic.
#
# - Otherwise, put your test in //src/test/shell/bazel. That suite can invoke
# actual Python 2 and 3 interpreters.
function test_python_binary_empty_files_in_runfiles_are_regular_files() {
mkdir -p test/mypackage
cat > test/BUILD <<'EOF'
py_test(
name = "a",
srcs = [
"a.py",
"mypackage/b.py",
],
main = "a.py"
)
EOF
cat >test/a.py <<'EOF'
from __future__ import print_function
import os.path
import sys
print("This is my name: %s" % __file__)
print("This is my working directory: %s" % os.getcwd())
os.chdir(os.path.dirname(__file__))
print("This is my new working directory: %s" % os.getcwd())
file_to_check = "mypackage/__init__.py"
if not os.path.exists(file_to_check):
print("mypackage/__init__.py does not exist")
sys.exit(1)
if os.path.islink(file_to_check):
print("mypackage/__init__.py is a symlink, expected a regular file")
sys.exit(1)
if not os.path.isfile(file_to_check):
print("mypackage/__init__.py is not a regular file")
sys.exit(1)
print("OK")
EOF
touch test/mypackage/b.py
bazel test --test_output=streamed //test:a &> $TEST_log || fail "test failed"
}
function test_building_transitive_py_binary_runfiles_trees() {
touch main.py script.sh
chmod u+x script.sh
cat > BUILD <<'EOF'
py_binary(
name = 'py-tool',
srcs = ['main.py'],
main = 'main.py',
)
sh_binary(
name = 'sh-tool',
srcs = ['script.sh'],
data = [':py-tool'],
)
EOF
bazel build --experimental_build_transitive_python_runfiles :sh-tool
[ -d "bazel-bin/py-tool${EXE_EXT}.runfiles" ] || fail "py_binary runfiles tree not built"
bazel clean
bazel build --noexperimental_build_transitive_python_runfiles :sh-tool
[ ! -e "bazel-bin/py-tool${EXE_EXT}.runfiles" ] || fail "py_binary runfiles tree built"
}
# Tests that Python 2 or Python 3 is actually invoked, with and without flag
# overrides.
function test_python_version() {
use_fake_python_runtimes
mkdir -p test
touch test/main2.py test/main3.py
cat > test/BUILD << EOF
py_binary(name = "main2",
default_python_version = "PY2",
srcs = ['main2.py'],
)
py_binary(name = "main3",
default_python_version = "PY3",
srcs = ["main3.py"],
)
EOF
# No flag, use the default from the rule.
bazel run //test:main2 \
&> $TEST_log || fail "bazel run failed"
expect_log "I am Python 2"
bazel run //test:main3 \
&> $TEST_log || fail "bazel run failed"
expect_log "I am Python 3"
# Force to Python 2.
bazel run //test:main2 --force_python=PY2 \
&> $TEST_log || fail "bazel run failed"
expect_log "I am Python 2"
bazel run //test:main3 --force_python=PY2 \
&> $TEST_log || fail "bazel run failed"
expect_log "I am Python 2"
# Force to Python 3.
bazel run //test:main2 --force_python=PY3 \
&> $TEST_log || fail "bazel run failed"
expect_log "I am Python 3"
bazel run //test:main3 --force_python=PY3 \
&> $TEST_log || fail "bazel run failed"
expect_log "I am Python 3"
}
# Helper function for test_select_on_python_version. Dependent on its setup.
function do_test_select_on_python_version() {
flags="$1"
result="$2"
bazel run //test:main $flags \
&> $TEST_log || fail "bazel run failed"
expect_log "I am $result" "Expected version $result for flags \"$flags\""
}
# Tests that selecting on the Python version gives the expected results for all
# combinations of the relevant flags.
function test_select_on_python_version() {
mkdir -p test
cat > test/BUILD << EOF
sh_binary(name = "main",
srcs = select({
"$TOOLS_REPOSITORY//tools/python:PY2": ["py2.sh"],
"$TOOLS_REPOSITORY//tools/python:PY3": ["py3.sh"],
}),
)
EOF
cat > test/py2.sh << EOF
#/bin/sh
echo "I am PY2"
EOF
sed s/PY2/PY3/ test/py2.sh > test/py3.sh
chmod u+x test/py2.sh test/py3.sh
DEFAULT_VERSION=PY2
EXPFLAG="--experimental_better_python_version_mixing=true"
NO_EXPFLAG="--experimental_better_python_version_mixing=false"
# --force_python and --python_version have three possible values (including
# not being set at all). The experimental flag has two possible values, but
# when it's disabled --python_version may not be set. So that's 12
# combinations to enumerate here.
do_test_select_on_python_version "$NO_EXPFLAG" "$DEFAULT_VERSION"
do_test_select_on_python_version "$NO_EXPFLAG --force_python=PY2" "PY2"
do_test_select_on_python_version "$NO_EXPFLAG --force_python=PY3" "PY3"
do_test_select_on_python_version "$EXPFLAG" "$DEFAULT_VERSION"
do_test_select_on_python_version "$EXPFLAG --force_python=PY2" "PY2"
do_test_select_on_python_version "$EXPFLAG --force_python=PY3" "PY3"
do_test_select_on_python_version "$EXPFLAG --python_version=PY2" "PY2"
do_test_select_on_python_version "$EXPFLAG --force_python=PY2 --python_version=PY2" "PY2"
do_test_select_on_python_version "$EXPFLAG --force_python=PY3 --python_version=PY2" "PY2"
do_test_select_on_python_version "$EXPFLAG --python_version=PY3" "PY3"
do_test_select_on_python_version "$EXPFLAG --force_python=PY2 --python_version=PY3" "PY3"
do_test_select_on_python_version "$EXPFLAG --force_python=PY3 --python_version=PY3" "PY3"
}
# Tests that the user is prevented from selecting on the native Python version
# flags, which should only be used internally.
function test_cannot_select_on_python_version_native_flags() {
mkdir -p test
cat > test/BUILD << EOF
config_setting(
name = "badconfig_pv",
values = {"python_version": "PY2"},
)
config_setting(
name = "badconfig_fp",
values = {"force_python": "PY2"},
)
config_setting(
name = "badconfig_hfp",
values = {"host_force_python": "PY2"},
)
sh_binary(name = "bad_pv",
srcs = select({
":badconfig_pv": ["bad.sh"],
"//conditions:default": ["bad.sh"],
}),
)
sh_binary(name = "bad_fp",
srcs = select({
":badconfig_fp": ["bad.sh"],
"//conditions:default": ["bad.sh"],
}),
)
sh_binary(name = "bad_hfp",
srcs = select({
":badconfig_hfp": ["bad.sh"],
"//conditions:default": ["bad.sh"],
}),
)
EOF
touch bad.sh
chmod u+x bad.sh
EXPFLAG="--experimental_better_python_version_mixing=true"
NO_EXPFLAG="--experimental_better_python_version_mixing=false"
# --python_version is always unselectable.
bazel build //test:bad_pv $NO_EXPFLAG \
&> $TEST_log && fail "expected analysis to fail"
expect_log "'python_version' cannot be used in a config_setting"
bazel build //test:bad_pv $EXPFLAG \
&> $TEST_log && fail "expected analysis to fail"
expect_log "'python_version' cannot be used in a config_setting"
# --force_python and --host_force_python are only unselectable with the
# experimental flag enabled.
bazel build //test:bad_fp $EXPFLAG \
&> $TEST_log && fail "expected analysis to fail"
expect_log "'force_python' cannot be used in a config_setting"
bazel build //test:bad_hfp $EXPFLAG \
&> $TEST_log && fail "expected analysis to fail"
expect_log "'host_force_python' cannot be used in a config_setting"
}
run_suite "Tests for the Python rules"