diff options
-rwxr-xr-x | bitbake/bin/toaster-eventreplay | 207 |
1 files changed, 82 insertions, 125 deletions
diff --git a/bitbake/bin/toaster-eventreplay b/bitbake/bin/toaster-eventreplay index 7de3967fe3..80967a0934 100755 --- a/bitbake/bin/toaster-eventreplay +++ b/bitbake/bin/toaster-eventreplay | |||
@@ -21,11 +21,13 @@ | |||
21 | # with this program; if not, write to the Free Software Foundation, Inc., | 21 | # with this program; if not, write to the Free Software Foundation, Inc., |
22 | # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | 22 | # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. |
23 | 23 | ||
24 | """ | ||
25 | This command takes a filename as a single parameter. The filename is read | ||
26 | as a build eventlog, and the ToasterUI is used to process events in the file | ||
27 | and log data in the database | ||
28 | """ | ||
24 | 29 | ||
25 | # This command takes a filename as a single parameter. The filename is read | 30 | import os |
26 | # as a build eventlog, and the ToasterUI is used to process events in the file | ||
27 | # and log data in the database | ||
28 | |||
29 | import sys | 31 | import sys |
30 | import json | 32 | import json |
31 | import pickle | 33 | import pickle |
@@ -40,130 +42,85 @@ sys.path.insert(0, join(dirname(dirname(abspath(__file__))), 'lib')) | |||
40 | import bb.cooker | 42 | import bb.cooker |
41 | from bb.ui import toasterui | 43 | from bb.ui import toasterui |
42 | 44 | ||
43 | class FileReadEventsServerConnection(): | 45 | class EventPlayer: |
44 | """ Emulates a connection to a bitbake server that feeds | 46 | """Emulate a connection to a bitbake server.""" |
45 | events coming actually read from a saved log file. | 47 | |
46 | """ | 48 | def __init__(self, eventfile, variables): |
47 | 49 | self.eventfile = eventfile | |
48 | class MockConnection(): | 50 | self.variables = variables |
49 | """ fill-in for the proxy to the server. we just return generic data | 51 | self.eventmask = [] |
52 | |||
53 | def waitEvent(self, _timeout): | ||
54 | """Read event from the file.""" | ||
55 | line = self.eventfile.readline().strip() | ||
56 | if not line: | ||
57 | return | ||
58 | try: | ||
59 | event_str = json.loads(line)['vars'].encode('utf-8') | ||
60 | event = pickle.loads(codecs.decode(event_str, 'base64')) | ||
61 | event_name = "%s.%s" % (event.__module__, event.__class__.__name__) | ||
62 | if event_name not in self.eventmask: | ||
63 | return | ||
64 | return event | ||
65 | except ValueError as err: | ||
66 | print("Failed loading ", line) | ||
67 | raise err | ||
68 | |||
69 | def runCommand(self, command_line): | ||
70 | """Emulate running a command on the server.""" | ||
71 | name = command_line[0] | ||
72 | |||
73 | if name == "getVariable": | ||
74 | var_name = command_line[1] | ||
75 | variable = self.variables.get(var_name) | ||
76 | if variable: | ||
77 | return variable['v'], None | ||
78 | return None, "Missing variable %s" % var_name | ||
79 | |||
80 | elif name == "getAllKeysWithFlags": | ||
81 | dump = {} | ||
82 | flaglist = command_line[1] | ||
83 | for key, val in self.variables.items(): | ||
84 | try: | ||
85 | if not key.startswith("__"): | ||
86 | dump[key] = { | ||
87 | 'v': val['v'], | ||
88 | 'history' : val['history'], | ||
89 | } | ||
90 | for flag in flaglist: | ||
91 | dump[key][flag] = val[flag] | ||
92 | except Exception as err: | ||
93 | print(err) | ||
94 | return (dump, None) | ||
95 | |||
96 | elif name == 'setEventMask': | ||
97 | self.eventmask = command_line[-1] | ||
98 | return True, None | ||
99 | |||
100 | else: | ||
101 | raise Exception("Command %s not implemented" % command_line[0]) | ||
102 | |||
103 | def getEventHandle(self): | ||
50 | """ | 104 | """ |
51 | def __init__(self, sc): | 105 | This method is called by toasterui. |
52 | self._sc = sc | 106 | The return value is passed to self.runCommand but not used there. |
53 | self.eventmask = [] | 107 | """ |
54 | 108 | pass | |
55 | def runCommand(self, commandArray): | 109 | |
56 | """ emulates running a command on the server; only read-only commands are accepted """ | 110 | def main(argv): |
57 | command_name = commandArray[0] | 111 | with open(argv[-1]) as eventfile: |
58 | 112 | # load variables from the first line | |
59 | if command_name == "getVariable": | 113 | variables = json.loads(eventfile.readline().strip())['allvariables'] |
60 | if commandArray[1] in self._sc._variables: | 114 | |
61 | return (self._sc._variables[commandArray[1]]['v'], None) | 115 | params = namedtuple('ConfigParams', ['observe_only'])(True) |
62 | return (None, "Missing variable") | 116 | player = EventPlayer(eventfile, variables) |
63 | 117 | ||
64 | elif command_name == "getAllKeysWithFlags": | 118 | return toasterui.main(player, player, params) |
65 | dump = {} | ||
66 | flaglist = commandArray[1] | ||
67 | for k in self._sc._variables.keys(): | ||
68 | try: | ||
69 | if not k.startswith("__"): | ||
70 | v = self._sc._variables[k]['v'] | ||
71 | dump[k] = { | ||
72 | 'v' : v , | ||
73 | 'history' : self._sc._variables[k]['history'], | ||
74 | } | ||
75 | for d in flaglist: | ||
76 | dump[k][d] = self._sc._variables[k][d] | ||
77 | except Exception as e: | ||
78 | print(e) | ||
79 | return (dump, None) | ||
80 | |||
81 | elif command_name == 'setEventMask': | ||
82 | self.eventmask = commandArray[-1] | ||
83 | return True, None | ||
84 | |||
85 | else: | ||
86 | raise Exception("Command %s not implemented" % commandArray[0]) | ||
87 | |||
88 | def terminateServer(self): | ||
89 | """ do not do anything """ | ||
90 | pass | ||
91 | |||
92 | def getEventHandle(self): | ||
93 | pass | ||
94 | |||
95 | |||
96 | class EventReader(): | ||
97 | def __init__(self, sc): | ||
98 | self._sc = sc | ||
99 | self.firstraise = 0 | ||
100 | |||
101 | def _create_event(self, line): | ||
102 | def _import_class(name): | ||
103 | assert len(name) > 0 | ||
104 | assert "." in name, name | ||
105 | |||
106 | components = name.strip().split(".") | ||
107 | modulename = ".".join(components[:-1]) | ||
108 | moduleklass = components[-1] | ||
109 | |||
110 | module = __import__(modulename, fromlist=[str(moduleklass)]) | ||
111 | return getattr(module, moduleklass) | ||
112 | |||
113 | # we build a toaster event out of current event log line | ||
114 | try: | ||
115 | event_data = json.loads(line.strip()) | ||
116 | event_class = _import_class(event_data['class']) | ||
117 | event_str = event_data['vars'].encode('utf-8') | ||
118 | event_object = pickle.loads(codecs.decode(event_str, 'base64')) | ||
119 | except ValueError as e: | ||
120 | print("Failed loading ", line) | ||
121 | raise e | ||
122 | |||
123 | if not isinstance(event_object, event_class): | ||
124 | raise Exception("Error loading objects %s class %s ", event_object, event_class) | ||
125 | |||
126 | return event_object | ||
127 | |||
128 | def waitEvent(self, timeout): | ||
129 | |||
130 | nextline = self._sc._eventfile.readline() | ||
131 | if len(nextline) == 0: | ||
132 | # the build data ended, while toasterui still waits for events. | ||
133 | # this happens when the server was abruptly stopped, so we simulate this | ||
134 | self.firstraise += 1 | ||
135 | if self.firstraise == 1: | ||
136 | raise KeyboardInterrupt() | ||
137 | else: | ||
138 | return None | ||
139 | else: | ||
140 | self._sc.lineno += 1 | ||
141 | return self._create_event(nextline) | ||
142 | |||
143 | |||
144 | def _readVariables(self, variableline): | ||
145 | self._variables = json.loads(variableline.strip())['allvariables'] | ||
146 | |||
147 | |||
148 | def __init__(self, file_name): | ||
149 | self.connection = FileReadEventsServerConnection.MockConnection(self) | ||
150 | self._eventfile = open(file_name, "r") | ||
151 | |||
152 | # we expect to have the variable dump at the start of the file | ||
153 | self.lineno = 1 | ||
154 | self._readVariables(self._eventfile.readline()) | ||
155 | |||
156 | self.events = FileReadEventsServerConnection.EventReader(self) | ||
157 | 119 | ||
158 | # run toaster ui on our mock bitbake class | 120 | # run toaster ui on our mock bitbake class |
159 | if __name__ == "__main__": | 121 | if __name__ == "__main__": |
160 | if len(sys.argv) < 2: | 122 | if len(sys.argv) != 2: |
161 | print("Usage: %s event.log " % sys.argv[0]) | 123 | print("Usage: %s <event file>" % os.path.basename(sys.argv[0])) |
162 | sys.exit(1) | 124 | sys.exit(1) |
163 | 125 | ||
164 | file_name = sys.argv[-1] | 126 | sys.exit(main(sys.argv)) |
165 | mock_connection = FileReadEventsServerConnection(file_name) | ||
166 | configParams = namedtuple('ConfigParams', ['observe_only'])(True) | ||
167 | |||
168 | # run the main program and set exit code to the returned value | ||
169 | sys.exit(toasterui.main(mock_connection.connection, mock_connection.events, configParams)) | ||