| import contextlib |
| import os |
| import platform |
| import shutil |
| import sysconfig |
| from pathlib import Path |
| from typing import Generator |
| |
| import setuptools |
| from setuptools.command import build_ext |
| |
| PYTHON_INCLUDE_PATH_PLACEHOLDER = "<PYTHON_INCLUDE_PATH>" |
| |
| IS_WINDOWS = platform.system() == "Windows" |
| IS_MAC = platform.system() == "Darwin" |
| |
| |
| @contextlib.contextmanager |
| def temp_fill_include_path(fp: str) -> Generator[None, None, None]: |
| """Temporarily set the Python include path in a file.""" |
| with open(fp, "r+") as f: |
| try: |
| content = f.read() |
| replaced = content.replace( |
| PYTHON_INCLUDE_PATH_PLACEHOLDER, |
| Path(sysconfig.get_paths()["include"]).as_posix(), |
| ) |
| f.seek(0) |
| f.write(replaced) |
| f.truncate() |
| yield |
| finally: |
| # revert to the original content after exit |
| f.seek(0) |
| f.write(content) |
| f.truncate() |
| |
| |
| class BazelExtension(setuptools.Extension): |
| """A C/C++ extension that is defined as a Bazel BUILD target.""" |
| |
| def __init__(self, name: str, bazel_target: str): |
| super().__init__(name=name, sources=[]) |
| |
| self.bazel_target = bazel_target |
| stripped_target = bazel_target.split("//")[-1] |
| self.relpath, self.target_name = stripped_target.split(":") |
| |
| |
| class BuildBazelExtension(build_ext.build_ext): |
| """A command that runs Bazel to build a C/C++ extension.""" |
| |
| def run(self): |
| for ext in self.extensions: |
| self.bazel_build(ext) |
| super().run() |
| # explicitly call `bazel shutdown` for graceful exit |
| self.spawn(["bazel", "shutdown"]) |
| |
| def copy_extensions_to_source(self): |
| """ |
| Copy generated extensions into the source tree. |
| This is done in the ``bazel_build`` method, so it's not necessary to |
| do again in the `build_ext` base class. |
| """ |
| pass |
| |
| def bazel_build(self, ext: BazelExtension) -> None: |
| """Runs the bazel build to create the package.""" |
| with temp_fill_include_path("WORKSPACE"): |
| temp_path = Path(self.build_temp) |
| |
| bazel_argv = [ |
| "bazel", |
| "build", |
| ext.bazel_target, |
| "--enable_bzlmod=false", |
| f"--symlink_prefix={temp_path / 'bazel-'}", |
| f"--compilation_mode={'dbg' if self.debug else 'opt'}", |
| # C++17 is required by nanobind |
| f"--cxxopt={'/std:c++17' if IS_WINDOWS else '-std=c++17'}", |
| ] |
| |
| if IS_WINDOWS: |
| # Link with python*.lib. |
| for library_dir in self.library_dirs: |
| bazel_argv.append("--linkopt=/LIBPATH:" + library_dir) |
| elif IS_MAC: |
| if platform.machine() == "x86_64": |
| # C++17 needs macOS 10.14 at minimum |
| bazel_argv.append("--macos_minimum_os=10.14") |
| |
| # cross-compilation for Mac ARM64 on GitHub Mac x86 runners. |
| # ARCHFLAGS is set by cibuildwheel before macOS wheel builds. |
| archflags = os.getenv("ARCHFLAGS", "") |
| if "arm64" in archflags: |
| bazel_argv.append("--cpu=darwin_arm64") |
| bazel_argv.append("--macos_cpus=arm64") |
| |
| elif platform.machine() == "arm64": |
| bazel_argv.append("--macos_minimum_os=11.0") |
| |
| self.spawn(bazel_argv) |
| |
| shared_lib_suffix = ".dll" if IS_WINDOWS else ".so" |
| ext_name = ext.target_name + shared_lib_suffix |
| ext_bazel_bin_path = ( |
| temp_path / "bazel-bin" / ext.relpath / ext_name |
| ) |
| |
| ext_dest_path = Path(self.get_ext_fullpath(ext.name)) |
| shutil.copyfile(ext_bazel_bin_path, ext_dest_path) |
| |
| |
| setuptools.setup( |
| cmdclass=dict(build_ext=BuildBazelExtension), |
| ext_modules=[ |
| BazelExtension( |
| name="google_benchmark._benchmark", |
| bazel_target="//bindings/python/google_benchmark:_benchmark", |
| ) |
| ], |
| ) |