1#!/usr/bin/python2
2
3"""A script that provides convertion between models.job and a protocol
4buffer object.
5
6This script contains only one class that takes an job instance and
7convert it into a protocol buffer object. The class will also be
8responsible for serializing the job instance via protocol buffers.
9
10"""
11
12# import python libraries
13import datetime
14import time
15import logging
16
17# import autotest libraries
18from autotest_lib.tko import models
19from autotest_lib.tko import tko_pb2
20from autotest_lib.tko import utils
21
22__author__ = 'darrenkuo@google.com (Darren Kuo)'
23
24mktime = time.mktime
25datetime = datetime.datetime
26
27class JobSerializer(object):
28    """A class that takes a job object of the tko module and package
29    it with a protocol buffer.
30
31    This class will take a model.job object as input and create a
32    protocol buffer to include all the content of the job object. This
33    protocol buffer object will be serialized into a binary file.
34    """
35
36    def __init__(self):
37
38        self.job_type_dict = {'dir':str, 'tests':list, 'user':str,
39                              'label':str, 'machine':str,
40                              'queued_time':datetime,
41                              'started_time':datetime,
42                              'finished_time':datetime,
43                              'machine_owner':str,
44                              'machine_group':str, 'aborted_by':str,
45                              'aborted_on':datetime,
46                              'keyval_dict':dict,
47                              'afe_parent_job_id':str,
48                              'build_version':str,
49                              'suite':str,
50                              'board':str}
51
52        self.test_type_dict = {'subdir':str, 'testname':str,
53                               'status':str, 'reason':str,
54                               'kernel':models.kernel, 'machine':str,
55                               'started_time':datetime,
56                               'finished_time':datetime,
57                               'iterations':list, 'attributes':dict,
58                               'labels':list}
59
60        self.kernel_type_dict = {'base':str, 'kernel_hash':str}
61
62        self.iteration_type_dict = {'index':int, 'attr_keyval':dict,
63                                    'perf_keyval':dict}
64
65
66    def deserialize_from_binary(self, infile):
67        """Takes in a binary file name and returns a tko job object.
68
69        The method first deserialize the binary into a protocol buffer
70        job object and then converts the job object into a tko job
71        object.
72
73
74        @param infile: the name of the binary file that will be deserialized.
75
76        @return: a tko job that is represented by the binary file will
77        be returned.
78        """
79
80        job_pb = tko_pb2.Job()
81
82        binary = open(infile, 'r')
83        try:
84            job_pb.ParseFromString(binary.read())
85        finally:
86            binary.close()
87
88        return self.get_tko_job(job_pb)
89
90
91    def serialize_to_binary(self, the_job, tag, binaryfilename):
92        """Serializes the tko job object into a binary by using a
93        protocol buffer.
94
95        The method takes a tko job object and constructs a protocol
96        buffer job object. Then invokes the native serializing
97        function on the object to get a binary string. The string is
98        then written to outfile.
99
100        Precondition: Assumes that all the information about the job
101        is already in the job object. Any fields that is None will be
102        provided a default value.
103
104        @param the_job: the tko job object that will be serialized.
105        tag: contains the job name and the afe_job_id
106        binaryfilename: the name of the file that will be written to
107        @param tag: The job tag string.
108        @param binaryfilename: The output filename.
109
110        @return: the filename of the file that contains the
111        binary of the serialized object.
112        """
113
114        pb_job = tko_pb2.Job()
115        self.set_pb_job(the_job, pb_job, tag)
116
117        out = open(binaryfilename, 'wb')
118        try:
119            out.write(pb_job.SerializeToString())
120        finally:
121            out.close()
122
123
124    def set_afe_job_id_and_tag(self, pb_job, tag):
125        """Sets the pb job's afe_job_id and tag field.
126
127        @param
128        pb_job: the pb job that will have it's fields set.
129        tag: used to set pb_job.tag and pb_job.afe_job_id.
130        """
131        pb_job.tag = tag
132        pb_job.afe_job_id = utils.get_afe_job_id(tag)
133
134
135    # getter setter methods
136    def get_tko_job(self, job):
137        """Creates a a new tko job object from the pb job object.
138
139        Uses getter methods on the pb objects to extract all the
140        attributes and finally constructs a tko job object using the
141        models.job constructor.
142
143        @param
144        job: a pb job where data is being extracted from.
145
146        @return a tko job object.
147        """
148
149        fields_dict = self.get_trivial_attr(job, self.job_type_dict)
150
151        fields_dict['tests'] = [self.get_tko_test(test) for test in job.tests]
152
153        fields_dict['keyval_dict'] = dict((keyval.name, keyval.value)
154                                          for keyval in job.keyval_dict)
155
156        newjob = models.job(fields_dict['dir'], fields_dict['user'],
157                            fields_dict['label'],
158                            fields_dict['machine'],
159                            fields_dict['queued_time'],
160                            fields_dict['started_time'],
161                            fields_dict['finished_time'],
162                            fields_dict['machine_owner'],
163                            fields_dict['machine_group'],
164                            fields_dict['aborted_by'],
165                            fields_dict['aborted_on'],
166                            fields_dict['keyval_dict'])
167
168        newjob.tests.extend(fields_dict['tests'])
169
170        return newjob
171
172
173    def set_pb_job(self, tko_job, pb_job, tag):
174        """Set the fields for the new job object.
175
176        Method takes in a tko job and an empty protocol buffer job
177        object.  Then safely sets all the appropriate field by first
178        testing if the value in the original object is None.
179
180        @param
181        tko_job: a tko job instance that will have it's values
182        transfered to the new job
183        pb_job: a new instance of the job class provided in the
184        protocol buffer.
185        tag: used to set pb_job.tag and pb_job.afe_job_id.
186        """
187
188        self.set_trivial_attr(tko_job, pb_job, self.job_type_dict)
189        self.set_afe_job_id_and_tag(pb_job, tag)
190        if hasattr(tko_job, 'index'):
191            pb_job.job_idx = tko_job.index
192
193        for test in tko_job.tests:
194            newtest = pb_job.tests.add()
195            self.set_pb_test(test, newtest)
196
197        for key, val in tko_job.keyval_dict.iteritems():
198            newkeyval = pb_job.keyval_dict.add()
199            newkeyval.name = key
200            newkeyval.value = str(val)
201
202
203    def get_tko_test(self, test):
204        """Creates a tko test from pb_test.
205
206        Extracts data from pb_test by calling helper methods and
207        creates a tko test using the models.test constructor.
208
209        @param:
210        test: a pb_test where fields will be extracted from.
211
212        @return a new instance of models.test
213        """
214        fields_dict = self.get_trivial_attr(test, self.test_type_dict)
215
216        fields_dict['kernel'] = self.get_tko_kernel(test.kernel)
217
218        fields_dict['iterations'] = [self.get_tko_iteration(iteration)
219                                     for iteration in test.iterations]
220
221        fields_dict['attributes'] = dict((keyval.name, keyval.value)
222                                         for keyval in test.attributes)
223
224        fields_dict['labels'] = list(test.labels)
225
226        # The constructor for models.test accepts a "perf_values" list that
227        # represents performance values of the test.  The empty list argument
228        # in the constructor call below represents this value and makes this
229        # code adhere properly to the models.test constructor argument list.
230        # However, the effect of the empty list is that perf values are
231        # ignored in the job_serializer module.  This is ok for now because
232        # autotest does not use the current module.  If job_serializer is used
233        # in the future, we need to modify the "tko.proto" protobuf file to
234        # understand the notion of perf_values, then modify this file
235        # accordingly to use it.
236        return models.test(fields_dict['subdir'],
237                           fields_dict['testname'],
238                           fields_dict['status'],
239                           fields_dict['reason'],
240                           fields_dict['kernel'],
241                           fields_dict['machine'],
242                           fields_dict['started_time'],
243                           fields_dict['finished_time'],
244                           fields_dict['iterations'],
245                           fields_dict['attributes'],
246                           [],
247                           fields_dict['labels'])
248
249
250    def set_pb_test(self, tko_test, pb_test):
251        """Sets the various fields of test object of the tko protocol.
252
253        Method takes a tko test and a new test of the protocol buffer and
254        transfers the values in the tko test to the new test.
255
256        @param
257        tko_test: a tko test instance.
258        pb_test: an empty protocol buffer test instance.
259
260        """
261
262        self.set_trivial_attr(tko_test, pb_test, self.test_type_dict)
263
264        self.set_pb_kernel(tko_test.kernel, pb_test.kernel)
265        if hasattr(tko_test, 'test_idx'):
266            pb_test.test_idx = tko_test.test_idx
267
268        for current_iteration in tko_test.iterations:
269            pb_iteration = pb_test.iterations.add()
270            self.set_pb_iteration(current_iteration, pb_iteration)
271
272        for key, val in tko_test.attributes.iteritems():
273            newkeyval = pb_test.attributes.add()
274            newkeyval.name = key
275            newkeyval.value = str(val)
276
277        for current_label in tko_test.labels:
278            pb_test.labels.append(current_label)
279
280
281    def get_tko_kernel(self, kernel):
282        """Constructs a new tko kernel object from a pb kernel object.
283
284        Uses all the getter methods on the pb kernel object to extract
285        the attributes and constructs a new tko kernel object using
286        the model.kernel constructor.
287
288        @param
289        kernel: a pb kernel object where data will be extracted.
290
291        @return a new tko kernel object.
292        """
293
294        fields_dict = self.get_trivial_attr(kernel, self.kernel_type_dict)
295
296        return models.kernel(fields_dict['base'], [], fields_dict['kernel_hash'])
297
298
299    def set_pb_kernel(self, tko_kernel, pb_kernel):
300        """Set a specific kernel of a test.
301
302        Takes the same form of all the other setting methods.  It
303        seperates the string variables from the int variables and set
304        them safely.
305
306        @param
307        tko_kernel: a tko kernel.
308        pb_kernel: an empty protocol buffer kernel.
309
310        """
311
312        self.set_trivial_attr(tko_kernel, pb_kernel, self.kernel_type_dict)
313
314
315    def get_tko_iteration(self, iteration):
316        """Creates a new tko iteration with the data in the provided
317        pb iteration.
318
319        Uses the data in the pb iteration and the models.iteration
320        constructor to create a new tko iterations
321
322        @param
323        iteration: a pb iteration instance
324
325        @return a tko iteration instance with the same data.
326        """
327
328        fields_dict = self.get_trivial_attr(iteration,
329                                            self.iteration_type_dict)
330
331        fields_dict['attr_keyval'] = dict((keyval.name, keyval.value)
332                                          for keyval in iteration.attr_keyval)
333
334        fields_dict['perf_keyval'] = dict((keyval.name, keyval.value)
335                                          for keyval in iteration.perf_keyval)
336
337        return models.iteration(fields_dict['index'],
338                                fields_dict['attr_keyval'],
339                                fields_dict['perf_keyval'])
340
341
342    def set_pb_iteration(self, tko_iteration, pb_iteration):
343        """Sets all fields for a particular iteration.
344
345        Takes same form as all the other setting methods. Sets int,
346        str and datetime variables safely.
347
348        @param
349        tko_iteration: a tko test iteration.
350        pb_iteration: an empty pb test iteration.
351
352        """
353
354        self.set_trivial_attr(tko_iteration, pb_iteration,
355                              self.iteration_type_dict)
356
357        for key, val in tko_iteration.attr_keyval.iteritems():
358            newkeyval = pb_iteration.attr_keyval.add()
359            newkeyval.name = key
360            newkeyval.value = str(val)
361
362        for key, val in tko_iteration.perf_keyval.iteritems():
363            newkeyval = pb_iteration.perf_keyval.add()
364            newkeyval.name = key
365            newkeyval.value = str(val)
366
367
368    def get_trivial_attr(self, obj, objdict):
369        """Get all trivial attributes from the object.
370
371        This function is used to extract attributes from a pb job. The
372        dictionary specifies the types of each attribute in each tko
373        class.
374
375        @param
376        obj: the pb object that is being extracted.
377        objdict: the dict that specifies the type.
378
379        @return a dict of each attr name and it's corresponding value.
380        """
381
382        resultdict = {}
383        for field, field_type in objdict.items():
384            value = getattr(obj, field)
385            if field_type in (str, int, long):
386                resultdict[field] = field_type(value)
387            elif field_type == datetime:
388                resultdict[field] = (
389                            datetime.fromtimestamp(value/1000.0))
390
391        return resultdict
392
393
394    def set_trivial_attr(self, tko_obj, pb_obj, objdict):
395        """Sets all the easy attributes appropriately according to the
396        type.
397
398        This function is used to set all the trivial attributes
399        provided by objdict, the dictionary that specifies the types
400        of each attribute in each tko class.
401
402        @param
403        tko_obj: the original object that has the data being copied.
404        pb_obj: the new pb object that is being copied into.
405        objdict: specifies the type of each attribute in the class we
406        are working with.
407
408        """
409        for attr, attr_type in objdict.iteritems():
410            if attr_type == datetime:
411                t = getattr(tko_obj, attr)
412                if not t:
413                    self.set_attr_safely(pb_obj, attr, t, int)
414                else:
415                    t = mktime(t.timetuple()) + 1e-6 * t.microsecond
416                    setattr(pb_obj, attr, long(t*1000))
417            else:
418                value = getattr(tko_obj, attr)
419                self.set_attr_safely(pb_obj, attr, value, attr_type)
420
421
422    def set_attr_safely(self, var, attr, value, vartype):
423        """Sets a particular attribute of var if the provided value is
424        not None.
425
426        Checks if value is None. If not, set the attribute of the var
427        to be the default value. This is necessary for the special
428        required fields of the protocol buffer.
429
430        @param
431        var: the variable of which one of the attribute is being set.
432        attr: the attribute that is being set.
433        value: the value that is being checked
434        vartype: the expected type of the attr
435
436        """
437
438        supported_types = [int, long, str]
439        if vartype in supported_types:
440            if value is None:
441                value = vartype()
442            elif not isinstance(value, vartype):
443                logging.warning('Unexpected type %s for attr %s, should be %s',
444                                type(value), attr, vartype)
445
446            setattr(var, attr, vartype(value))
447