# 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)