Skip to content

Commit 84ff577

Browse files
comiusrickeylev
andauthored
chore: support removal of builtin providers (bazel-contrib#2274)
The builtin providers PyInfo, PyRuntimeInfo, and PyCcLinkParamsProvider are being removed, which means Bazel throws an error while compiling bzl files if there is a reference to a top-level symbol that doesn't exist anymore. For backwards compatibility, rules_python consumes/produces these providers, so the symbols are used in various places. To fix, use `native.legacy_globals` and Bazel version detection to conditionally emit the symbols into `@rules_python_internal`. If they aren't present, they are reported as None. This mimics equivalent functionality in bazel_features; bazel_features isn't used because it would require users to update their WORKSPACE to initialize some dependencies before rules_python can perform its initialization. Removal of the builtin symbols is controlled by `--incompatible_autoload_externally` (which is in Bazel 8 and has been cherry-picked into earlier version). If the flag is enabled with "@rules_python" or "-@rules_python" the providers are removed from Bazel. --------- Co-authored-by: Richard Levasseur <[email protected]>
1 parent a2773de commit 84ff577

15 files changed

+72
-37
lines changed

python/config_settings/transition.bzl

+2-2
Original file line numberDiff line numberDiff line change
@@ -101,12 +101,12 @@ def _transition_py_impl(ctx):
101101
]
102102
if PyInfo in target:
103103
providers.append(target[PyInfo])
104-
if BuiltinPyInfo in target and PyInfo != BuiltinPyInfo:
104+
if BuiltinPyInfo != None and BuiltinPyInfo in target and PyInfo != BuiltinPyInfo:
105105
providers.append(target[BuiltinPyInfo])
106106

107107
if PyRuntimeInfo in target:
108108
providers.append(target[PyRuntimeInfo])
109-
if BuiltinPyRuntimeInfo in target and PyRuntimeInfo != BuiltinPyRuntimeInfo:
109+
if BuiltinPyRuntimeInfo != None and BuiltinPyRuntimeInfo in target and PyRuntimeInfo != BuiltinPyRuntimeInfo:
110110
providers.append(target[BuiltinPyRuntimeInfo])
111111
return providers
112112

python/private/BUILD.bazel

+4-1
Original file line numberDiff line numberDiff line change
@@ -344,7 +344,10 @@ bzl_library(
344344
visibility = [
345345
"//:__subpackages__",
346346
],
347-
deps = [":bazel_tools_bzl"],
347+
deps = [
348+
":bazel_tools_bzl",
349+
"@rules_python_internal//:rules_python_config_bzl",
350+
],
348351
)
349352

350353
bzl_library(

python/private/common/attributes.bzl

+3-2
Original file line numberDiff line numberDiff line change
@@ -253,15 +253,16 @@ COMMON_ATTRS = union_attrs(
253253
allow_none = True,
254254
)
255255

256+
_MaybeBuiltinPyInfo = [[BuiltinPyInfo]] if BuiltinPyInfo != None else []
257+
256258
# Attributes common to rules accepting Python sources and deps.
257259
PY_SRCS_ATTRS = union_attrs(
258260
{
259261
"deps": attr.label_list(
260262
providers = [
261263
[PyInfo],
262264
[CcInfo],
263-
[BuiltinPyInfo],
264-
],
265+
] + _MaybeBuiltinPyInfo,
265266
# TODO(b/228692666): Google-specific; remove these allowances once
266267
# the depot is cleaned up.
267268
allow_rules = DEPS_ATTR_ALLOW_RULES,

python/private/common/common.bzl

+4-4
Original file line numberDiff line numberDiff line change
@@ -280,7 +280,7 @@ def collect_imports(ctx, semantics):
280280
dep[BuiltinPyInfo].imports
281281
for dep in ctx.attr.deps
282282
if BuiltinPyInfo in dep
283-
])
283+
] if BuiltinPyInfo != None else [])
284284

285285
def collect_runfiles(ctx, files = depset()):
286286
"""Collects the necessary files from the rule's context.
@@ -374,7 +374,7 @@ def create_py_info(ctx, *, direct_sources, direct_pyc_files, imports):
374374

375375
for target in ctx.attr.deps:
376376
# PyInfo may not be present e.g. cc_library rules.
377-
if PyInfo in target or BuiltinPyInfo in target:
377+
if PyInfo in target or (BuiltinPyInfo != None and BuiltinPyInfo in target):
378378
py_info.merge(_get_py_info(target))
379379
else:
380380
# TODO(b/228692666): Remove this once non-PyInfo targets are no
@@ -395,7 +395,7 @@ def create_py_info(ctx, *, direct_sources, direct_pyc_files, imports):
395395
for target in ctx.attr.data:
396396
# TODO(b/234730058): Remove checking for PyInfo in data once depot
397397
# cleaned up.
398-
if PyInfo in target or BuiltinPyInfo in target:
398+
if PyInfo in target or (BuiltinPyInfo != None and BuiltinPyInfo in target):
399399
info = _get_py_info(target)
400400
py_info.merge_uses_shared_libraries(info.uses_shared_libraries)
401401
else:
@@ -410,7 +410,7 @@ def create_py_info(ctx, *, direct_sources, direct_pyc_files, imports):
410410
return py_info.build(), deps_transitive_sources, py_info.build_builtin_py_info()
411411

412412
def _get_py_info(target):
413-
return target[PyInfo] if PyInfo in target else target[BuiltinPyInfo]
413+
return target[PyInfo] if PyInfo in target or BuiltinPyInfo == None else target[BuiltinPyInfo]
414414

415415
def create_instrumented_files_info(ctx):
416416
return _coverage_common.instrumented_files_info(

python/private/common/py_executable.bzl

+3-2
Original file line numberDiff line numberDiff line change
@@ -855,7 +855,7 @@ def _create_providers(
855855
# builtin py_runtime rule or defined their own. We can't directly detect
856856
# the type of the provider object, but the rules_python PyRuntimeInfo
857857
# object has an extra attribute that the builtin one doesn't.
858-
if hasattr(py_runtime_info, "interpreter_version_info"):
858+
if hasattr(py_runtime_info, "interpreter_version_info") and BuiltinPyRuntimeInfo != None:
859859
providers.append(BuiltinPyRuntimeInfo(
860860
interpreter_path = py_runtime_info.interpreter_path,
861861
interpreter = py_runtime_info.interpreter,
@@ -890,7 +890,8 @@ def _create_providers(
890890
)
891891

892892
providers.append(py_info)
893-
providers.append(builtin_py_info)
893+
if builtin_py_info:
894+
providers.append(builtin_py_info)
894895
providers.append(create_output_group_info(py_info.transitive_sources, output_groups))
895896

896897
extra_providers = semantics.get_extra_providers(

python/private/common/py_library.bzl

+4-2
Original file line numberDiff line numberDiff line change
@@ -98,14 +98,16 @@ def py_library_impl(ctx, *, semantics):
9898
dependency_transitive_python_sources = deps_transitive_sources,
9999
)
100100

101-
return [
101+
providers = [
102102
DefaultInfo(files = default_outputs, runfiles = runfiles),
103103
py_info,
104-
builtins_py_info,
105104
create_instrumented_files_info(ctx),
106105
PyCcLinkParamsProvider(cc_info = cc_info),
107106
create_output_group_info(py_info.transitive_sources, extra_groups = {}),
108107
]
108+
if builtins_py_info:
109+
providers.append(builtins_py_info)
110+
return providers
109111

110112
_DEFAULT_PY_LIBRARY_DOC = """
111113
A library of Python code that can be depended upon.

python/private/common/py_runtime_rule.bzl

+8-6
Original file line numberDiff line numberDiff line change
@@ -125,18 +125,20 @@ def _py_runtime_impl(ctx):
125125
if not IS_BAZEL_7_OR_HIGHER:
126126
builtin_py_runtime_info_kwargs.pop("bootstrap_template")
127127

128-
return [
128+
providers = [
129129
PyRuntimeInfo(**py_runtime_info_kwargs),
130-
# Return the builtin provider for better compatibility.
131-
# 1. There is a legacy code path in py_binary that
132-
# checks for the provider when toolchains aren't used
133-
# 2. It makes it easier to transition from builtins to rules_python
134-
BuiltinPyRuntimeInfo(**builtin_py_runtime_info_kwargs),
135130
DefaultInfo(
136131
files = runtime_files,
137132
runfiles = runfiles,
138133
),
139134
]
135+
if BuiltinPyRuntimeInfo != None and BuiltinPyRuntimeInfo != PyRuntimeInfo:
136+
# Return the builtin provider for better compatibility.
137+
# 1. There is a legacy code path in py_binary that
138+
# checks for the provider when toolchains aren't used
139+
# 2. It makes it easier to transition from builtins to rules_python
140+
providers.append(BuiltinPyRuntimeInfo(**builtin_py_runtime_info_kwargs))
141+
return providers
140142

141143
# Bind to the name "py_runtime" to preserve the kind/rule_class it shows up
142144
# as elsewhere.

python/private/internal_config_repo.bzl

+15
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,9 @@ _ENABLE_PYSTAR_DEFAULT = "1"
2424
_CONFIG_TEMPLATE = """\
2525
config = struct(
2626
enable_pystar = {enable_pystar},
27+
BuiltinPyInfo = getattr(getattr(native, "legacy_globals", None), "PyInfo", {builtin_py_info_symbol}),
28+
BuiltinPyRuntimeInfo = getattr(getattr(native, "legacy_globals", None), "PyRuntimeInfo", {builtin_py_runtime_info_symbol}),
29+
BuiltinPyCcLinkParamsProvider = getattr(getattr(native, "legacy_globals", None), "PyCcLinkParamsProvider", {builtin_py_cc_link_params_provider}),
2730
)
2831
"""
2932

@@ -65,8 +68,20 @@ def _internal_config_repo_impl(rctx):
6568
else:
6669
enable_pystar = False
6770

71+
if native.bazel_version.startswith("8."):
72+
builtin_py_info_symbol = "None"
73+
builtin_py_runtime_info_symbol = "None"
74+
builtin_py_cc_link_params_provider = "None"
75+
else:
76+
builtin_py_info_symbol = "PyInfo"
77+
builtin_py_runtime_info_symbol = "PyRuntimeInfo"
78+
builtin_py_cc_link_params_provider = "PyCcLinkParamsProvider"
79+
6880
rctx.file("rules_python_config.bzl", _CONFIG_TEMPLATE.format(
6981
enable_pystar = enable_pystar,
82+
builtin_py_info_symbol = builtin_py_info_symbol,
83+
builtin_py_runtime_info_symbol = builtin_py_runtime_info_symbol,
84+
builtin_py_cc_link_params_provider = builtin_py_cc_link_params_provider,
7085
))
7186

7287
if enable_pystar:

python/private/py_info.bzl

+5-2
Original file line numberDiff line numberDiff line change
@@ -118,7 +118,7 @@ This field is currently unused in Bazel and may go away in the future.
118118
)
119119

120120
# The "effective" PyInfo is what the canonical //python:py_info.bzl%PyInfo symbol refers to
121-
_EffectivePyInfo = PyInfo if config.enable_pystar else BuiltinPyInfo
121+
_EffectivePyInfo = PyInfo if (config.enable_pystar or BuiltinPyInfo == None) else BuiltinPyInfo
122122

123123
def PyInfoBuilder():
124124
# buildifier: disable=uninitialized
@@ -206,7 +206,7 @@ def _PyInfoBuilder_merge_all(self, transitive, *, direct = []):
206206
def _PyInfoBuilder_merge_target(self, target):
207207
if PyInfo in target:
208208
self.merge(target[PyInfo])
209-
elif BuiltinPyInfo in target:
209+
elif BuiltinPyInfo != None and BuiltinPyInfo in target:
210210
self.merge(target[BuiltinPyInfo])
211211
return self
212212

@@ -234,6 +234,9 @@ def _PyInfoBuilder_build(self):
234234
)
235235

236236
def _PyInfoBuilder_build_builtin_py_info(self):
237+
if BuiltinPyInfo == None:
238+
return None
239+
237240
return BuiltinPyInfo(
238241
has_py2_only_sources = self._has_py2_only_sources[0],
239242
has_py3_only_sources = self._has_py3_only_sources[0],

python/private/py_runtime_pair_rule.bzl

+5-3
Original file line numberDiff line numberDiff line change
@@ -56,7 +56,7 @@ def _get_py_runtime_info(target):
5656
# py_binary (implemented in Java) performs a type check on the provider
5757
# value to verify it is an instance of the Java-implemented PyRuntimeInfo
5858
# class.
59-
if IS_BAZEL_7_OR_HIGHER and PyRuntimeInfo in target:
59+
if (IS_BAZEL_7_OR_HIGHER and PyRuntimeInfo in target) or BuiltinPyRuntimeInfo == None:
6060
return target[PyRuntimeInfo]
6161
else:
6262
return target[BuiltinPyRuntimeInfo]
@@ -70,21 +70,23 @@ def _is_py2_disabled(ctx):
7070
return False
7171
return ctx.fragments.py.disable_py2
7272

73+
_MaybeBuiltinPyRuntimeInfo = [[BuiltinPyRuntimeInfo]] if BuiltinPyRuntimeInfo != None else []
74+
7375
py_runtime_pair = rule(
7476
implementation = _py_runtime_pair_impl,
7577
attrs = {
7678
# The two runtimes are used by the py_binary at runtime, and so need to
7779
# be built for the target platform.
7880
"py2_runtime": attr.label(
79-
providers = [[PyRuntimeInfo], [BuiltinPyRuntimeInfo]],
81+
providers = [[PyRuntimeInfo]] + _MaybeBuiltinPyRuntimeInfo,
8082
cfg = "target",
8183
doc = """\
8284
The runtime to use for Python 2 targets. Must have `python_version` set to
8385
`PY2`.
8486
""",
8587
),
8688
"py3_runtime": attr.label(
87-
providers = [[PyRuntimeInfo], [BuiltinPyRuntimeInfo]],
89+
providers = [[PyRuntimeInfo]] + _MaybeBuiltinPyRuntimeInfo,
8890
cfg = "target",
8991
doc = """\
9092
The runtime to use for Python 3 targets. Must have `python_version` set to

python/private/reexports.bzl

+6-5
Original file line numberDiff line numberDiff line change
@@ -30,11 +30,12 @@ inaccessible. So instead we access the builtin here and export it under a
3030
different name. Then we can load it from elsewhere.
3131
"""
3232

33-
# Don't use underscore prefix, since that would make the symbol local to this
34-
# file only. Use a non-conventional name to emphasize that this is not a public
35-
# symbol.
33+
load("@rules_python_internal//:rules_python_config.bzl", "config")
34+
35+
# NOTE: May be None (Bazel 8 autoloading rules_python)
3636
# buildifier: disable=name-conventions
37-
BuiltinPyInfo = PyInfo
37+
BuiltinPyInfo = config.BuiltinPyInfo
3838

39+
# NOTE: May be None (Bazel 8 autoloading rules_python)
3940
# buildifier: disable=name-conventions
40-
BuiltinPyRuntimeInfo = PyRuntimeInfo
41+
BuiltinPyRuntimeInfo = config.BuiltinPyRuntimeInfo

python/py_cc_link_params_info.bzl

+5-1
Original file line numberDiff line numberDiff line change
@@ -3,4 +3,8 @@
33
load("@rules_python_internal//:rules_python_config.bzl", "config")
44
load("//python/private/common:providers.bzl", _starlark_PyCcLinkParamsProvider = "PyCcLinkParamsProvider")
55

6-
PyCcLinkParamsInfo = _starlark_PyCcLinkParamsProvider if config.enable_pystar else PyCcLinkParamsProvider
6+
PyCcLinkParamsInfo = (
7+
_starlark_PyCcLinkParamsProvider if (
8+
config.enable_pystar or config.BuiltinPyCcLinkParamsProvider == None
9+
) else config.BuiltinPyCcLinkParamsProvider
10+
)

python/py_info.bzl

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,4 +18,4 @@ load("@rules_python_internal//:rules_python_config.bzl", "config")
1818
load("//python/private:py_info.bzl", _starlark_PyInfo = "PyInfo")
1919
load("//python/private:reexports.bzl", "BuiltinPyInfo")
2020

21-
PyInfo = _starlark_PyInfo if config.enable_pystar else BuiltinPyInfo
21+
PyInfo = _starlark_PyInfo if config.enable_pystar or BuiltinPyInfo == None else BuiltinPyInfo

tests/base_rules/py_executable_base_tests.bzl

+5-5
Original file line numberDiff line numberDiff line change
@@ -19,14 +19,13 @@ load("@rules_testing//lib:analysis_test.bzl", "analysis_test")
1919
load("@rules_testing//lib:truth.bzl", "matching")
2020
load("@rules_testing//lib:util.bzl", rt_util = "util")
2121
load("//python:py_executable_info.bzl", "PyExecutableInfo")
22+
load("//python/private:reexports.bzl", "BuiltinPyRuntimeInfo") # buildifier: disable=bzl-visibility
2223
load("//python/private:util.bzl", "IS_BAZEL_7_OR_HIGHER") # buildifier: disable=bzl-visibility
2324
load("//tests/base_rules:base_tests.bzl", "create_base_tests")
2425
load("//tests/base_rules:util.bzl", "WINDOWS_ATTR", pt_util = "util")
2526
load("//tests/support:py_executable_info_subject.bzl", "PyExecutableInfoSubject")
2627
load("//tests/support:support.bzl", "CC_TOOLCHAIN", "CROSSTOOL_TOP", "LINUX_X86_64", "WINDOWS_X86_64")
2728

28-
_BuiltinPyRuntimeInfo = PyRuntimeInfo
29-
3029
_tests = []
3130

3231
def _test_basic_windows(name, config):
@@ -359,9 +358,10 @@ def _test_py_runtime_info_provided_impl(env, target):
359358
# Make sure that the rules_python loaded symbol is provided.
360359
env.expect.that_target(target).has_provider(RulesPythonPyRuntimeInfo)
361360

362-
# For compatibility during the transition, the builtin PyRuntimeInfo should
363-
# also be provided.
364-
env.expect.that_target(target).has_provider(_BuiltinPyRuntimeInfo)
361+
if BuiltinPyRuntimeInfo != None:
362+
# For compatibility during the transition, the builtin PyRuntimeInfo should
363+
# also be provided.
364+
env.expect.that_target(target).has_provider(BuiltinPyRuntimeInfo)
365365

366366
_tests.append(_test_py_runtime_info_provided)
367367

tests/base_rules/py_info/py_info_tests.bzl

+2-1
Original file line numberDiff line numberDiff line change
@@ -191,7 +191,8 @@ def _test_py_info_builder_impl(env, targets):
191191
])
192192

193193
check(builder.build())
194-
check(builder.build_builtin_py_info())
194+
if BuiltinPyInfo != None:
195+
check(builder.build_builtin_py_info())
195196

196197
builder.set_has_py2_only_sources(False)
197198
builder.set_has_py3_only_sources(False)

0 commit comments

Comments
 (0)