Files
poky/bitbake/bin/toaster-eventreplay
Alexandru DAMIAN 7759cd4931 bitbake: toasterui: proper exit code on toaster errors
This patch modifies the toasterui to properly return the exit
code based on the errors found in the toaster itself.

The upload event file API call will not delete event logs for which
toasterui showed an error. This will facilitate debugging.

Minor enhancement in the buildinfohelper to reduce the number
of lookups on unknown layer objects (prevented testing of the patch).

(Bitbake rev: 1ddd6a9e4280a4adf971132ff1fe7ec9b3252905)

Signed-off-by: Alexandru DAMIAN <alexandru.damian@intel.com>
Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
2015-05-08 17:42:05 +01:00

6.0 KiB
Executable File

#!/usr/bin/env python

ex:ts=4:sw=4:sts=4:et

-- tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil --

Copyright (C) 2014 Alex Damian

This file re-uses code spread throughout other Bitbake source files.

As such, all other copyrights belong to their own right holders.

This program is free software; you can redistribute it and/or modify

it under the terms of the GNU General Public License version 2 as

published by the Free Software Foundation.

This program is distributed in the hope that it will be useful,

but WITHOUT ANY WARRANTY; without even the implied warranty of

MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the

GNU General Public License for more details.

You should have received a copy of the GNU General Public License along

with this program; if not, write to the Free Software Foundation, Inc.,

51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.

This command takes a filename as a single parameter. The filename is read

as a build eventlog, and the ToasterUI is used to process events in the file

and log data in the database

from future import print_function import os import sys, logging

mangle syspath to allow easy import of modules

sys.path.insert(0, os.path.join(os.path.dirname(os.path.dirname(os.path.abspath(file))), 'lib'))

import bb.cooker from bb.ui import toasterui import sys import logging

import json, pickle

class FileReadEventsServerConnection(): """ Emulates a connection to a bitbake server that feeds events coming actually read from a saved log file. """

class MockConnection():
    """ fill-in for the proxy to the server. we just return generic data
    """
    def __init__(self, sc):
        self._sc = sc

    def runCommand(self, commandArray):
        """ emulates running a command on the server; only read-only commands are accepted """
        command_name = commandArray[0]

        if command_name == "getVariable":
            if commandArray[1] in self._sc._variables:
                return (self._sc._variables[commandArray[1]]['v'], None)
            return (None, "Missing variable")

        elif command_name == "getAllKeysWithFlags":
            dump = {}
            flaglist = commandArray[1]
            for k in self._sc._variables.keys():
                try:
                    if not k.startswith("__"):
                        v = self._sc._variables[k]['v']
                        dump[k] = {
                            'v' : v ,
                            'history' : self._sc._variables[k]['history'],
                        }
                        for d in flaglist:
                            dump[k][d] = self._sc._variables[k][d]
                except Exception as e:
                    print(e)
            return (dump, None)
        else:
            raise Exception("Command %s not implemented" % commandArray[0])

    def terminateServer(self):
        """ do not do anything """
        pass



class EventReader():
    def __init__(self, sc):
        self._sc = sc
        self.firstraise = 0

    def _create_event(self, line):
        def _import_class(name):
            assert len(name) > 0
            assert "." in name, name

            components = name.strip().split(".")
            modulename = ".".join(components[:-1])
            moduleklass = components[-1]

            module = __import__(modulename, fromlist=[str(moduleklass)])
            return getattr(module, moduleklass)

        # we build a toaster event out of current event log line
        try:
            event_data = json.loads(line.strip())
            event_class = _import_class(event_data['class'])
            event_object = pickle.loads(json.loads(event_data['vars']))
        except ValueError as e:
            print("Failed loading ", line)
            raise e

        if not isinstance(event_object, event_class):
            raise Exception("Error loading objects %s class %s ", event_object, event_class)

        return event_object

    def waitEvent(self, timeout):

        nextline = self._sc._eventfile.readline()
        if len(nextline) == 0:
            # the build data ended, while toasterui still waits for events.
            # this happens when the server was abruptly stopped, so we simulate this
            self.firstraise += 1
            if self.firstraise == 1:
                raise KeyboardInterrupt()
            else:
                return None
        else:
            self._sc.lineno += 1
        return self._create_event(nextline)


def _readVariables(self, variableline):
    self._variables = json.loads(variableline.strip())['allvariables']


def __init__(self, file_name):
    self.connection = FileReadEventsServerConnection.MockConnection(self)
    self._eventfile = open(file_name, "r")

    # we expect to have the variable dump at the start of the file
    self.lineno = 1
    self._readVariables(self._eventfile.readline())

    self.events = FileReadEventsServerConnection.EventReader(self)

class MockConfigParameters(): """ stand-in for cookerdata.ConfigParameters; as we don't really config a cooker, this serves just to supply needed interfaces for the toaster ui to work """ def init(self): self.observe_only = True # we can only read files

run toaster ui on our mock bitbake class

if name == "main": if len(sys.argv) < 2: print("Usage: %s event.log " % sys.argv[0]) sys.exit(1)

file_name = sys.argv[-1]
mock_connection = FileReadEventsServerConnection(file_name)
configParams = MockConfigParameters()

# run the main program and set exit code to the returned value
sys.exit(toasterui.main(mock_connection.connection, mock_connection.events, configParams))