Skip to content

Commit 09145b9

Browse files
authored
feat: uv lock rule instead of genrule (#2657)
This change re-implements the `uv pip compile` as a set of rules instead of using a `genrule`. This makes the setup more RBE friendly and it also fixes some of existing issues in the exec tools toolchain. The `lock` macro in the `//python/uv:lock.bzl` now creates three public targets: `<name>`, `<name>.update` and `<name>.run`. The first will provide you with the locked `requirements.txt` file that is used in the `<name>.update` executable target when updating the in-source copy of the file. The `<name>.run` provides an executable target that hardcodes all of the `uv` args from the `<name>` rule in a shell script and allows user to debug the execution and add extra arguments at the command line. The `test` target is no longer included, but users can define it themselves with the help of `native_test`. Things that I could not test and would benefit from the community help: * Windows support - the repository has a rudimentary script, but I am almost sure that it is likely not working, so PRs there are welcome. * The integration tests are not running on RBE because of the current RBE cluster setup. If you see issues in your RBE setup, PRs are welcome. * `keyring` integration to pull packages from private index servers is untested as of now, but I see no reason why it should not work. Work towards #1325 Work towards #1975 Related #2663
1 parent bfa59b9 commit 09145b9

27 files changed

+1085
-106
lines changed

CHANGELOG.md

+12
Original file line numberDiff line numberDiff line change
@@ -53,6 +53,10 @@ Unreleased changes template.
5353

5454
{#v0-0-0-changed}
5555
### Changed
56+
* (toolchain) The `exec` configuration toolchain now has the forwarded
57+
`exec_interpreter` now also forwards the `ToolchainInfo` provider. This is
58+
for increased compatibility with the `RBE` setups where access to the `exec`
59+
configuration interpreter is needed.
5660
* (toolchains) Use the latest astrahl-sh toolchain release [20250317] for Python versions:
5761
* 3.9.21
5862
* 3.10.16
@@ -75,6 +79,14 @@ Unreleased changes template.
7579

7680
{#v0-0-0-added}
7781
### Added
82+
* (uv) A {obj}`lock` rule that is the replacement for the
83+
{obj}`compile_pip_requirements`. This may still have rough corners
84+
so please report issues with it in the
85+
[#1975](https://github.com/bazel-contrib/rules_python/issues/1975).
86+
Main highlights - the locking can be done within a build action or outside
87+
it, there is no more automatic `test` target (but it can be added on the user
88+
side by using `native_test`). For customizing the `uv` version that is used,
89+
please check the {obj}`uv.configure` tag class.
7890
* Add support for riscv64 linux platform.
7991
* (toolchains) Add python 3.13.2 and 3.12.9 toolchains
8092

docs/BUILD.bazel

+6-2
Original file line numberDiff line numberDiff line change
@@ -176,8 +176,12 @@ lock(
176176
name = "requirements",
177177
srcs = ["pyproject.toml"],
178178
out = "requirements.txt",
179-
upgrade = True,
180-
visibility = ["//private:__pkg__"],
179+
args = [
180+
"--emit-index-url",
181+
"--universal",
182+
"--upgrade",
183+
],
184+
visibility = ["//:__subpackages__"],
181185
)
182186

183187
# Temporary compatibility aliases for some other projects depending on the old

examples/BUILD.bazel

+5
Original file line numberDiff line numberDiff line change
@@ -21,5 +21,10 @@ lock(
2121
name = "bzlmod_requirements_3_9",
2222
srcs = ["bzlmod/requirements.in"],
2323
out = "bzlmod/requirements_lock_3_9.txt",
24+
args = [
25+
"--emit-index-url",
26+
"--universal",
27+
"--python-version=3.9",
28+
],
2429
python_version = "3.9.19",
2530
)

examples/bzlmod/requirements_lock_3_9.txt

+4-4
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ imagesize==1.4.1 \
4646
--hash=sha256:0d8d18d08f840c19d0ee7ca1fd82490fdc3729b7ac93f49870406ddde8ef8d8b \
4747
--hash=sha256:69150444affb9cb0d5cc5a92b3676f0b2fb7cd9ae39e947a5e11a36b4497cd4a
4848
# via sphinx
49-
importlib-metadata==8.4.0 ; python_version < '3.10' \
49+
importlib-metadata==8.4.0 ; python_full_version < '3.10' \
5050
--hash=sha256:66f342cc6ac9818fc6ff340576acd24d65ba0b3efabb2b4ac08b598965a4a2f1 \
5151
--hash=sha256:9a547d3bc3608b025f93d403fdd1aae741c24fbb8314df4b155675742ce303c5
5252
# via sphinx
@@ -316,15 +316,15 @@ tabulate==0.9.0 \
316316
--hash=sha256:0095b12bf5966de529c0feb1fa08671671b3368eec77d7ef7ab114be2c068b3c \
317317
--hash=sha256:024ca478df22e9340661486f85298cff5f6dcdba14f3813e8830015b9ed1948f
318318
# via -r examples/bzlmod/requirements.in
319-
tomli==2.0.1 ; python_version < '3.11' \
319+
tomli==2.0.1 ; python_full_version < '3.11' \
320320
--hash=sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc \
321321
--hash=sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f
322322
# via pylint
323323
tomlkit==0.11.6 \
324324
--hash=sha256:07de26b0d8cfc18f871aec595fda24d95b08fef89d147caa861939f37230bf4b \
325325
--hash=sha256:71b952e5721688937fb02cf9d354dbcf0785066149d2855e44531ebdd2b65d73
326326
# via pylint
327-
typing-extensions==4.12.2 ; python_version < '3.10' \
327+
typing-extensions==4.12.2 ; python_full_version < '3.10' \
328328
--hash=sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d \
329329
--hash=sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8
330330
# via
@@ -480,7 +480,7 @@ yamllint==1.28.0 \
480480
--hash=sha256:89bb5b5ac33b1ade059743cf227de73daa34d5e5a474b06a5e17fc16583b0cf2 \
481481
--hash=sha256:9e3d8ddd16d0583214c5fdffe806c9344086721f107435f68bad990e5a88826b
482482
# via -r examples/bzlmod/requirements.in
483-
zipp==3.20.0 ; python_version < '3.10' \
483+
zipp==3.20.0 ; python_full_version < '3.10' \
484484
--hash=sha256:0145e43d89664cfe1a2e533adc75adafed82fe2da404b4bbb6b026c0157bdb31 \
485485
--hash=sha256:58da6168be89f0be59beb194da1250516fdaa062ccebd30127ac65d30045e10d
486486
# via importlib-metadata

private/BUILD.bazel

+2
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ multirun(
1515
] + [
1616
"//docs:requirements.update",
1717
],
18+
tags = ["manual"],
1819
)
1920

2021
# NOTE: The requirements for the pip dependencies may sometimes break the build
@@ -24,4 +25,5 @@ multirun(
2425
alias(
2526
name = "whl_library_requirements.update",
2627
actual = "//tools/private/update_deps:update_pip_deps",
28+
tags = ["manual"],
2729
)

python/private/BUILD.bazel

+1
Original file line numberDiff line numberDiff line change
@@ -361,6 +361,7 @@ bzl_library(
361361
name = "py_exec_tools_toolchain_bzl",
362362
srcs = ["py_exec_tools_toolchain.bzl"],
363363
deps = [
364+
":common_bzl",
364365
":py_exec_tools_info_bzl",
365366
":sentinel_bzl",
366367
":toolchain_types_bzl",

python/private/py_exec_tools_info.bzl

+15-4
Original file line numberDiff line numberDiff line change
@@ -24,15 +24,26 @@ When running it in an action, use `DefaultInfo.files_to_run` to ensure all its
2424
files are appropriately available. An exec interpreter may not be available,
2525
e.g. if all the exec tools are prebuilt binaries.
2626
27-
NOTE: this interpreter is really only for use when a build tool cannot use
27+
:::{note}
28+
this interpreter is really only for use when a build tool cannot use
2829
the Python toolchain itself. When possible, prefeer to define a `py_binary`
2930
instead and use it via a `cfg=exec` attribute; this makes it much easier
3031
to setup the runtime environment for the binary. See also:
3132
`py_interpreter_program` rule.
33+
:::
3234
33-
NOTE: What interpreter is used depends on the toolchain constraints. Ensure
34-
the proper target constraints are being applied when obtaining this from
35-
the toolchain.
35+
:::{note}
36+
What interpreter is used depends on the toolchain constraints. Ensure the
37+
proper target constraints are being applied when obtaining this from the
38+
toolchain.
39+
:::
40+
41+
:::{warning}
42+
This does not work correctly in case of RBE, please use exec_runtime instead.
43+
44+
Once https://github.com/bazelbuild/bazel/issues/23620 is resolved this warning
45+
may be removed.
46+
:::
3647
""",
3748
"precompiler": """
3849
:type: Target | None

python/private/py_exec_tools_toolchain.bzl

+18-6
Original file line numberDiff line numberDiff line change
@@ -29,13 +29,15 @@ def _py_exec_tools_toolchain_impl(ctx):
2929
if SentinelInfo in ctx.attr.exec_interpreter:
3030
exec_interpreter = None
3131

32-
return [platform_common.ToolchainInfo(
33-
exec_tools = PyExecToolsInfo(
34-
exec_interpreter = exec_interpreter,
35-
precompiler = ctx.attr.precompiler,
32+
return [
33+
platform_common.ToolchainInfo(
34+
exec_tools = PyExecToolsInfo(
35+
exec_interpreter = exec_interpreter,
36+
precompiler = ctx.attr.precompiler,
37+
),
38+
**extra_kwargs
3639
),
37-
**extra_kwargs
38-
)]
40+
]
3941

4042
py_exec_tools_toolchain = rule(
4143
implementation = _py_exec_tools_toolchain_impl,
@@ -51,6 +53,11 @@ This provides `ToolchainInfo` with the following attributes:
5153
attrs = {
5254
"exec_interpreter": attr.label(
5355
default = "//python/private:current_interpreter_executable",
56+
providers = [
57+
DefaultInfo,
58+
# Add the toolchain provider so that we can forward provider fields.
59+
platform_common.ToolchainInfo,
60+
],
5461
cfg = "exec",
5562
doc = """
5663
An interpreter that is directly usable in the exec configuration
@@ -69,6 +76,11 @@ handle all the necessary transitions and runtime setup to invoke a program.
6976
:::
7077
7178
See {obj}`PyExecToolsInfo.exec_interpreter` for further docs.
79+
80+
:::{versionchanged} VERSION_NEXT_FEATURE
81+
From now on the provided target also needs to provide `platform_common.ToolchainInfo`
82+
so that the toolchain `py_runtime` field can be correctly forwarded.
83+
:::
7284
""",
7385
),
7486
"precompiler": attr.label(

python/private/sentinel.bzl

+5-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,10 @@ SentinelInfo = provider(
2525

2626
def _sentinel_impl(ctx):
2727
_ = ctx # @unused
28-
return [SentinelInfo()]
28+
return [
29+
SentinelInfo(),
30+
# Also output ToolchainInfo to allow it to be used for noop toolchains
31+
platform_common.ToolchainInfo(),
32+
]
2933

3034
sentinel = rule(implementation = _sentinel_impl)

python/uv/lock.bzl

+27-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,33 @@
1414

1515
"""The `uv` locking rule.
1616
17-
EXPERIMENTAL: This is experimental and may be removed without notice
17+
Differences with the legacy {obj}`compile_pip_requirements` rule:
18+
- This is implemented as a rule that performs locking in a build action.
19+
- Additionally one can use the runnable target.
20+
- Uses `uv`.
21+
- This does not error out if the output file does not exist yet.
22+
- Supports transitions out of the box.
23+
24+
Note, this does not provide a `test` target, if you would like to add a test
25+
target that always does the locking automatically to ensure that the
26+
`requirements.txt` file is up-to-date, add something similar to:
27+
28+
```starlark
29+
load("@bazel_skylib//rules:native_binary.bzl", "native_test")
30+
load("@rules_python//python/uv:lock.bzl", "lock")
31+
32+
lock(
33+
name = "requirements",
34+
srcs = ["pyproject.toml"],
35+
)
36+
37+
native_test(
38+
name = "requirements_test",
39+
src = "requirements.update",
40+
)
41+
```
42+
43+
EXPERIMENTAL: This is experimental and may be changed without notice.
1844
"""
1945

2046
load("//python/uv/private:lock.bzl", _lock = "lock")

python/uv/private/BUILD.bazel

+24-1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,15 @@
1313
# limitations under the License.
1414

1515
load("@bazel_skylib//:bzl_library.bzl", "bzl_library")
16+
load("//python/private:bzlmod_enabled.bzl", "BZLMOD_ENABLED") # buildifier: disable=bzl-visibility
17+
18+
exports_files(
19+
srcs = [
20+
"lock_copier.py",
21+
],
22+
# only because this is used from a macro to template
23+
visibility = ["//visibility:public"],
24+
)
1625

1726
filegroup(
1827
name = "distribution",
@@ -31,9 +40,13 @@ bzl_library(
3140
srcs = ["lock.bzl"],
3241
visibility = ["//python/uv:__subpackages__"],
3342
deps = [
43+
":toolchain_types_bzl",
3444
"//python:py_binary_bzl",
3545
"//python/private:bzlmod_enabled_bzl",
36-
"@bazel_skylib//rules:write_file",
46+
"//python/private:full_version_bzl",
47+
"//python/private:toolchain_types_bzl",
48+
"@bazel_skylib//lib:shell",
49+
"@pythons_hub//:versions_bzl",
3750
],
3851
)
3952

@@ -81,3 +94,13 @@ bzl_library(
8194
"//python/private:text_util_bzl",
8295
],
8396
)
97+
98+
filegroup(
99+
name = "lock_template",
100+
srcs = select({
101+
"@platforms//os:windows": ["lock.bat"],
102+
"//conditions:default": ["lock.sh"],
103+
}),
104+
target_compatible_with = [] if BZLMOD_ENABLED else ["@platforms//:incompatible"],
105+
visibility = ["//visibility:public"],
106+
)

python/uv/private/lock.bat

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
if defined BUILD_WORKSPACE_DIRECTORY (
2+
set "out=%BUILD_WORKSPACE_DIRECTORY%\{{src_out}}"
3+
) else (
4+
exit /b 1
5+
)
6+
7+
"{{args}}" --output-file "%out%" %*

0 commit comments

Comments
 (0)