|  | 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", | 
|  | ) | 
|  | ], | 
|  | ) |