-
Notifications
You must be signed in to change notification settings - Fork 3
/
request_build.py
275 lines (215 loc) · 8.69 KB
/
request_build.py
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
#!/usr/bin/env python3
#
# start a package build via backend API
#
import contextlib
import fcntl
import json
import logging
import logging.handlers
import os
import re
import sqlite3
import time
import urllib.error
import urllib.request
import carpetbag
import gh_token
import appveyor_token
# subclass TimedRotatingFileHandler with open umask
class SharedTimedRotatingFileHandler(logging.handlers.TimedRotatingFileHandler):
def _open(self):
old_umask = os.umask(0o000)
rtv = logging.handlers.RotatingFileHandler._open(self)
os.umask(old_umask)
return rtv
rfh = SharedTimedRotatingFileHandler('/sourceware/cygwin-staging/logs/build-request.log', backupCount=48, when='midnight')
rfh.setFormatter(logging.Formatter('%(asctime)s - %(levelname)-8s - %(message)s'))
rfh.setLevel(logging.DEBUG)
logging.getLogger().addHandler(rfh)
logging.getLogger().setLevel(logging.NOTSET)
@contextlib.contextmanager
def locked():
old_umask = os.umask(0o000)
lockfile = open('/tmp/scallywag.request_build.lock', 'w+')
os.umask(old_umask)
fcntl.flock(lockfile.fileno(), fcntl.LOCK_EX)
logging.info("acquired request_build lock")
try:
yield lockfile
finally:
logging.info("releasing request_build lock")
fcntl.flock(lockfile.fileno(), fcntl.LOCK_UN)
lockfile.close()
def _appveyor_build_request(package, maintainer, commit, reference, default_tokens, buildnumber):
slug = 'scallywag'
account, token = appveyor_token.fetch_token()
data = {
"accountName": account,
"projectSlug": slug,
"branch": "master",
"environmentVariables": {
"BUILDNUMBER": buildnumber,
"PACKAGE": package,
"MAINTAINER": maintainer,
"COMMIT": commit,
"REFERENCE": reference,
"DEFAULT_TOKENS": default_tokens,
}
}
req = urllib.request.Request('https://ci.appveyor.com/api/builds')
req.add_header('Content-Type', 'application/json')
req.add_header('Accept', 'application/json')
req.add_header('Authorization', 'Bearer ' + token)
try:
response = urllib.request.urlopen(req, json.dumps(data).encode('utf-8'))
except urllib.error.URLError as e:
response = e
status = response.getcode()
if status != 200:
print('scallywag: AppVeyor REST API failed status %s' % (status))
return -1
j = json.loads(response.read().decode('utf-8'))
return j['buildId']
def _github_most_recent_wfr_id():
data = {
"event": "repository_dispatch",
"per_page": 1
}
qs = urllib.parse.urlencode(data)
req = urllib.request.Request('https://api.github.com/repos/cygwin/scallywag/actions/runs' + '?' + qs)
req.add_header('Accept', 'application/vnd.github.v3+json')
req.add_header('Authorization', 'Bearer ' + gh_token.fetch_iat())
try:
response = urllib.request.urlopen(req)
except urllib.error.URLError as e:
response = e
status = response.getcode()
logging.info("runs REST API status %s" % status)
if status != 200:
print('scallywag: GitHub REST API failed status %s' % (status))
return 0, None
resp = response.read().decode('utf-8')
logging.info("runs REST API response %s" % resp)
j = json.loads(resp)
wfr = j['workflow_runs']
if len(wfr) <= 0:
logging.info("no most recent wrf_id available")
return 0, None
logging.info("most recent wrf_id %s" % wfr[0]['id'])
return wfr[0]['id'], wfr[0]['html_url']
def _github_workflow_trigger(package, maintainer, commit, reference, default_tokens, buildnumber):
for _i in range(1, 60):
prev_wfr_id, _ = _github_most_recent_wfr_id()
if prev_wfr_id != 0:
break
logging.info("waiting before retry")
time.sleep(1)
else:
logging.info("timeout waiting for GitHub to report previous wfr_id")
print('scallywag: timeout waiting for GitHub to report previous wfr_id')
# strip out any over-quoting in the token, as it's harmful to passing the
# client_payload into scallywag via the command line
default_tokens = re.sub(r'[\'"]', r'', default_tokens)
data = {
"event_type": "(%s) %s" % (buildnumber, package), # 'display_title', appears as the run name in UI
"client_payload": {
"BUILDNUMBER": buildnumber,
"PACKAGE": package,
"MAINTAINER": maintainer,
"COMMIT": commit,
"REFERENCE": reference,
"DEFAULT_TOKENS": default_tokens,
}
}
req = urllib.request.Request('https://api.github.com/repos/cygwin/scallywag/dispatches')
req.add_header('Accept', 'application/vnd.github.v3+json')
req.add_header('Authorization', 'Bearer ' + gh_token.fetch_iat())
try:
response = urllib.request.urlopen(req, data=json.dumps(data).encode('utf-8'))
except urllib.error.URLError as e:
response = e
status = response.getcode()
if status != 204:
print('scallywag: GitHub REST API failed status %s' % (status))
return -1, None
# response has no content, and doesn't give an id for the workflow that
# we've just requested. all we can do is poll the workflow runs list and
# guess that the most recent one is ours.
#
# (it seems that it takes a little while for the requested run to appear in
# the workflow run list, with status 'queued', and then some time later it
# changes to status 'in_progress'.)
#
# and since there may exist other runs with status 'in_progress', the only
# half-way reliable way to do this is to poll until a new wfr id appears...
#
# see https://github.community/t/repository-dispatch-response/17950
for _i in range(1, 60):
wfr_id, buildurl = _github_most_recent_wfr_id()
if wfr_id != prev_wfr_id:
return wfr_id, buildurl
logging.info("waiting before retry")
time.sleep(1)
logging.info("timeout waiting for GitHub to assign a wfr_id")
print('scallywag: timeout waiting for GitHub to assign a wfr_id')
return 0, None
def request_build(commit, reference, package, maintainer, tokens=''):
default_tokens = ''
try:
with open(os.path.join('/sourceware/cygwin-staging/home', maintainer, '!scallywag')) as f:
default_tokens = ''.join([l.strip() for l in f.readlines()])
except FileNotFoundError:
pass
if tokens:
default_tokens = default_tokens + ' ' + tokens
if 'disable' in default_tokens:
print('scallywag: disabled by you')
return
if 'nobuild' in default_tokens:
print('scallywag: not building due to nobuild')
return
# record job as requested and generate buildnumber
now = time.time()
with sqlite3.connect(carpetbag.dbfile) as conn:
cursor = conn.execute('INSERT INTO jobs (srcpkg, hash, ref, user, status, timestamp, tokens) VALUES (?, ?, ?, ?, ?, ?, ?)',
(package, commit, reference, maintainer, 'requested', now, tokens))
buildnumber = cursor.lastrowid
conn.commit()
conn.close()
# request job
if 'appveyor' in default_tokens:
backend = 'appveyor'
bbid = _appveyor_build_request(package, maintainer, commit, reference, default_tokens, buildnumber)
buildurl = None
else:
backend = 'github'
with locked():
bbid, buildurl = _github_workflow_trigger(package, maintainer, commit, reference, default_tokens, buildnumber)
# an error occurred requesting the job
if bbid < 0:
return
print('scallywag: build {0} queued on {1}'.format(buildnumber, backend))
print('scallywag: https://cygwin.com/cgi-bin2/jobs.cgi?id={0}'.format(buildnumber))
# record job as pending
with sqlite3.connect(carpetbag.dbfile) as conn:
conn.execute('UPDATE jobs SET status = ?, logurl = ?, backend = ?, backend_id = ? WHERE id = ?',
('pending', buildurl, backend, bbid, buildnumber))
conn.commit()
conn.close()
def _github_workflow_cancel(wfr_id):
req = urllib.request.Request('https://api.github.com/repos/cygwin/scallywag/actions/runs/{}/cancel'.format(wfr_id), method='POST')
req.add_header('Accept', 'application/vnd.github.v3+json')
req.add_header('Authorization', 'Bearer ' + gh_token.fetch_iat())
try:
response = urllib.request.urlopen(req)
except urllib.error.URLError as e:
response = e
status = response.getcode()
if status != 202:
print('scallywag: GitHub REST API failed status %s' % (status))
def cancel_build(backend, bbid):
if backend == 'github':
_github_workflow_cancel(bbid)
else:
print('job cancellation not implemented for backend %s' % backend)