| # Copyright 2018 The Bazel Authors. All rights reserved. |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| """Defines a repository rule that generates an archive consisting of the specified files to fetch""" |
| |
| load("//src/tools/bzlmod:utils.bzl", "parse_http_artifacts", "parse_registry_files") |
| |
| _BUILD = """ |
| load("@rules_pkg//pkg:tar.bzl", "pkg_tar") |
| |
| filegroup( |
| name="files", |
| srcs = {srcs}, |
| visibility = ["//visibility:public"], |
| ) |
| |
| pkg_tar( |
| name="archives", |
| srcs = [":files"], |
| strip_prefix = "{strip_prefix}", |
| package_dir = "{dirname}", |
| visibility = ["//visibility:public"], |
| ) |
| |
| """ |
| |
| def _distdir_tar_impl(ctx): |
| for name in ctx.attr.archives: |
| ctx.download(ctx.attr.urls[name], name, ctx.attr.sha256[name], False) |
| ctx.file("WORKSPACE", "") |
| ctx.file( |
| "BUILD", |
| _BUILD.format(srcs = ctx.attr.archives, strip_prefix = "", dirname = ctx.attr.dirname), |
| ) |
| |
| _distdir_tar_attrs = { |
| "archives": attr.string_list(), |
| "sha256": attr.string_dict(), |
| "urls": attr.string_list_dict(), |
| "dirname": attr.string(default = "distdir"), |
| } |
| |
| _distdir_tar = repository_rule( |
| implementation = _distdir_tar_impl, |
| attrs = _distdir_tar_attrs, |
| ) |
| |
| def distdir_tar(name, dist_deps): |
| """Creates a repository whose content is a set of tar files. |
| |
| Args: |
| name: repo name. |
| dist_deps: map of repo names to dict of archive, sha256, and urls. |
| """ |
| archives = [] |
| sha256 = {} |
| urls = {} |
| for _, info in dist_deps.items(): |
| archive_file = info["archive"] |
| archives.append(archive_file) |
| sha256[archive_file] = info["sha256"] |
| urls[archive_file] = info["urls"] |
| _distdir_tar( |
| name = name, |
| archives = archives, |
| sha256 = sha256, |
| urls = urls, |
| ) |
| |
| def _repo_cache_tar_impl(ctx): |
| """Generate a repository cache as a tar file. |
| |
| This repository rule does the following: |
| 1. parse all http artifacts required for generating the given list of repositories from the lock file. |
| 2. downloads all http artifacts to create a repository cache directory structure. |
| 3. creates a pkg_tar target which packages the repository cache directory structure. |
| """ |
| lockfile_path = ctx.path(ctx.attr.lockfile) |
| http_artifacts = parse_http_artifacts(ctx, lockfile_path, ctx.attr.repos) |
| registry_files = parse_registry_files(ctx, lockfile_path, ctx.attr.module_files) |
| |
| if "protobuf+" not in ctx.attr.repos: |
| # HACK: protobuf is currently an archive_override, so it doesn't show up in the lockfile. |
| # we manually add it to the tar entry here. |
| http_artifacts.append({ |
| "url": "https://github.com/protocolbuffers/protobuf/releases/download/v29.0-rc1/protobuf-29.0-rc1.zip", |
| "integrity": "sha256-tSay4N4FspF+VnsNCTGtMH3xV4ZrtHioxNeB/bjQhsI=", |
| }) |
| |
| archive_files = [] |
| readme_content = "This directory contains repository cache artifacts for the following URLs:\n\n" |
| for artifact in http_artifacts + registry_files: |
| url = artifact["url"] |
| if "integrity" in artifact: |
| # ./tempfile could be a hard link if --experimental_repository_cache_hardlinks is used, |
| # therefore we must delete it before creating or writing it again. |
| ctx.delete("./tempfile") |
| checksum = ctx.download(url, "./tempfile", executable = False, integrity = artifact["integrity"]) |
| artifact["sha256"] = checksum.sha256 |
| |
| if "sha256" in artifact: |
| sha256 = artifact["sha256"] |
| output_file = "content_addressable/sha256/%s/file" % sha256 |
| ctx.download(url, output_file, sha256, executable = False) |
| archive_files.append(output_file) |
| readme_content += "- %s (SHA256: %s)\n" % (url, sha256) |
| else: |
| fail("Could not find integrity or sha256 hash for artifact %s" % url) |
| |
| ctx.file("README.md", readme_content) |
| ctx.file( |
| "BUILD", |
| _BUILD.format( |
| srcs = archive_files + ["README.md"], |
| strip_prefix = "external/" + ctx.attr.name, |
| dirname = ctx.attr.dirname, |
| ), |
| ) |
| |
| _repo_cache_tar_attrs = { |
| "lockfile": attr.label(default = Label("//:MODULE.bazel.lock")), |
| "dirname": attr.string(default = "repository_cache"), |
| "repos": attr.string_list(), |
| "module_files": attr.label_list(), |
| } |
| |
| repo_cache_tar = repository_rule( |
| implementation = _repo_cache_tar_impl, |
| attrs = _repo_cache_tar_attrs, |
| ) |