-
Notifications
You must be signed in to change notification settings - Fork 185
/
Copy pathutils.libsonnet
148 lines (139 loc) · 5.79 KB
/
utils.libsonnet
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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
# File is formatted with
# `jsonnet fmt --indent 2 --max-blank-lines 2 --sort-imports --string-style d --comment-style h -i utils.libsonnet`
# Helper functions for CI definition. They only help with the structure
# and warn about broken rules. With few small modifications in final build
# field for comprehension they could be completely dropped and this file would
# still produce the same result.
{
# It computes full name for each part (e.g. $.platform.linux) and adds field
# `included_parts+: ['$.platform.linux']` into each part. When parts are
# added together to form a build the array in `included_parts` composes and
# keeps track of all parts used to compose the build.
# The function returns part_definitions with the included_parts added to
# each part.
add_inclusion_tracking:: function(part_definitions, prefix, contains_parts)
{
local content = part_definitions[k],
local new_prefix = prefix + "." + k,
[k]: (
if std.type(content) == "object" && contains_parts
then
# process the part object
content {
included_parts+:
[new_prefix],
} + (
if std.objectHasAll(content, "is_before")
then { is_before:: [[new_prefix, t] for t in content.is_before] }
else {}
) + (
if std.objectHasAll(content, "is_before_optional")
then { is_before_optional:: [[new_prefix, t] for t in content.is_before_optional] }
else {}
) + (
if std.objectHasAll(content, "is_after")
then { is_after:: [[new_prefix, t] for t in content.is_after] }
else {}
) + (
if std.objectHasAll(content, "is_after_optional")
then { is_after_optional:: [[new_prefix, t] for t in content.is_after_optional] }
else {}
)
else
# look recursively for parts
if std.type(content) == "object"
then $.add_inclusion_tracking(
content,
new_prefix,
# assume parts are always nested 1 level in a group
true
)
else content
)
for k in std.objectFields(part_definitions)
},
check_order_inner:: function(build, where, optional)
local field_name = "is_" + where + if optional then "_optional" else "";
if std.objectHasAll(build, field_name)
then
std.foldl(
function(r, specifee_target)
local specifee = specifee_target[0],
target = specifee_target[1],
specifee_index = $.index(build.included_parts, specifee),
target_index = $.index(build.included_parts, target),
check = ((optional && target_index == "none") ||
(target_index != "none" &&
if where == "before"
then specifee_index < target_index
else if where == "after"
then specifee_index > target_index
else error "bad where argument: " + where));
(r && check) ||
error specifee + " has to be " + where + " " + target +
"\nfound in " + build.name + "\nhaving " + build.included_parts,
build[field_name],
true
)
else true,
# checks order of parts in included_parts optionally
# specified in is_after and is_before
check_order:: function(build)
local before_check = $.check_order_inner(build, "before", false);
local before_optional_check = $.check_order_inner(build, "before", true);
local after_check = $.check_order_inner(build, "after", false);
local after_optional_check = $.check_order_inner(build, "after", true);
if before_check && before_optional_check && after_check && after_optional_check then build,
# Ensures that no part is included twice using `included_parts` field.
included_once_check:: function(build)
local name = if std.objectHas(build, "name") then build.name else build;
# collect repeated parts
local repeated = std.foldl(function(r, i)
if std.count(build.included_parts, i) == 1
then r else r + [i],
build.included_parts,
[]);
if std.length(repeated) == 0
then build
else error "Parts " + repeated +
" are used more than once in build: " + name +
". See for duplicates: " + build.included_parts,
# should build be used with given restrictions
used:: function(restriction, build)
if std.length(restriction) == 0
then true
else std.count(restriction, build.name) > 0,
# perform all checks
check_builds:: function(restrict_to, builds)
[
$.included_once_check($.check_order(build)) {
environment+: {
# Add PARTS_INCLUDED_IN_BUILD env var so the parts used
# in the build are printed in its log
PARTS_INCLUDED_IN_BUILD:
std.join(
", ",
std.map(
# there is no way to escape $ afaik, stripping it instead,
# otherwise it is interpreted as variable
function(name) std.substr(name, 2, std.length(name) - 2),
build.included_parts
)
),
},
}
for build in builds
if $.used(restrict_to, build)
],
# Try to read a field, if not present print error including the name
# of the object
debug_read:: function(obj, field)
if std.objectHasAll(obj, field)
then obj[field]
else error "missing field: " + field + " in " + obj.name,
index:: function(arr, obj)
local indexes = std.filter(function(i) obj == arr[i], std.range(0, std.length(arr) - 1));
if std.length(indexes) == 0
then "none"
else indexes[0],
}