blob: ae298edb4f6077ee94369e07b8704ccc0371b0ab [file] [log] [blame]
# Copyright 2023 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 common to py_test, py_binary, and py_library rules."""
load("@rules_testing//lib:analysis_test.bzl", "analysis_test")
load("@rules_testing//lib:truth.bzl", "matching")
load("@rules_testing//lib:util.bzl", "PREVENT_IMPLICIT_BUILDING_TAGS", rt_util = "util")
load("//python:defs.bzl", "PyInfo")
load("//python/private:reexports.bzl", "BuiltinPyInfo") # buildifier: disable=bzl-visibility
load("//tests/base_rules:util.bzl", pt_util = "util")
load("//tests/support:py_info_subject.bzl", "py_info_subject")
_tests = []
_PRODUCES_PY_INFO_ATTRS = {
"imports": attr.string_list(),
"srcs": attr.label_list(allow_files = True),
}
def _create_py_info(ctx, provider_type):
return [provider_type(
transitive_sources = depset(ctx.files.srcs),
imports = depset(ctx.attr.imports),
)]
def _produces_builtin_py_info_impl(ctx):
return _create_py_info(ctx, BuiltinPyInfo)
_produces_builtin_py_info = rule(
implementation = _produces_builtin_py_info_impl,
attrs = _PRODUCES_PY_INFO_ATTRS,
)
def _produces_py_info_impl(ctx):
return _create_py_info(ctx, BuiltinPyInfo)
_produces_py_info = rule(
implementation = _produces_py_info_impl,
attrs = _PRODUCES_PY_INFO_ATTRS,
)
def _not_produces_py_info_impl(ctx):
_ = ctx # @unused
return [DefaultInfo()]
_not_produces_py_info = rule(
implementation = _not_produces_py_info_impl,
)
def _py_info_propagation_setup(name, config, produce_py_info_rule, test_impl):
rt_util.helper_target(
config.base_test_rule,
name = name + "_subject",
deps = [name + "_produces_builtin_py_info"],
)
rt_util.helper_target(
produce_py_info_rule,
name = name + "_produces_builtin_py_info",
srcs = [rt_util.empty_file(name + "_produce.py")],
imports = ["custom-import"],
)
analysis_test(
name = name,
target = name + "_subject",
impl = test_impl,
)
def _py_info_propagation_test_impl(env, target, provider_type):
info = env.expect.that_target(target).provider(
provider_type,
factory = py_info_subject,
)
info.transitive_sources().contains("{package}/{test_name}_produce.py")
info.imports().contains("custom-import")
def _test_py_info_propagation_builtin(name, config):
_py_info_propagation_setup(
name,
config,
_produces_builtin_py_info,
_test_py_info_propagation_builtin_impl,
)
def _test_py_info_propagation_builtin_impl(env, target):
_py_info_propagation_test_impl(env, target, BuiltinPyInfo)
_tests.append(_test_py_info_propagation_builtin)
def _test_py_info_propagation(name, config):
_py_info_propagation_setup(
name,
config,
_produces_py_info,
_test_py_info_propagation_impl,
)
def _test_py_info_propagation_impl(env, target):
_py_info_propagation_test_impl(env, target, PyInfo)
_tests.append(_test_py_info_propagation)
def _test_requires_provider(name, config):
rt_util.helper_target(
config.base_test_rule,
name = name + "_subject",
deps = [name + "_nopyinfo"],
)
rt_util.helper_target(
_not_produces_py_info,
name = name + "_nopyinfo",
)
analysis_test(
name = name,
target = name + "_subject",
impl = _test_requires_provider_impl,
expect_failure = True,
)
def _test_requires_provider_impl(env, target):
env.expect.that_target(target).failures().contains_predicate(
matching.str_matches("mandatory*PyInfo"),
)
_tests.append(_test_requires_provider)
def _test_data_sets_uses_shared_library(name, config):
rt_util.helper_target(
config.base_test_rule,
name = name + "_subject",
data = [rt_util.empty_file(name + "_dso.so")],
)
analysis_test(
name = name,
target = name + "_subject",
impl = _test_data_sets_uses_shared_library_impl,
)
def _test_data_sets_uses_shared_library_impl(env, target):
env.expect.that_target(target).provider(
PyInfo,
factory = py_info_subject,
).uses_shared_libraries().equals(True)
_tests.append(_test_data_sets_uses_shared_library)
def _test_tags_can_be_tuple(name, config):
# We don't use a helper because we want to ensure that value passed is
# a tuple.
config.base_test_rule(
name = name + "_subject",
tags = ("one", "two") + tuple(PREVENT_IMPLICIT_BUILDING_TAGS),
)
analysis_test(
name = name,
target = name + "_subject",
impl = _test_tags_can_be_tuple_impl,
)
def _test_tags_can_be_tuple_impl(env, target):
env.expect.that_target(target).tags().contains_at_least([
"one",
"two",
])
_tests.append(_test_tags_can_be_tuple)
def create_base_tests(config):
return pt_util.create_tests(_tests, config = config)