summaryrefslogtreecommitdiffstats
path: root/meta/classes/tinderclient.bbclass
diff options
context:
space:
mode:
Diffstat (limited to 'meta/classes/tinderclient.bbclass')
-rw-r--r--meta/classes/tinderclient.bbclass332
1 files changed, 332 insertions, 0 deletions
diff --git a/meta/classes/tinderclient.bbclass b/meta/classes/tinderclient.bbclass
new file mode 100644
index 0000000000..6e10d0f34b
--- /dev/null
+++ b/meta/classes/tinderclient.bbclass
@@ -0,0 +1,332 @@
1def tinder_http_post(server, selector, content_type, body):
2 import httplib
3 # now post it
4 for i in range(0,5):
5 try:
6 h = httplib.HTTP(server)
7 h.putrequest('POST', selector)
8 h.putheader('content-type', content_type)
9 h.putheader('content-length', str(len(body)))
10 h.endheaders()
11 h.send(body)
12 errcode, errmsg, headers = h.getreply()
13 #print errcode, errmsg, headers
14 return (errcode,errmsg, headers, h.file)
15 except:
16 # try again
17 pass
18
19def tinder_form_data(bound, dict, log):
20 output = []
21 #br
22 # for each key in the dictionary
23 for name in dict:
24 output.append( "--" + bound )
25 output.append( 'Content-Disposition: form-data; name="%s"' % name )
26 output.append( "" )
27 output.append( dict[name] )
28 if log:
29 output.append( "--" + bound )
30 output.append( 'Content-Disposition: form-data; name="log"; filename="log.txt"' )
31 output.append( '' )
32 output.append( log )
33 output.append( '--' + bound + '--' )
34 output.append( '' )
35
36 return "\r\n".join(output)
37
38def tinder_time_string():
39 """
40 Return the time as GMT
41 """
42 return ""
43
44def tinder_format_http_post(d,status,log):
45 """
46 Format the Tinderbox HTTP post with the data needed
47 for the tinderbox to be happy.
48 """
49
50 from bb import data, build
51 import os,random
52
53 # the variables we will need to send on this form post
54 variables = {
55 "tree" : data.getVar('TINDER_TREE', d, True),
56 "machine_name" : data.getVar('TINDER_MACHINE', d, True),
57 "os" : os.uname()[0],
58 "os_version" : os.uname()[2],
59 "compiler" : "gcc",
60 "clobber" : data.getVar('TINDER_CLOBBER', d, True)
61 }
62
63 # optionally add the status
64 if status:
65 variables["status"] = str(status)
66
67 # try to load the machine id
68 # we only need on build_status.pl but sending it
69 # always does not hurt
70 try:
71 f = file(data.getVar('TMPDIR',d,True)+'/tinder-machine.id', 'r')
72 id = f.read()
73 variables['machine_id'] = id
74 except:
75 pass
76
77 # the boundary we will need
78 boundary = "----------------------------------%d" % int(random.random()*1000000000000)
79
80 # now format the body
81 body = tinder_form_data( boundary, variables, log )
82
83 return ("multipart/form-data; boundary=%s" % boundary),body
84
85
86def tinder_build_start(d):
87 """
88 Inform the tinderbox that a build is starting. We do this
89 by posting our name and tree to the build_start.pl script
90 on the server.
91 """
92 from bb import data
93
94 # get the body and type
95 content_type, body = tinder_format_http_post(d,None,None)
96 server = data.getVar('TINDER_HOST', d, True )
97 url = data.getVar('TINDER_URL', d, True )
98
99 selector = url + "/xml/build_start.pl"
100
101 #print "selector %s and url %s" % (selector, url)
102
103 # now post it
104 errcode, errmsg, headers, h_file = tinder_http_post(server,selector,content_type, body)
105 #print errcode, errmsg, headers
106 report = h_file.read()
107
108 # now let us find the machine id that was assigned to us
109 search = "<machine id='"
110 report = report[report.find(search)+len(search):]
111 report = report[0:report.find("'")]
112
113 import bb
114 bb.note("Machine ID assigned by tinderbox: %s" % report )
115
116 # now we will need to save the machine number
117 # we will override any previous numbers
118 f = file(data.getVar('TMPDIR', d, True)+"/tinder-machine.id", 'w')
119 f.write(report)
120
121
122def tinder_send_http(d, status, log):
123 """
124 Send this log as build status
125 """
126 from bb import data
127
128
129 # get the body and type
130 content_type, body = tinder_format_http_post(d,status,log)
131 server = data.getVar('TINDER_HOST', d, True )
132 url = data.getVar('TINDER_URL', d, True )
133
134 selector = url + "/xml/build_status.pl"
135
136 # now post it
137 errcode, errmsg, headers, h_file = tinder_http_post(server,selector,content_type, body)
138 #print errcode, errmsg, headers
139 #print h.file.read()
140
141
142def tinder_print_info(d):
143 """
144 Print the TinderBox Info
145 Including informations of the BaseSystem and the Tree
146 we use.
147 """
148
149 from bb import data
150 import os
151 # get the local vars
152
153 time = tinder_time_string()
154 ops = os.uname()[0]
155 version = os.uname()[2]
156 url = data.getVar( 'TINDER_URL' , d, True )
157 tree = data.getVar( 'TINDER_TREE', d, True )
158 branch = data.getVar( 'TINDER_BRANCH', d, True )
159 srcdate = data.getVar( 'SRCDATE', d, True )
160 machine = data.getVar( 'MACHINE', d, True )
161 distro = data.getVar( 'DISTRO', d, True )
162 bbfiles = data.getVar( 'BBFILES', d, True )
163 tarch = data.getVar( 'TARGET_ARCH', d, True )
164 fpu = data.getVar( 'TARGET_FPU', d, True )
165 oerev = data.getVar( 'OE_REVISION', d, True ) or "unknown"
166
167 # there is a bug with tipple quoted strings
168 # i will work around but will fix the original
169 # bug as well
170 output = []
171 output.append("== Tinderbox Info" )
172 output.append("Time: %(time)s" )
173 output.append("OS: %(ops)s" )
174 output.append("%(version)s" )
175 output.append("Compiler: gcc" )
176 output.append("Tinderbox Client: 0.1" )
177 output.append("Tinderbox Client Last Modified: yesterday" )
178 output.append("Tinderbox Protocol: 0.1" )
179 output.append("URL: %(url)s" )
180 output.append("Tree: %(tree)s" )
181 output.append("Config:" )
182 output.append("branch = '%(branch)s'" )
183 output.append("TARGET_ARCH = '%(tarch)s'" )
184 output.append("TARGET_FPU = '%(fpu)s'" )
185 output.append("SRCDATE = '%(srcdate)s'" )
186 output.append("MACHINE = '%(machine)s'" )
187 output.append("DISTRO = '%(distro)s'" )
188 output.append("BBFILES = '%(bbfiles)s'" )
189 output.append("OEREV = '%(oerev)s'" )
190 output.append("== End Tinderbox Client Info" )
191
192 # now create the real output
193 return "\n".join(output) % vars()
194
195
196def tinder_print_env():
197 """
198 Print the environment variables of this build
199 """
200 from bb import data
201 import os
202
203 time_start = tinder_time_string()
204 time_end = tinder_time_string()
205
206 # build the environment
207 env = ""
208 for var in os.environ:
209 env += "%s=%s\n" % (var, os.environ[var])
210
211 output = []
212 output.append( "---> TINDERBOX RUNNING env %(time_start)s" )
213 output.append( env )
214 output.append( "<--- TINDERBOX FINISHED (SUCCESS) %(time_end)s" )
215
216 return "\n".join(output) % vars()
217
218def tinder_tinder_start(d, event):
219 """
220 PRINT the configuration of this build
221 """
222
223 time_start = tinder_time_string()
224 config = tinder_print_info(d)
225 #env = tinder_print_env()
226 time_end = tinder_time_string()
227 packages = " ".join( event.getPkgs() )
228
229 output = []
230 output.append( "---> TINDERBOX PRINTING CONFIGURATION %(time_start)s" )
231 output.append( config )
232 #output.append( env )
233 output.append( "<--- TINDERBOX FINISHED PRINTING CONFIGURATION %(time_end)s" )
234 output.append( "---> TINDERBOX BUILDING '%(packages)s'" )
235 output.append( "<--- TINDERBOX STARTING BUILD NOW" )
236
237 output.append( "" )
238
239 return "\n".join(output) % vars()
240
241def tinder_do_tinder_report(event):
242 """
243 Report to the tinderbox:
244 On the BuildStart we will inform the box directly
245 On the other events we will write to the TINDER_LOG and
246 when the Task is finished we will send the report.
247
248 The above is not yet fully implemented. Currently we send
249 information immediately. The caching/queuing needs to be
250 implemented. Also sending more or less information is not
251 implemented yet.
252 """
253 from bb.event import getName
254 from bb import data, mkdirhier, build
255 import os, glob
256
257 # variables
258 name = getName(event)
259 log = ""
260 status = 1
261 #print asd
262 # Check what we need to do Build* shows we start or are done
263 if name == "BuildStarted":
264 tinder_build_start(event.data)
265 log = tinder_tinder_start(event.data,event)
266
267 try:
268 # truncate the tinder log file
269 f = file(data.getVar('TINDER_LOG', event.data, True), 'rw+')
270 f.truncate(0)
271 f.close()
272 except IOError:
273 pass
274
275 # Append the Task-Log (compile,configure...) to the log file
276 # we will send to the server
277 if name == "TaskSucceeded" or name == "TaskFailed":
278 log_file = glob.glob("%s/log.%s.*" % (data.getVar('T', event.data, True), event.task))
279
280 if len(log_file) != 0:
281 to_file = data.getVar('TINDER_LOG', event.data, True)
282 log += "".join(open(log_file[0], 'r').readlines())
283
284 # set the right 'HEADER'/Summary for the TinderBox
285 if name == "TaskStarted":
286 log += "---> TINDERBOX Task %s started\n" % event.task
287 elif name == "TaskSucceeded":
288 log += "<--- TINDERBOX Task %s done (SUCCESS)\n" % event.task
289 elif name == "TaskFailed":
290 log += "<--- TINDERBOX Task %s failed (FAILURE)\n" % event.task
291 elif name == "PkgStarted":
292 log += "---> TINDERBOX Package %s started\n" % data.getVar('P', event.data, True)
293 elif name == "PkgSucceeded":
294 log += "<--- TINDERBOX Package %s done (SUCCESS)\n" % data.getVar('P', event.data, True)
295 elif name == "PkgFailed":
296 build.exec_task('do_clean', event.data)
297 log += "<--- TINDERBOX Package %s failed (FAILURE)\n" % data.getVar('P', event.data, True)
298 status = 200
299 elif name == "BuildCompleted":
300 log += "Build Completed\n"
301 status = 100
302 elif name == "MultipleProviders":
303 log += "---> TINDERBOX Multiple Providers\n"
304 log += "multiple providers are available (%s);\n" % ", ".join(event.getCandidates())
305 log += "consider defining PREFERRED_PROVIDER_%s\n" % event.getItem()
306 log += "is runtime: %d\n" % event.isRuntime()
307 log += "<--- TINDERBOX Multiple Providers\n"
308 elif name == "NoProvider":
309 log += "Error: No Provider for: %s\n" % event.getItem()
310 log += "Error:Was Runtime: %d\n" % event.isRuntime()
311 status = 200
312
313 # now post the log
314 if len(log) == 0:
315 return
316
317 # for now we will use the http post method as it is the only one
318 log_post_method = tinder_send_http
319 log_post_method(event.data, status, log)
320
321
322# we want to be an event handler
323addhandler tinderclient_eventhandler
324python tinderclient_eventhandler() {
325 from bb import note, error, data
326 from bb.event import NotHandled
327 do_tinder_report = data.getVar('TINDER_REPORT', e.data, True)
328 if do_tinder_report and do_tinder_report == "1":
329 tinder_do_tinder_report(e)
330
331 return NotHandled
332}