-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathapiLib.sml
346 lines (298 loc) · 11.3 KB
/
apiLib.sml
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
337
338
339
340
341
342
343
344
345
346
(*
The API that the server and worker agree on.
Reference:
GET methods:
waiting:
returns space-separated list of ids of waiting jobs
job id:
returns information on job <id>
including:
- commits (including pull request integration, if any)
- (worker) name and time started (if any)
- output so far
POST methods:
refresh:
update the queues according to the current state on GitHub
returns "Refreshed"
claim id name:
worker <name> claims job <id>.
name=<name> is the POST data.
returns "Claimed"
fails (409) if <id> is not currently waiting
append id line:
record <line> as additional output for job <id> with a timestamp.
line=<line> is the POST data.
returns "Appended"
fails (409) if <id> is not currently running
log id data:
append <data> as additional output for job <id>.
<data> is the POST data.
returns "Logged"
fails (409) if <id> is not currently running
upload id name data:
save a file called <name> containing <data> as
an artefact generated by job <id>.
name=<name>&<data> is the POST data, and <name> must not contain &.
returns "Uploaded"
fails (409) if a file called <name> already exists for <id>
finish id:
mark job <id> as finished
returns "Finished"
sends email with output
fails (409) if <id> is not currently running
abort id:
mark job <id> as aborted
returns "Aborted"
fails (409) if <id> is not currently finished
release id:
make a GitHub release for job <id>
returns "Released"
fails (409) if <id> is not releasable
all failures return text starting with "Error:"
Jobs move (right only) between these states:
waiting, running, finished, aborted
waiting = ready to be run, waiting for a worker
running = claimed to be running by a worker
finished = finished either with success or failure
aborted = the worker did not finish properly
When the waiting queue is refreshed, the commits of running or finished jobs are
not considered to need running again, whereas the commits of aborted jobs are
(as long as they are still the latest commits).
*)
use "utilLib.sml";
use "config.sml";
open Config;
structure apiLib = struct
open utilLib
type id = int
type worker_name = string
type line = string
fun check_id f id =
0 <= id andalso Int.toString id = f
val server = String.concat[host,base_url]
val cakeml_token = until_space (file_to_string "cakeml-token")
handle IO.Io _ => (
TextIO.output(TextIO.stdErr,"Could not find cakeml-token. Try sha1sum worker.sml >cakeml-token.\n");
OS.Process.exit OS.Process.failure)
datatype get_api = Waiting | Job of id
datatype post_api =
Refresh
| Claim of id * worker_name
| Append of id * line (* not including newline *)
| Log of id * string * int
| Upload of id * string * int
| Finish of id
| Abort of id
| Release of id
datatype api = G of get_api | P of post_api
fun post_response Refresh = "Refreshed\n"
| post_response (Claim _) = "Claimed\n"
| post_response (Append _) = "Appended\n"
| post_response (Finish _) = "Finished\n"
| post_response (Abort _) = "Aborted\n"
| post_response (Log _) = "Logged\n"
| post_response (Upload _) = "Uploaded\n"
| post_response (Release _) = "Released\n"
fun formdata_decode s =
let
fun loop ss acc =
let
val (chunk,ss) = Substring.splitl (not o equal #"%") ss
in
if Substring.isEmpty ss then
Substring.concat(List.rev(chunk::acc))
else
let
val (ns,ss) = Substring.splitAt(Substring.triml 1 ss,2)
val n = #1 (Option.valOf (Int.scan StringCvt.HEX Substring.getc ns))
val c = Substring.full (String.str (Char.chr n))
in
loop ss (c::chunk::acc)
end
end
val plus2space = CharVector.map (fn #"+" => #" " | c => c)
in
loop (Substring.full (plus2space s)) []
handle e => (TextIO.output(TextIO.stdErr,String.concat["form-data decode failed on ",s,"\n",exnMessage e,"\n"]); raise e)
end
fun api_to_string (G Waiting) = "/waiting"
| api_to_string (P Refresh) = "/refresh"
| api_to_string (P (Log (id,_,_))) = String.concat["/log/",Int.toString id]
| api_to_string (P (Upload (id,_,_))) = String.concat["/upload/",Int.toString id]
| api_to_string (G (Job id)) = String.concat["/job/",Int.toString id]
| api_to_string (P (Claim (id,_))) = String.concat["/claim/",Int.toString id]
| api_to_string (P (Append (id,_))) = String.concat["/append/",Int.toString id]
| api_to_string (P (Finish id)) = String.concat["/finish/",Int.toString id]
| api_to_string (P (Abort id)) = String.concat["/abort/",Int.toString id]
| api_to_string (P (Release id)) = String.concat["/release/",Int.toString id]
fun post_curl_args (Append (_,line)) = ["--data-urlencode",String.concat["line=",line]]
| post_curl_args (Claim (_,name)) = ["--data-urlencode",String.concat["name=",name]]
| post_curl_args (Log (_,file,_)) = ["--data-binary",String.concat["@",file]]
| post_curl_args (Upload (_,file,_)) = ["--data",String.concat["name=",List.last(#arcs(OS.Path.fromString file))],"--data-binary",String.concat["@",file]]
| post_curl_args (Finish _) = ["--data",""]
| post_curl_args (Abort _) = ["--data",""]
| post_curl_args (Refresh) = ["--data",""]
| post_curl_args (Release _) = ["--data",""]
fun api_curl_args (G _) = []
| api_curl_args (P p) = post_curl_args p
fun id_from_string n =
case Int.fromString n of NONE => NONE
| SOME id => if check_id n id then SOME id else NONE
fun read_query prefix len =
case String.tokens (equal #"&") (TextIO.inputN(TextIO.stdIn,len))
of [s] =>
if String.isPrefix (String.concat[prefix,"="]) s then
SOME (formdata_decode (String.extract(s,String.size prefix + 1,NONE)))
else NONE
| _ => NONE
fun get_from_string s =
if s = "/waiting" then SOME Waiting else
case String.tokens (equal #"/") s of
["job",n] => Option.map Job (id_from_string n)
| _ => NONE
fun read_name l =
let
fun assert b = if b then () else raise Option
val prefix = "name="
val () = assert (String.size prefix <= l)
val () = assert (TextIO.inputN(TextIO.stdIn,String.size prefix) = prefix)
fun loop (acc,l) =
let
val c = Option.valOf (TextIO.input1 TextIO.stdIn)
in
if c = #"&"
then (String.implode(List.rev acc),l-1)
else loop (c::acc,l-1)
end
in SOME (loop ([],l)) end
handle Option => NONE
fun post_from_string s len =
if s = "/refresh" then SOME Refresh
else (case String.tokens (equal #"/") s of
["claim",n] => Option.mapPartial
(fn id => Option.map (fn s => Claim(id,s))
(Option.mapPartial (read_query "name") len))
(id_from_string n)
| ["append",n] => Option.mapPartial
(fn id => Option.map (fn s => Append(id,s))
(Option.mapPartial (read_query "line") len))
(id_from_string n)
| ["log",n] => Option.mapPartial
(fn id => Option.map (fn l => Log(id,"",l)) len)
(id_from_string n)
| ["upload",n] => Option.mapPartial
(fn id => Option.map (fn (name,l) => Upload(id,name,l))
(Option.mapPartial read_name len))
(id_from_string n)
| ["finish",n] => Option.map Finish (id_from_string n)
| ["abort",n] => Option.map Abort (id_from_string n)
| ["release",n] => Option.map Release (id_from_string n)
| _ => NONE)
type bare_pr = { head_sha : string, base_sha : string }
datatype bare_integration = Bbr of string | Bpr of bare_pr
type bare_snapshot = { bcml : bare_integration, bhol : string }
fun read_bare_snapshot inp =
let
fun read_line () = Option.valOf (TextIO.inputLine inp)
val head_sha = extract_prefix_trimr "CakeML: " (read_line())
val _ = read_line ()
val line = read_line ()
val (line,base_sha) =
if String.isPrefix "#" line then
let
val line = read_line ()
val _ = read_line ()
in (read_line(), SOME (extract_prefix_trimr "Merging into: " line)) end
else (line, NONE)
val hol_sha = extract_prefix_trimr "HOL: " line
in
{ bcml = case base_sha
of NONE => Bbr head_sha
| SOME base_sha => Bpr { head_sha = head_sha, base_sha = base_sha }
, bhol = hol_sha }
end
fun read_job_pr inp =
let
fun read_line () = Option.valOf (TextIO.inputLine inp)
val _ = read_line () (* CakeML *)
val _ = read_line () (* msg *)
val line = read_line ()
in
if String.isPrefix "#" line then
SOME (extract_word line)
else NONE
end
fun read_job_worker inp =
let
fun loop acc =
case TextIO.inputLine inp of
NONE => acc
| SOME line => if String.isPrefix "Machine: " line
then extract_prefix_trimr "Machine: " line
else loop acc
in loop "No worker"
end
fun read_secs timing =
let
val secs_millisecs = String.tokens (equal #".") timing
val whole_secs = List.nth(secs_millisecs,0)
in
if List.all Char.isDigit (String.explode whole_secs)
then Option.valOf(Int.fromString whole_secs)
else (* TODO: only for supporting legacy %E format
could just update the files and remove this *)
let val ls = String.tokens (equal #":") whole_secs
in
Option.valOf(Int.fromString(List.nth(ls,0))) * 60 * 60 +
Option.valOf(Int.fromString(List.nth(ls,1))) * 60 +
Option.valOf(Int.fromString(List.nth(ls,2)))
handle Subscript =>
Option.valOf(Int.fromString(List.nth(ls,0))) * 60 +
Option.valOf(Int.fromString(List.nth(ls,1)))
end
end
fun read_total_time dir inp =
let
fun loop () =
case TextIO.inputLine inp of NONE => NONE
| SOME line =>
let
val (_,rest) = extract_word line
val prefix = " Resuming "
val resuming = Substring.isPrefix prefix rest
val prefix = " Finished "
val finished = Substring.isPrefix prefix rest
val rest = Substring.triml (String.size prefix) rest
val (dir',rest) = Substring.splitl (not o Char.isSpace) rest
val dirs_match = Substring.isPrefix dir dir' andalso
String.size dir = Substring.size dir'
val rest = Substring.dropl Char.isSpace rest
val (timing,_) = Substring.splitl (not o Char.isSpace) rest
in
if dirs_match then
if finished then
SOME (read_secs (Substring.string timing))
handle Option => loop () | Subscript => loop ()
else if resuming then NONE
else loop ()
else loop ()
end
in loop () end
val max_dir_length = 50
datatype status = Pending | Success | Failure | Aborted
fun status_to_string Pending = "Pending"
| status_to_string Success = "Success"
| status_to_string Failure = "Failure"
| status_to_string Aborted = "Aborted"
fun read_status inp =
let
fun loop () =
case TextIO.inputLine inp of NONE => Aborted
| SOME line =>
if String.isSubstring "FAILED" line
then Failure
else if String.isSubstring "SUCCESS" line
then Success
else loop ()
in loop () end
end