-
Notifications
You must be signed in to change notification settings - Fork 4.1k
/
utils.bzl
99 lines (83 loc) · 4.03 KB
/
utils.bzl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
# Copyright 2023 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.
"""Helper functions for Bzlmod build"""
def get_canonical_repo_name(apparent_repo_name):
"""Returns the canonical repo name for the given apparent repo name seen by the module this bzl file belongs to."""
if not apparent_repo_name.startswith("@"):
apparent_repo_name = "@" + apparent_repo_name
return Label(apparent_repo_name).workspace_name
def extract_url(attributes):
"""Extracts the url from the given attributes.
Args:
attributes: The attributes to extract the url from.
Returns:
The url extracted from the given attributes.
"""
if "urls" in attributes:
return attributes["urls"][0][2:]
elif "url" in attributes:
return attributes["url"][2:]
else:
fail("Could not find url in attributes %s" % attributes)
def parse_http_artifacts(ctx, lockfile_path, required_repos):
"""Parses the http artifacts required from for fetching the given repos from the lockfile.
Args:
ctx: the repository / module extension ctx object.
lockfile_path: The path of the lockfile to extract the http artifacts from.
required_repos: The list of required repos to extract the http artifacts for,
only support `http_archive`, `http_file` and `http_jar` repo rules.
Returns:
A list of http artifacts in the form of
[{"integrity": <integrity value>, "url": <url>}, {"sha256": <sha256 value>, "url": <url>}, ...]
All lockfile string values are prefixed with `--`, hence the `[2:]` is needed to remove the prefix.
"""
lockfile = json.decode(ctx.read(lockfile_path))
http_artifacts = []
found_repos = []
for _, module in lockfile["moduleDepGraph"].items():
if "repoSpec" in module and module["repoSpec"]["ruleClassName"] == "http_archive":
repo_spec = module["repoSpec"]
attributes = repo_spec["attributes"]
repo_name = attributes["name"][2:]
if repo_name not in required_repos:
continue
found_repos.append(repo_name)
http_artifacts.append({
"integrity": attributes["integrity"][2:],
"url": extract_url(attributes),
})
if "remote_patches" in attributes:
for patch, integrity in attributes["remote_patches"].items():
http_artifacts.append({
"integrity": integrity[2:],
"url": patch[2:],
})
for _, extension in lockfile["moduleExtensions"].items():
# TODO(pcloudy): update this when lockfile format changes (https://github.com/bazelbuild/bazel/issues/19154)
for _, repo_spec in extension["generatedRepoSpecs"].items():
rule_class = repo_spec["ruleClassName"]
if rule_class == "http_archive" or rule_class == "http_file" or rule_class == "http_jar":
attributes = repo_spec["attributes"]
repo_name = attributes["name"][2:]
if repo_name not in required_repos:
continue
found_repos.append(repo_name)
http_artifacts.append({
"sha256": attributes["sha256"][2:],
"url": extract_url(attributes),
})
missing_repos = [repo for repo in required_repos if repo not in found_repos]
if missing_repos:
fail("Could not find all required repos, missing: %s" % missing_repos)
return http_artifacts