-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcoqbot-config.sh
336 lines (298 loc) · 12 KB
/
coqbot-config.sh
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
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
#!/usr/bin/env bash
DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" >/dev/null 2>&1 && pwd )"
export nl=$'\n'
export dquote='"'
export squote="'"
export backtick='`'
export start_code='```'"${nl}"
export start_coq_code='```coq'"${nl}"
export end_code="${nl}"'```'
. "$DIR/github-url.sh"
export BUG_TMP_DIR="$DIR/cwd"
export BUG_FILE="${BUG_TMP_DIR}/bug_01.v"
export TMP_FILE="${BUG_TMP_DIR}/tmp.v"
export TMP_LOG="${BUG_TMP_DIR}/tmp.log"
export FINAL_BUG_FILE="$DIR/bug.v" # must not change, since the deploy/artifact script looks for it
export FINAL_TMP_FILE="$DIR/tmp.v" # must not change, since the deploy/artifact script looks for it
export FINAL_TMP_LOG="$DIR/tmp.log" # must not change, since the deploy/artifact script looks for it
export BUILD_LOG="$DIR/build.log" # must not change, since the deploy/artifact script looks for it
export BUG_LOG="$DIR/bug.log" # must not change, since the deploy/artifact script looks for it
export VERBOSE_BUG_LOG="$DIR/bug.verbose.log" # must not change, since the deploy/artifact script looks for it
export BACKUP_BUG_LOG="$DIR/bug.backup.log"
export METADATA_FILE="$DIR/metadata" # must not change, since the deploy/artifact script looks for it
export FINAL_TMP_FOLDER="$DIR/tmp" # must not change, since the deploy/artifact script looks for it
export CUSTOM_REPLY_COQBOT_FILE="$DIR/custom-reply-coqbot.sh" # must not change, since we run this file from GH actions # file we write to so we can reply after stamping urls
export TIMEDOUT_STAMP_FILE="$DIR/timedout"
export COQBOT_URL="$(cat "$DIR/coqbot.url")"
export COQBOT_RESUME_MINIMIZATION_URL="$(cat "$DIR/coqbot.resume-minimization-url" 2>/dev/null)"
export SURVEY_URL="$(cat "$DIR/coqbot.survey-url")"
export SURVEY_PR_URL_PARAMETER="$(cat "$DIR/coqbot.survey-pr-url-parameter")"
export ISSUE_NUMBER="$(cat "$DIR/coqbot.issue-number")"
export COMPILER="$(cat "$DIR/coqbot.compiler")"
export FAILING_ARTIFACT_URLS="$(echo $(cat "$DIR/coqbot.failing-artifact-urls"))"
export PASSING_ARTIFACT_URLS="$(echo $(cat "$DIR/coqbot.passing-artifact-urls"))"
export COQ_FAILING_SHA="$(echo $(cat "$DIR/coqbot.failing-sha"))"
export COQ_PASSING_SHA="$(echo $(cat "$DIR/coqbot.passing-sha"))"
# this one is tricky, we want to include trailing newlines so we don't
# lose potentially-empty extra arguments at the end, so we do as in
# https://stackoverflow.com/a/40717560/377022
RESUMPTION_ARGS="$(cat "$DIR/coqbot.resumption-args" 2>/dev/null; printf '.\n')"
RESUMPTION_ARGS="${RESUMPTION_ARGS:0:-1}"
export RESUMPTION_ARGS # Only used for communicating with coqbot on minimization resumption
export CI_TARGET="$(cat "$DIR/coqbot.ci-target")"
export CI_BASE_BUILD_DIR="$DIR/builds/coq"
export COQ_CI_BASE_BUILD_DIR="/builds/coq/coq"
export GITHUB_MAX_CHAR_COUNT="65536"
IFS=$'\n' export EXTRA_MINIMIZER_ARGUMENTS=($(cat "$DIR/coqbot.extra-args"))
if [[ "${CI_TARGET}" == "TAKE FROM"* ]]; then
CI_TARGET_FILE="$(printf "%s\n" "${CI_TARGET}" | sed 's/^\s*TAKE FROM //g')"
if [ -f "${CI_TARGET_FILE}" ]; then
export CI_TARGET="$(cd "$DIR" && grep '^Makefile.ci:.*recipe for target.*failed' "${CI_TARGET_FILE}" | tail -1 | sed "s/^Makefile.ci:.*recipe for target '//g; s/' failed\$//g")"
else
export CI_TARGET=""
fi
fi
if [ ! -z "${FAILING_ARTIFACT_URLS}" ] && [ ! -z "${PASSING_ARTIFACT_URLS}" ] && [ ! -z "${COQ_FAILING_SHA}" ] && [ ! -z "${COQ_PASSING_SHA}" ]; then
export RUN_KIND=coqbot-ci
else
export RUN_KIND=coqbot
fi
function backup() {
if [ -f "$1" ]; then
if [ -f "$2" ]; then
backup "$2" "$2.bak"
rm "$2"
fi
cp "$1" "$2"
fi
}
export -f backup
function wrap_file() {
local file="$1"
# coqdep output needs to be pristine for use in coq_makefile;
# coq_makefile errors if -o is given a non-relative path / a path
# to something not in the current directory; coqchk uses -o for
# something other than file output, so we just exclude these three
# files
if [[ "$file" != *.orig ]] && [[ "$file" != *coqdep* ]] && [[ "$file" != *coq_makefile* ]] && [[ "$file" != *coqchk* ]]; then
cat > "$file.new" <<EOF
#!/usr/bin/env bash
DIR="\$( cd "\$( dirname "\${BASH_SOURCE[0]}" )" >/dev/null 2>&1 && pwd )"
args=("\$DIR/$file.orig")
next_is_dir=no
next_is_special=no
next_next_is_special=no
fname=""
for i in "\$@"; do
if [ "\${next_is_dir}" == "yes" ]; then
args+=("\$(readlink -f "\$i")")
next_is_dir=no
next_is_special="\${next_next_is_special}"
next_next_is_special=no
elif [ "\${next_is_special}" == "yes" ]; then
args+=("\$i")
next_is_special="\${next_next_is_special}"
next_next_is_special=no
elif [ "\${next_is_v_file}" == "yes" ] || [[ "\$i" == *".v" ]]; then
fname="\$fname \$i \$(readlink -f "\$i")"
args+=("\$i") # ("\$(readlink -f "\$i")") # we absolutize this later instead of now, to preserve output tests in HB
next_is_v_file=no
else
args+=("\$i")
case "\$i" in
-R|-Q)
next_is_v_file=no
next_is_dir=yes
next_is_special=no
next_next_is_special=yes
;;
-I|-include|-coqlib|-exlcude-dir|-load-ml-object|-load-ml-source|-init-file|-dump-glob|-o|-time-file)
next_is_v_file=no
next_is_dir=yes
next_is_special=no
next_next_is_special=no
;;
-load-vernac-source|-l|-load-vernac-source-verbose|-lv)
next_is_v_file=yes
next_is_dir=no
next_is_special=no
next_next_is_special=no
;;
-arg|-compat|-w|-color|-diffs|-mangle-names|-set|-unset|-top|-topfile|-bytecode-compiler|-native-compiler)
next_is_special=yes
;;
-schedule-vio2vo|-schedule-vio-checking)
next_is_special=yes
next_next_is_special=yes
;;
*)
;;
esac
fi
done
debug_prefix="\$(mktemp --tmpdir tmp-coqbot-minimizer.XXXXXXXXXX)"
printf "%s" "\$0" > "\${debug_prefix}"
printf "%s" "\$COQPATH" > "\${debug_prefix}.coqpath"
printf "%s" "\$(pwd)" > "\${debug_prefix}.pwd"
printf "%q " "\${args[@]}" > "\${debug_prefix}.exec"
# extra, not strictly needed
>&2 printf "MINIMIZER_DEBUG_EXTRA: coqc: %s\n" "\$0"
>&2 printf "MINIMIZER_DEBUG_EXTRA: coqpath: %s\n" "\$(cat "\${debug_prefix}.coqpath")"
>&2 printf "MINIMIZER_DEBUG_EXTRA: pwd: PWD=%s\n" "\$(cat "\${debug_prefix}.pwd")"
>&2 printf "MINIMIZER_DEBUG_EXTRA: exec: %s\n" "\$(cat "\${debug_prefix}.exec")"
# the two important lines
>&2 printf "MINIMIZER_DEBUG: info: %s\n" "\${debug_prefix}"
>&2 printf "MINIMIZER_DEBUG: files: %s\n" "\${fname}"
exec "\${args[@]}"
EOF
chmod +x "$file.new"
# if file.new is not the same as file
if ! cmp -s "$file" "$file.new" || [ ! -f "$file.orig" ]; then
backup "$file" "$file.orig"
mv "$file.new" "$file"
else
rm "$file.new"
fi
fi
}
export -f wrap_file
# we will unwrap files before running opam commands, so that if
# we reinstall coq, we don't get tripped up when rewrapping
function unwrap_file() {
local file="$1"
# we only unwrap files that we have wrapped
if [[ "$file" != *.orig ]] && [[ "$file" != *coqdep* ]] && [[ "$file" != *coq_makefile* ]] && [[ "$file" != *coqchk* ]]; then
if [ -f "$file.orig" ]; then
mv -f "$file.orig" "$file" || exit $?
fi
fi
}
export -f unwrap_file
function wrap_opam() {
local file="$(which opam)"
if [ ! -f "$file.orig" ]; then
sudo mv "$file" "$file.orig" || exit $?
sudo touch "$file"
sudo chmod a+rw "$file"
cat > "$file" <<EOF
#!/usr/bin/env bash
DIR="\$( cd "\$( dirname "\${BASH_SOURCE[0]}" )" >/dev/null 2>&1 && pwd )"
source "$DIR/coqbot-config.sh"
"$file.orig" "\$@" || exit \$?
eval "\$("$file.orig" env)"
>&2 wrap_opam_and_files $@
EOF
sudo chmod --reference="$file.orig" "$file"
fi
}
export -f wrap_opam
function wrap_opam_and_files() {
printf '::group::opam wrap files\n'
printf 'wrapping %s\n' "$(which opam)"
wrap_opam "$@"
for i in "$@"; do
printf "attempting to wrap %s\n" "$i"
if command -v "$i" >/dev/null; then
printf 'wrapping %s\n' "$(which "$i")"
pushd "$(dirname "$(which "$i")")" >/dev/null
wrap_file "$i"
popd >/dev/null
fi
done
printf '::endgroup::\n'
}
export -f wrap_opam_and_files
function unwrap_files() {
printf '::group::unwrap files\n'
# printf 'unwrapping %s\n' "$(which opam)"
# unwrap_opam "$@"
for i in "$@"; do
printf "attempting to unwrap %s\n" "$i"
if command -v "$i" >/dev/null; then
printf 'unwrapping %s\n' "$(which "$i")"
pushd "$(dirname "$(which "$i")")" >/dev/null
unwrap_file "$i"
popd >/dev/null
fi
done
printf '::endgroup::\n'
}
export -f unwrap_files
# if_wrap_with_url file prefix_if_exists description_if_exists postfix_if_exists text_if_does_not_exist
# if file.url exists, returns an ${prefix_if_exists}<a href=$(cat $file.url)>${description_if_exists}</a>${postfix_if_exists}
# if file.api.url exists, api-href=... is also included in the a tag
# if file.url does not exist, just returns ${text_if_does_not_exist}
function if_wrap_with_url() {
file="$1"
prefix_if_exists="$2"
description_if_exists="$3"
postfix_if_exists="$4"
text_if_does_not_exist="$5"
if [ -f "${file}.url" ] && [ ! -z "$(printf -- $(cat "${file}.url"))" ]; then
printf '%s<a href="%s"' "${prefix_if_exists}" "$(printf -- $(cat "${file}.url"))"
if [ -f "${file}.api.url" ] && [ ! -z "$(printf -- $(cat "${file}.api.url"))" ]; then
# purely for eventual use by coqbot
printf ' api-href="%s"' "$(printf -- $(cat "${file}.api.url"))"
fi
printf '>%s</a>%s' "${description_if_exists}" "${postfix_if_exists}"
else
printf '%s' "${text_if_does_not_exist}"
fi
}
export -f if_wrap_with_url
# maybe_wrap_with_url description file
# if file.url exists, returns an <a href=$(cat $file.url)>$description</a>
# if file.api.url exists, api-href=... is also included in the a tag
# if file.url does not exist, just returns description
function maybe_wrap_with_url() {
description="$1"
file="$2"
if_wrap_with_url "$file" "" "$description" "" "$description"
}
export -f maybe_wrap_with_url
# print_file max_len title start_code filepath end_code
function print_file() {
head_tail="$1"
max_file_size="$2"
title="$3"
extra_title_unless_truncated="$4"
start_code="$5"
fname="$6"
end_code="$7"
filesize="$(stat -c%s "${fname}")"
head_tail_separator="${nl}${nl}[...]${nl}${nl}"
if (( filesize > max_file_size )); then
case "${head_tail}" in
head-tail)
max_filesize="$(( ( ${max_file_size} - ${#head_tail_separator} ) / 2 ))"
contents="$(head -c ${max_file_size} "${fname}")${head_tail_separator}$(tail -c ${max_file_size} "${fname}")"
truncated="first and last "
;;
head)
contents="$(head -c ${max_file_size} "${fname}")"
truncated=""
;;
tail)
contents="$(tail -c ${max_file_size} "${fname}")"
truncated="last "
;;
*)
contents="Invalid head_tail='${head_tail}'$(printf '\n'; head -c ${max_file_size} "${fname}")"
truncated=""
;;
esac
filesize_pretty="$(numfmt --to=iec-i --suffix=B "${filesize}")"
max_file_size_pretty="$(numfmt --to=iec-i --suffix=B "${max_file_size}")"
fname_code="<code>$(realpath --relative-to "$DIR" "${fname}")</code>"
title="${title} (truncated to ${truncated}${max_file_size_pretty}; full ${filesize_pretty} file on <a href=\"${GITHUB_WORKFLOW_URL}\">GitHub Actions Artifacts</a> under $(maybe_wrap_with_url "${fname_code}" "${fname}"))"
else
title="${title}${extra_title_unless_truncated}"
contents="$(cat "${fname}")"
fi
printf "\n\n<details><summary>%s</summary>\n\n" "${title}"
printf "%s" "${start_code}"
printf "%s" "${contents}"
printf "%s" "${end_code}"
printf "\n</details>"
}
export -f print_file