Joshua Haberman | 5d8c3db | 2021-08-18 15:16:30 -0700 | [diff] [blame] | 1 | # Copyright (c) 2009-2021, Google LLC |
| 2 | # All rights reserved. |
| 3 | # |
Joshua Haberman | 2d6e554 | 2023-09-08 17:13:32 -0700 | [diff] [blame] | 4 | # Use of this source code is governed by a BSD-style |
| 5 | # license that can be found in the LICENSE file or at |
| 6 | # https://developers.google.com/open-source/licenses/bsd |
Joshua Haberman | 5d8c3db | 2021-08-18 15:16:30 -0700 | [diff] [blame] | 7 | |
| 8 | """Repository rule for using Python 3.x headers from the system.""" |
Joshua Haberman | 94ece04 | 2021-08-18 12:38:26 -0700 | [diff] [blame] | 9 | |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 10 | # Mock out rules_python's pip.bzl for cases where no system python is found. |
| 11 | _mock_pip = """ |
| 12 | def _pip_install_impl(repository_ctx): |
| 13 | repository_ctx.file("BUILD.bazel", ''' |
| 14 | py_library( |
| 15 | name = "noop", |
| 16 | visibility = ["//visibility:public"], |
| 17 | ) |
| 18 | ''') |
| 19 | repository_ctx.file("requirements.bzl", ''' |
| 20 | def install_deps(*args, **kwargs): |
| 21 | print("WARNING: could not install pip dependencies") |
| 22 | |
| 23 | def requirement(*args, **kwargs): |
| 24 | return "@{}//:noop" |
| 25 | '''.format(repository_ctx.attr.name)) |
| 26 | pip_install = repository_rule( |
| 27 | implementation = _pip_install_impl, |
| 28 | attrs = { |
| 29 | "requirements": attr.string(), |
| 30 | "requirements_overrides": attr.string_dict(), |
| 31 | "python_interpreter_target": attr.string(), |
| 32 | }, |
| 33 | ) |
| 34 | pip_parse = pip_install |
| 35 | """ |
| 36 | |
Deanna Garcia | 92dbe4b | 2022-12-05 11:13:38 -0800 | [diff] [blame] | 37 | # Alias rules_python's pip.bzl for cases where a system python is found. |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 38 | _alias_pip = """ |
| 39 | load("@rules_python//python:pip.bzl", _pip_install = "pip_install", _pip_parse = "pip_parse") |
Deanna Garcia | 92dbe4b | 2022-12-05 11:13:38 -0800 | [diff] [blame] | 40 | |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 41 | def _get_requirements(requirements, requirements_overrides): |
| 42 | for version, override in requirements_overrides.items(): |
| 43 | if version in "{python_version}": |
| 44 | requirements = override |
| 45 | break |
| 46 | return requirements |
| 47 | |
| 48 | def pip_install(requirements, requirements_overrides={{}}, **kwargs): |
| 49 | _pip_install( |
| 50 | python_interpreter_target = "@{repo}//:interpreter", |
| 51 | requirements = _get_requirements(requirements, requirements_overrides), |
| 52 | **kwargs, |
| 53 | ) |
| 54 | def pip_parse(requirements, requirements_overrides={{}}, **kwargs): |
| 55 | _pip_parse( |
| 56 | python_interpreter_target = "@{repo}//:interpreter", |
| 57 | requirements = _get_requirements(requirements, requirements_overrides), |
| 58 | **kwargs, |
| 59 | ) |
Mike Kruskal | ea1996f | 2022-12-12 10:39:09 -0800 | [diff] [blame] | 60 | """ |
| 61 | |
| 62 | _mock_fuzzing_py = """ |
| 63 | def fuzzing_py_install_deps(): |
| 64 | print("WARNING: could not install fuzzing_py dependencies") |
| 65 | """ |
| 66 | |
| 67 | # Alias rules_fuzzing's requirements.bzl for cases where a system python is found. |
| 68 | _alias_fuzzing_py = """ |
| 69 | load("@fuzzing_py_deps//:requirements.bzl", _fuzzing_py_install_deps = "install_deps") |
Deanna Garcia | 92dbe4b | 2022-12-05 11:13:38 -0800 | [diff] [blame] | 70 | |
| 71 | def fuzzing_py_install_deps(): |
| 72 | _fuzzing_py_install_deps() |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 73 | """ |
| 74 | |
Joshua Haberman | 94ece04 | 2021-08-18 12:38:26 -0700 | [diff] [blame] | 75 | _build_file = """ |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 76 | load("@bazel_skylib//lib:selects.bzl", "selects") |
| 77 | load("@bazel_skylib//rules:common_settings.bzl", "string_flag") |
Joshua Haberman | 7183780 | 2021-08-24 07:58:37 -0700 | [diff] [blame] | 78 | load("@bazel_tools//tools/python:toolchain.bzl", "py_runtime_pair") |
Protobuf Team | ee6b1ab | 2022-04-13 09:06:17 -0700 | [diff] [blame] | 79 | |
Joshua Haberman | 94ece04 | 2021-08-18 12:38:26 -0700 | [diff] [blame] | 80 | cc_library( |
| 81 | name = "python_headers", |
Adam Cozzette | 94c4c32 | 2023-08-23 06:36:52 -0700 | [diff] [blame] | 82 | hdrs = glob(["python/**/*.h"], allow_empty = True), |
Joshua Haberman | 94ece04 | 2021-08-18 12:38:26 -0700 | [diff] [blame] | 83 | includes = ["python"], |
| 84 | visibility = ["//visibility:public"], |
| 85 | ) |
Joshua Haberman | 7183780 | 2021-08-24 07:58:37 -0700 | [diff] [blame] | 86 | |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 87 | string_flag( |
| 88 | name = "internal_python_support", |
| 89 | build_setting_default = "{support}", |
| 90 | values = [ |
| 91 | "None", |
| 92 | "Supported", |
| 93 | "Unsupported", |
| 94 | ] |
| 95 | ) |
| 96 | |
| 97 | config_setting( |
| 98 | name = "none", |
| 99 | flag_values = {{ |
| 100 | ":internal_python_support": "None", |
| 101 | }}, |
| 102 | visibility = ["//visibility:public"], |
| 103 | ) |
| 104 | |
| 105 | config_setting( |
| 106 | name = "supported", |
| 107 | flag_values = {{ |
| 108 | ":internal_python_support": "Supported", |
| 109 | }}, |
| 110 | visibility = ["//visibility:public"], |
| 111 | ) |
| 112 | |
| 113 | config_setting( |
| 114 | name = "unsupported", |
| 115 | flag_values = {{ |
| 116 | ":internal_python_support": "Unsupported", |
| 117 | }}, |
| 118 | visibility = ["//visibility:public"], |
| 119 | ) |
| 120 | |
| 121 | selects.config_setting_group( |
| 122 | name = "exists", |
| 123 | match_any = [":supported", ":unsupported"], |
| 124 | visibility = ["//visibility:public"], |
| 125 | ) |
| 126 | |
| 127 | sh_binary( |
| 128 | name = "interpreter", |
| 129 | srcs = ["interpreter"], |
| 130 | visibility = ["//visibility:public"], |
| 131 | ) |
| 132 | |
Joshua Haberman | 7183780 | 2021-08-24 07:58:37 -0700 | [diff] [blame] | 133 | py_runtime( |
| 134 | name = "py3_runtime", |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 135 | interpreter_path = "{interpreter}", |
Joshua Haberman | 7183780 | 2021-08-24 07:58:37 -0700 | [diff] [blame] | 136 | python_version = "PY3", |
| 137 | ) |
| 138 | |
| 139 | py_runtime_pair( |
| 140 | name = "runtime_pair", |
| 141 | py3_runtime = ":py3_runtime", |
| 142 | ) |
| 143 | |
| 144 | toolchain( |
| 145 | name = "python_toolchain", |
| 146 | toolchain = ":runtime_pair", |
| 147 | toolchain_type = "@rules_python//python:toolchain_type", |
| 148 | ) |
Joshua Haberman | 94ece04 | 2021-08-18 12:38:26 -0700 | [diff] [blame] | 149 | """ |
| 150 | |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 151 | _register = """ |
| 152 | def register_system_python(): |
| 153 | native.register_toolchains("@{}//:python_toolchain") |
| 154 | """ |
Protobuf Team | ee6b1ab | 2022-04-13 09:06:17 -0700 | [diff] [blame] | 155 | |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 156 | _mock_register = """ |
| 157 | def register_system_python(): |
| 158 | pass |
| 159 | """ |
| 160 | |
| 161 | def _get_python_version(repository_ctx): |
| 162 | py_program = "import sys; print(str(sys.version_info.major) + '.' + str(sys.version_info.minor) + '.' + str(sys.version_info.micro))" |
| 163 | result = repository_ctx.execute(["python3", "-c", py_program]) |
| 164 | return (result.stdout).strip().split(".") |
| 165 | |
| 166 | def _get_python_path(repository_ctx): |
Joshua Haberman | 711885d | 2022-02-26 09:48:05 -0800 | [diff] [blame] | 167 | py_program = "import sysconfig; print(sysconfig.get_config_var('%s'), end='')" |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 168 | result = repository_ctx.execute(["python3", "-c", py_program % ("INCLUDEPY")]) |
Joshua Haberman | 711885d | 2022-02-26 09:48:05 -0800 | [diff] [blame] | 169 | if result.return_code != 0: |
Joshua Haberman | 7f9135b | 2022-04-22 15:29:11 -0700 | [diff] [blame] | 170 | return None |
Joshua Haberman | 711885d | 2022-02-26 09:48:05 -0800 | [diff] [blame] | 171 | return result.stdout |
Joshua Haberman | 94ece04 | 2021-08-18 12:38:26 -0700 | [diff] [blame] | 172 | |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 173 | def _populate_package(ctx, path, python3, python_version): |
| 174 | ctx.symlink(path, "python") |
Mike Kruskal | ee56471 | 2022-12-02 09:16:31 -0800 | [diff] [blame] | 175 | supported = True |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 176 | for idx, v in enumerate(ctx.attr.minimum_python_version.split(".")): |
| 177 | if int(python_version[idx]) < int(v): |
Mike Kruskal | ee56471 | 2022-12-02 09:16:31 -0800 | [diff] [blame] | 178 | supported = False |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 179 | break |
Mike Kruskal | 008a9af | 2023-02-01 08:04:27 -0800 | [diff] [blame] | 180 | if "win" in ctx.os.name: |
| 181 | # buildifier: disable=print |
| 182 | print("WARNING: python is not supported on Windows") |
| 183 | supported = False |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 184 | |
| 185 | build_file = _build_file.format( |
| 186 | interpreter = python3, |
Mike Kruskal | ee56471 | 2022-12-02 09:16:31 -0800 | [diff] [blame] | 187 | support = "Supported" if supported else "Unsupported", |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 188 | ) |
| 189 | |
Vincent Thiberville | 558c772 | 2023-06-20 08:13:46 -0700 | [diff] [blame] | 190 | ctx.file("interpreter", "#!/bin/sh\nexec {} \"$@\"".format(python3)) |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 191 | ctx.file("BUILD.bazel", build_file) |
| 192 | ctx.file("version.bzl", "SYSTEM_PYTHON_VERSION = '{}{}'".format(python_version[0], python_version[1])) |
| 193 | ctx.file("register.bzl", _register.format(ctx.attr.name)) |
Mike Kruskal | ee56471 | 2022-12-02 09:16:31 -0800 | [diff] [blame] | 194 | if supported: |
| 195 | ctx.file("pip.bzl", _alias_pip.format( |
| 196 | python_version = ".".join(python_version), |
| 197 | repo = ctx.attr.name, |
| 198 | )) |
Mike Kruskal | ea1996f | 2022-12-12 10:39:09 -0800 | [diff] [blame] | 199 | ctx.file("fuzzing_py.bzl", _alias_fuzzing_py) |
Mike Kruskal | ee56471 | 2022-12-02 09:16:31 -0800 | [diff] [blame] | 200 | else: |
Mike Kruskal | ea1996f | 2022-12-12 10:39:09 -0800 | [diff] [blame] | 201 | # Dependencies are unlikely to be satisfiable for unsupported versions of python. |
Mike Kruskal | ee56471 | 2022-12-02 09:16:31 -0800 | [diff] [blame] | 202 | ctx.file("pip.bzl", _mock_pip) |
Mike Kruskal | ea1996f | 2022-12-12 10:39:09 -0800 | [diff] [blame] | 203 | ctx.file("fuzzing_py.bzl", _mock_fuzzing_py) |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 204 | |
| 205 | def _populate_empty_package(ctx): |
| 206 | # Mock out all the entrypoints we need to run from WORKSPACE. Targets that |
| 207 | # actually need python should use `target_compatible_with` and the generated |
| 208 | # @system_python//:exists or @system_python//:supported constraints. |
| 209 | ctx.file( |
| 210 | "BUILD.bazel", |
| 211 | _build_file.format( |
| 212 | interpreter = "", |
| 213 | support = "None", |
| 214 | ), |
| 215 | ) |
| 216 | ctx.file("version.bzl", "SYSTEM_PYTHON_VERSION = 'None'") |
| 217 | ctx.file("register.bzl", _mock_register) |
| 218 | ctx.file("pip.bzl", _mock_pip) |
Mike Kruskal | ea1996f | 2022-12-12 10:39:09 -0800 | [diff] [blame] | 219 | ctx.file("fuzzing_py.bzl", _mock_fuzzing_py) |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 220 | |
| 221 | def _system_python_impl(repository_ctx): |
| 222 | path = _get_python_path(repository_ctx) |
Joshua Haberman | 711885d | 2022-02-26 09:48:05 -0800 | [diff] [blame] | 223 | python3 = repository_ctx.which("python3") |
Protobuf Team | ee6b1ab | 2022-04-13 09:06:17 -0700 | [diff] [blame] | 224 | python_version = _get_python_version(repository_ctx) |
Joshua Haberman | 94ece04 | 2021-08-18 12:38:26 -0700 | [diff] [blame] | 225 | |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 226 | if path and python_version[0] == "3": |
| 227 | _populate_package(repository_ctx, path, python3, python_version) |
| 228 | else: |
| 229 | # buildifier: disable=print |
| 230 | print("WARNING: no system python available, builds against system python will fail") |
| 231 | _populate_empty_package(repository_ctx) |
| 232 | |
| 233 | # The system_python() repository rule exposes information from the version of python installed in the current system. |
Joshua Haberman | 5d8c3db | 2021-08-18 15:16:30 -0700 | [diff] [blame] | 234 | # |
| 235 | # In WORKSPACE: |
Joshua Haberman | 7183780 | 2021-08-24 07:58:37 -0700 | [diff] [blame] | 236 | # system_python( |
| 237 | # name = "system_python_repo", |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 238 | # minimum_python_version = "3.7", |
Joshua Haberman | 5d8c3db | 2021-08-18 15:16:30 -0700 | [diff] [blame] | 239 | # ) |
| 240 | # |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 241 | # This repository exposes some repository rules for configuring python in Bazel. The python toolchain |
| 242 | # *must* be registered in your WORKSPACE: |
| 243 | # load("@system_python_repo//:register.bzl", "register_system_python") |
| 244 | # register_system_python() |
| 245 | # |
| 246 | # Pip dependencies can optionally be specified using a wrapper around rules_python's repository rules: |
| 247 | # load("@system_python//:pip.bzl", "pip_install") |
| 248 | # pip_install( |
| 249 | # name="pip_deps", |
| 250 | # requirements = "@com_google_protobuf//python:requirements.txt", |
| 251 | # ) |
| 252 | # An optional argument `requirements_overrides` takes a dictionary mapping python versions to alternate |
| 253 | # requirements files. This works around the requirement for fully pinned dependencies in python_rules. |
| 254 | # |
| 255 | # Four config settings are exposed from this repository to help declare target compatibility in Bazel. |
| 256 | # For example, `@system_python_repo//:exists` will be true if a system python version has been found. |
| 257 | # The `none` setting will be true only if no python version was found, and `supported`/`unsupported` |
| 258 | # correspond to whether or not the system version is compatible with `minimum_python_version`. |
| 259 | # |
| 260 | # This repository also exposes a header rule that you can depend on from BUILD files: |
Joshua Haberman | 5d8c3db | 2021-08-18 15:16:30 -0700 | [diff] [blame] | 261 | # cc_library( |
| 262 | # name = "foobar", |
| 263 | # srcs = ["foobar.cc"], |
Joshua Haberman | 7183780 | 2021-08-24 07:58:37 -0700 | [diff] [blame] | 264 | # deps = ["@system_python_repo//:python_headers"], |
Joshua Haberman | 5d8c3db | 2021-08-18 15:16:30 -0700 | [diff] [blame] | 265 | # ) |
| 266 | # |
Joshua Haberman | ae16098 | 2021-08-18 16:45:42 -0700 | [diff] [blame] | 267 | # The headers should correspond to the version of python obtained by running |
| 268 | # the `python3` command on the system. |
Joshua Haberman | 7183780 | 2021-08-24 07:58:37 -0700 | [diff] [blame] | 269 | system_python = repository_rule( |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 270 | implementation = _system_python_impl, |
Joshua Haberman | 94ece04 | 2021-08-18 12:38:26 -0700 | [diff] [blame] | 271 | local = True, |
Mike Kruskal | 248ed86 | 2022-11-30 19:54:51 -0800 | [diff] [blame] | 272 | attrs = { |
| 273 | "minimum_python_version": attr.string(default = "3.7"), |
| 274 | }, |
Joshua Haberman | 94ece04 | 2021-08-18 12:38:26 -0700 | [diff] [blame] | 275 | ) |