| "Provide access to a BSD tar" |
| |
| load(":repo_utils.bzl", "repo_utils") |
| |
| BSDTAR_PLATFORMS = { |
| "linux_amd64": struct( |
| compatible_with = [ |
| "@platforms//os:linux", |
| "@platforms//cpu:x86_64", |
| ], |
| ), |
| "linux_arm64": struct( |
| compatible_with = [ |
| "@platforms//os:linux", |
| "@platforms//cpu:aarch64", |
| ], |
| ), |
| "windows_amd64": struct( |
| release_platform = "win64", |
| compatible_with = [ |
| "@platforms//os:windows", |
| "@platforms//cpu:x86_64", |
| ], |
| ), |
| # WARNING: host toolchain should always come last to make it a fallback toolchain. |
| "host": struct( |
| # loaded by the macro |
| compatible_with = "HOST_CONSTRAINTS", |
| ), |
| } |
| |
| WINDOWS_DEPS = ( |
| "e06f10043b1b148eb38ad06cff678af05beade0bdd2edd8735a198c521fa3993", |
| "https://github.com/libarchive/libarchive/releases/download/v3.7.2/libarchive-v3.7.2-amd64.zip", |
| ) |
| |
| # note, using Ubuntu Focal packages as they link with older glibc versions. |
| # Ubuntu Jammy packages will fail on ubuntu 20.02 with |
| # bsdtar: /lib/x86_64-linux-gnu/libc.so.6: version `GLIBC_2.33' not found |
| # bsdtar: /lib/x86_64-linux-gnu/libc.so.6: version `GLIBC_2.34' not found |
| # |
| # TODO: this is only a partial listing of the transitive deps of libarchive-tools |
| # so we expect a bunch of compress modes are broken, for example. |
| |
| LINUX_LIB_DEPS = { |
| "linux_arm64": [ |
| ( |
| "6d18525e248e84b8a4ee39a226fd1195ca9b9d0d5a1c7909ae4f997d46378848", |
| "http://ports.ubuntu.com/pool/main/n/nettle/libnettle7_3.5.1+really3.5.1-2ubuntu0.2_arm64.deb", |
| ), |
| ( |
| "aa5e31d05a9d6bde8093137bd1c82b5a20a5f470bd5109642014f895c20f323a", |
| "http://ports.ubuntu.com/pool/main/liba/libarchive/libarchive13_3.4.0-2ubuntu1_arm64.deb", |
| ), |
| ( |
| "6d089f878507b536d8ca51b1ad80a80706a1dd7dbbcce7600800d3f9f98be2ab", |
| "http://ports.ubuntu.com/pool/main/liba/libarchive/libarchive-tools_3.2.1-2~ubuntu16.04.1_arm64.deb", |
| ), |
| ( |
| "6242892cb032859044ddfcfbe61bac5678a95c585d8fff4525acaf45512e3d39", |
| "http://ports.ubuntu.com/pool/main/libx/libxml2/libxml2_2.9.10+dfsg-5_arm64.deb", |
| ), |
| ( |
| "6302e309ab002af30ddfa0d68de26c68f7c034ed2f45b1d97a712bff1a03999a", |
| "http://ports.ubuntu.com/pool/main/i/icu/libicu66_66.1-2ubuntu2_arm64.deb", |
| ), |
| ], |
| "linux_amd64": [ |
| # https://packages.ubuntu.com/focal/amd64/libarchive-tools/download |
| ( |
| "12a19878d34b407e6f4893d3b26b7758a26c5534a066d76184c8b764b2df1652", |
| "http://security.ubuntu.com/ubuntu/pool/universe/liba/libarchive/libarchive-tools_3.4.0-2ubuntu1.2_amd64.deb", |
| ), |
| # https://packages.ubuntu.com/focal/amd64/libarchive13/download |
| ( |
| "8ba7507f61bb3ea8da488702ec0badcbfb726d36ea6886e3421ac59082aaf2d1", |
| "http://security.ubuntu.com/ubuntu/pool/main/liba/libarchive/libarchive13_3.4.0-2ubuntu1.2_amd64.deb", |
| ), |
| # https://packages.ubuntu.com/focal/amd64/libnettle7/download |
| ( |
| "3496aed83407fde71e0dc5988b28e8fd7f07a2f27fcf3e0f214c7cd86667eecd", |
| "http://security.ubuntu.com/ubuntu/pool/main/n/nettle/libnettle7_3.5.1+really3.5.1-2ubuntu0.2_amd64.deb", |
| ), |
| # https://packages.ubuntu.com/focal/amd64/libxml2/download |
| ( |
| "a8cbd10a0d74ff8ec43a7e6c09ad07629f20efea9972799d9ff7f63c4e82bfcf", |
| "http://security.ubuntu.com/ubuntu/pool/main/libx/libxml2/libxml2_2.9.10+dfsg-5ubuntu0.20.04.6_amd64.deb", |
| ), |
| # https://packages.ubuntu.com/focal/amd64/libicu66/download |
| ( |
| "00d0de456134668f41bd9ea308a076bc0a6a805180445af8a37209d433f41efe", |
| "http://security.ubuntu.com/ubuntu/pool/main/i/icu/libicu66_66.1-2ubuntu2.1_amd64.deb", |
| ), |
| ], |
| } |
| |
| def _find_usable_system_tar(rctx, tar_name): |
| tar = rctx.which(tar_name) |
| if not tar: |
| fail("tar not found on PATH, and we don't handle this case yet") |
| |
| # Run tar --version and see if we are satisfied to use it |
| tar_version = rctx.execute([tar, "--version"]).stdout.strip() |
| |
| # TODO: also check if it's really ancient or compiled without gzip support or something? |
| # TODO: document how users could fetch the source and compile it themselves |
| if tar_version.find("bsdtar") >= 0: |
| return tar |
| |
| fail("tar isn't a BSD tar") |
| |
| def _bsdtar_binary_repo(rctx): |
| tar_name = "tar.exe" if repo_utils.is_windows(rctx) else "tar" |
| build_header = """\ |
| # @generated by @aspect_bazel_lib//lib/private:tar_toolchain.bzl |
| |
| load("@aspect_bazel_lib//lib/private:tar_toolchain.bzl", "tar_toolchain") |
| |
| package(default_visibility = ["//visibility:public"]) |
| |
| """ |
| |
| # On MacOS, the system `tar` binary on the PATH should already work |
| if rctx.attr.platform == "host": |
| tar = _find_usable_system_tar(rctx, tar_name) |
| output = rctx.path(tar_name) |
| rctx.symlink(tar, output) |
| rctx.file("BUILD.bazel", build_header + """tar_toolchain(name = "bsdtar_toolchain", binary = "tar")""") |
| return |
| |
| if repo_utils.is_windows(rctx): |
| rctx.download_and_extract( |
| url = WINDOWS_DEPS[1], |
| type = "zip", |
| sha256 = WINDOWS_DEPS[0], |
| ) |
| rctx.file("BUILD.bazel", build_header + """tar_toolchain(name = "bsdtar_toolchain", binary = "libarchive/bin/bsdtar.exe")""") |
| return |
| |
| # Other platforms, we have more work to do. |
| libs_dir = "usr/lib/x86_64-linux-gnu" if rctx.attr.platform.endswith("amd64") else "usr/lib/aarch64-linux-gnu" |
| |
| for lib in LINUX_LIB_DEPS[rctx.attr.platform]: |
| rctx.download_and_extract( |
| url = lib[1], |
| type = "deb", |
| sha256 = lib[0], |
| ) |
| rctx.extract("data.tar.xz") |
| |
| rctx.file("bsdtar.sh", """#!/usr/bin/env bash |
| readonly wksp="$(dirname "${{BASH_SOURCE[0]}}")" |
| LD_LIBRARY_PATH=$wksp/{libs_dir} exec $wksp/usr/bin/bsdtar $@ |
| """.format(name = rctx.name, libs_dir = libs_dir)) |
| |
| rctx.file("BUILD.bazel", build_header + """\ |
| tar_toolchain( |
| name = "bsdtar_toolchain", |
| files = glob(["{libs}/*.so.*"]) + ["usr/bin/bsdtar"], |
| binary = "bsdtar.sh", |
| visibility = ["//visibility:public"], |
| ) |
| """.format(libs = libs_dir, name = rctx.name)) |
| |
| bsdtar_binary_repo = repository_rule( |
| implementation = _bsdtar_binary_repo, |
| attrs = { |
| "platform": attr.string(mandatory = True, values = BSDTAR_PLATFORMS.keys()), |
| }, |
| ) |
| |
| TarInfo = provider( |
| doc = "Provide info for executing BSD tar", |
| fields = { |
| "binary": "bsdtar executable", |
| }, |
| ) |
| |
| def _tar_toolchain_impl(ctx): |
| binary = ctx.executable.binary |
| |
| # Make the $(BSDTAR_BIN) variable available in places like genrules. |
| # See https://docs.bazel.build/versions/main/be/make-variables.html#custom_variables |
| template_variables = platform_common.TemplateVariableInfo({ |
| "BSDTAR_BIN": binary.path, |
| }) |
| |
| default_info = DefaultInfo( |
| files = depset(ctx.files.binary + ctx.files.files), |
| ) |
| tarinfo = TarInfo( |
| binary = binary, |
| ) |
| |
| # Export all the providers inside our ToolchainInfo |
| # so the resolved_toolchain rule can grab and re-export them. |
| toolchain_info = platform_common.ToolchainInfo( |
| tarinfo = tarinfo, |
| template_variables = template_variables, |
| default = default_info, |
| ) |
| |
| return [toolchain_info, template_variables, default_info] |
| |
| tar_toolchain = rule( |
| implementation = _tar_toolchain_impl, |
| attrs = { |
| "binary": attr.label( |
| doc = "a command to find on the system path", |
| allow_files = True, |
| executable = True, |
| cfg = "exec", |
| ), |
| "files": attr.label_list(allow_files = True), |
| }, |
| ) |
| |
| def _tar_toolchains_repo_impl(rctx): |
| # Expose a concrete toolchain which is the result of Bazel resolving the toolchain |
| # for the execution or target platform. |
| # Workaround for https://github.com/bazelbuild/bazel/issues/14009 |
| starlark_content = """\ |
| # @generated by @aspect_bazel_lib//lib/private:tar_toolchain.bzl |
| |
| # Forward all the providers |
| def _resolved_toolchain_impl(ctx): |
| toolchain_info = ctx.toolchains["@aspect_bazel_lib//lib:tar_toolchain_type"] |
| return [ |
| toolchain_info, |
| toolchain_info.default, |
| toolchain_info.tarinfo, |
| toolchain_info.template_variables, |
| ] |
| |
| # Copied from java_toolchain_alias |
| # https://cs.opensource.google/bazel/bazel/+/master:tools/jdk/java_toolchain_alias.bzl |
| resolved_toolchain = rule( |
| implementation = _resolved_toolchain_impl, |
| toolchains = ["@aspect_bazel_lib//lib:tar_toolchain_type"], |
| incompatible_use_toolchain_transition = True, |
| ) |
| """ |
| rctx.file("defs.bzl", starlark_content) |
| |
| build_content = """# @generated by @aspect_bazel_lib//lib/private:tar_toolchain.bzl |
| load(":defs.bzl", "resolved_toolchain") |
| load("@local_config_platform//:constraints.bzl", "HOST_CONSTRAINTS") |
| |
| resolved_toolchain(name = "resolved_toolchain", visibility = ["//visibility:public"])""" |
| |
| for [platform, meta] in BSDTAR_PLATFORMS.items(): |
| build_content += """ |
| toolchain( |
| name = "{platform}_toolchain", |
| exec_compatible_with = {compatible_with}, |
| toolchain = "@{user_repository_name}_{platform}//:bsdtar_toolchain", |
| toolchain_type = "@aspect_bazel_lib//lib:tar_toolchain_type", |
| ) |
| """.format( |
| platform = platform, |
| user_repository_name = rctx.attr.user_repository_name, |
| compatible_with = meta.compatible_with, |
| ) |
| |
| rctx.file("BUILD.bazel", build_content) |
| |
| tar_toolchains_repo = repository_rule( |
| _tar_toolchains_repo_impl, |
| doc = """Creates a repository that exposes a tar_toolchain_type target.""", |
| attrs = { |
| "user_repository_name": attr.string(doc = "Base name for toolchains repository"), |
| }, |
| ) |