blob: 9d480b946ea2ac8046c34a2a7b18b477b3bcdc12 [file] [log] [blame]
Greg Clayton825ab582015-09-22 16:29:15 +00001#!/usr/bin/env python
2
3"""
4 The LLVM Compiler Infrastructure
5
6 This file is distributed under the University of Illinois Open Source
7 License. See LICENSE.TXT for details.
8
9Configuration options for lldbtest.py set by dotest.py during initialization
10"""
11
Zachary Turnerff890da2015-10-19 23:45:41 +000012from __future__ import print_function
Zachary Turnerc1b7cd72015-11-05 19:22:28 +000013from __future__ import absolute_import
Zachary Turnerff890da2015-10-19 23:45:41 +000014
Zachary Turnerc1b7cd72015-11-05 19:22:28 +000015# System modules
Greg Clayton825ab582015-09-22 16:29:15 +000016import curses
Greg Clayton414dba52015-09-24 00:19:42 +000017import datetime
Greg Clayton414dba52015-09-24 00:19:42 +000018import math
Greg Clayton825ab582015-09-22 16:29:15 +000019import sys
Greg Clayton414dba52015-09-24 00:19:42 +000020import time
Greg Clayton825ab582015-09-22 16:29:15 +000021
Zachary Turnerc1b7cd72015-11-05 19:22:28 +000022# Third-party modules
23
24# LLDB modules
25from . import lldbcurses
Zachary Turner905a9882015-12-07 21:23:41 +000026from . import result_formatter
27from .result_formatter import EventBuilder
Todd Fiala46a4e342015-12-02 18:48:38 +000028
Zachary Turnerc1b7cd72015-11-05 19:22:28 +000029
Zachary Turner905a9882015-12-07 21:23:41 +000030class Curses(result_formatter.ResultsFormatter):
Greg Clayton825ab582015-09-22 16:29:15 +000031 """Receives live results from tests that are running and reports them to the terminal in a curses GUI"""
32
33 def __init__(self, out_file, options):
34 # Initialize the parent
35 super(Curses, self).__init__(out_file, options)
36 self.using_terminal = True
37 self.have_curses = True
38 self.initialize_event = None
39 self.jobs = [None] * 64
40 self.job_tests = [None] * 64
41 self.results = list()
Greg Clayton825ab582015-09-22 16:29:15 +000042 try:
Todd Fiala46a4e342015-12-02 18:48:38 +000043 self.main_window = lldbcurses.intialize_curses()
Greg Clayton414dba52015-09-24 00:19:42 +000044 self.main_window.add_key_action('\t', self.main_window.select_next_first_responder, "Switch between views that can respond to keyboard input")
Greg Clayton825ab582015-09-22 16:29:15 +000045 self.main_window.refresh()
46 self.job_panel = None
47 self.results_panel = None
48 self.status_panel = None
49 self.info_panel = None
Greg Claytond13c4fb2015-09-22 17:18:15 +000050 self.hide_status_list = list()
Greg Clayton414dba52015-09-24 00:19:42 +000051 self.start_time = time.time()
Greg Clayton825ab582015-09-22 16:29:15 +000052 except:
53 self.have_curses = False
54 lldbcurses.terminate_curses()
55 self.using_terminal = False
Zachary Turnerff890da2015-10-19 23:45:41 +000056 print("Unexpected error:", sys.exc_info()[0])
Greg Clayton825ab582015-09-22 16:29:15 +000057 raise
Todd Fiala46a4e342015-12-02 18:48:38 +000058
Greg Clayton825ab582015-09-22 16:29:15 +000059 self.line_dict = dict()
Todd Fiala46a4e342015-12-02 18:48:38 +000060 # self.events_file = open("/tmp/events.txt", "w")
Greg Clayton825ab582015-09-22 16:29:15 +000061 # self.formatters = list()
62 # if tee_results_formatter:
63 # self.formatters.append(tee_results_formatter)
64
65 def status_to_short_str(self, status):
Todd Fiala46a4e342015-12-02 18:48:38 +000066 if status == EventBuilder.STATUS_SUCCESS:
Greg Clayton825ab582015-09-22 16:29:15 +000067 return '.'
Todd Fiala46a4e342015-12-02 18:48:38 +000068 elif status == EventBuilder.STATUS_FAILURE:
Greg Clayton825ab582015-09-22 16:29:15 +000069 return 'F'
Todd Fiala46a4e342015-12-02 18:48:38 +000070 elif status == EventBuilder.STATUS_UNEXPECTED_SUCCESS:
Greg Clayton825ab582015-09-22 16:29:15 +000071 return '?'
Todd Fiala46a4e342015-12-02 18:48:38 +000072 elif status == EventBuilder.STATUS_EXPECTED_FAILURE:
Greg Clayton825ab582015-09-22 16:29:15 +000073 return 'X'
Todd Fiala46a4e342015-12-02 18:48:38 +000074 elif status == EventBuilder.STATUS_SKIP:
Greg Clayton825ab582015-09-22 16:29:15 +000075 return 'S'
Todd Fiala46a4e342015-12-02 18:48:38 +000076 elif status == EventBuilder.STATUS_ERROR:
Greg Clayton825ab582015-09-22 16:29:15 +000077 return 'E'
78 else:
79 return status
80
Greg Clayton414dba52015-09-24 00:19:42 +000081 def show_info_panel(self):
82 selected_idx = self.results_panel.get_selected_idx()
83 if selected_idx >= 0 and selected_idx < len(self.results):
84 if self.info_panel is None:
85 info_frame = self.results_panel.get_contained_rect(top_inset=10, left_inset=10, right_inset=10, height=30)
86 self.info_panel = lldbcurses.BoxedPanel(info_frame, "Result Details")
87 # Add a key action for any key that will hide this panel when any key is pressed
88 self.info_panel.add_key_action(-1, self.hide_info_panel, 'Hide the info panel')
89 self.info_panel.top()
90 else:
91 self.info_panel.show()
Todd Fiala46a4e342015-12-02 18:48:38 +000092
Greg Clayton37191a22015-10-07 20:00:28 +000093 self.main_window.push_first_responder(self.info_panel)
Greg Clayton414dba52015-09-24 00:19:42 +000094 test_start = self.results[selected_idx][0]
95 test_result = self.results[selected_idx][1]
96 self.info_panel.set_line(0, "File: %s" % (test_start['test_filename']))
97 self.info_panel.set_line(1, "Test: %s.%s" % (test_start['test_class'], test_start['test_name']))
98 self.info_panel.set_line(2, "Time: %s" % (test_result['elapsed_time']))
99 self.info_panel.set_line(3, "Status: %s" % (test_result['status']))
Greg Claytond13c4fb2015-09-22 17:18:15 +0000100
Greg Clayton414dba52015-09-24 00:19:42 +0000101 def hide_info_panel(self):
Greg Clayton37191a22015-10-07 20:00:28 +0000102 self.main_window.pop_first_responder(self.info_panel)
Todd Fiala46a4e342015-12-02 18:48:38 +0000103 self.info_panel.hide()
Greg Clayton825ab582015-09-22 16:29:15 +0000104 self.main_window.refresh()
Todd Fiala46a4e342015-12-02 18:48:38 +0000105
Greg Clayton414dba52015-09-24 00:19:42 +0000106 def toggle_status(self, status):
107 if status:
108 # Toggle showing and hiding results whose status matches "status" in "Results" window
109 if status in self.hide_status_list:
110 self.hide_status_list.remove(status)
Greg Claytond13c4fb2015-09-22 17:18:15 +0000111 else:
Greg Clayton414dba52015-09-24 00:19:42 +0000112 self.hide_status_list.append(status)
113 self.update_results()
Greg Claytond13c4fb2015-09-22 17:18:15 +0000114
115 def update_results(self, update=True):
116 '''Called after a category of test have been show/hidden to update the results list with
117 what the user desires to see.'''
118 self.results_panel.clear(update=False)
119 for result in self.results:
120 test_result = result[1]
121 status = test_result['status']
122 if status in self.hide_status_list:
123 continue
124 name = test_result['test_class'] + '.' + test_result['test_name']
125 self.results_panel.append_line('%s (%6.2f sec) %s' % (self.status_to_short_str(status), test_result['elapsed_time'], name))
126 if update:
Todd Fiala46a4e342015-12-02 18:48:38 +0000127 self.main_window.refresh()
128
Greg Clayton825ab582015-09-22 16:29:15 +0000129 def handle_event(self, test_event):
130 with self.lock:
131 super(Curses, self).handle_event(test_event)
132 # for formatter in self.formatters:
133 # formatter.process_event(test_event)
134 if self.have_curses:
135 worker_index = -1
136 if 'worker_index' in test_event:
137 worker_index = test_event['worker_index']
138 if 'event' in test_event:
139 check_for_one_key = True
Zachary Turner35d017f2015-10-23 17:04:29 +0000140 #print(str(test_event), file=self.events_file)
Todd Fiala46a4e342015-12-02 18:48:38 +0000141 event = test_event['event']
Greg Clayton414dba52015-09-24 00:19:42 +0000142 if self.status_panel:
143 self.status_panel.update_status('time', str(datetime.timedelta(seconds=math.floor(time.time() - self.start_time))))
Greg Clayton825ab582015-09-22 16:29:15 +0000144 if event == 'test_start':
145 name = test_event['test_class'] + '.' + test_event['test_name']
146 self.job_tests[worker_index] = test_event
147 if 'pid' in test_event:
148 line = 'pid: %5d ' % (test_event['pid']) + name
149 else:
150 line = name
151 self.job_panel.set_line(worker_index, line)
152 self.main_window.refresh()
153 elif event == 'test_result':
154 status = test_event['status']
155 self.status_panel.increment_status(status)
156 if 'pid' in test_event:
157 line = 'pid: %5d ' % (test_event['pid'])
158 else:
159 line = ''
160 self.job_panel.set_line(worker_index, line)
Greg Clayton825ab582015-09-22 16:29:15 +0000161 name = test_event['test_class'] + '.' + test_event['test_name']
162 elapsed_time = test_event['event_time'] - self.job_tests[worker_index]['event_time']
Greg Claytond13c4fb2015-09-22 17:18:15 +0000163 if not status in self.hide_status_list:
164 self.results_panel.append_line('%s (%6.2f sec) %s' % (self.status_to_short_str(status), elapsed_time, name))
Greg Clayton825ab582015-09-22 16:29:15 +0000165 self.main_window.refresh()
166 # Append the result pairs
167 test_event['elapsed_time'] = elapsed_time
168 self.results.append([self.job_tests[worker_index], test_event])
169 self.job_tests[worker_index] = ''
170 elif event == 'job_begin':
171 self.jobs[worker_index] = test_event
172 if 'pid' in test_event:
173 line = 'pid: %5d ' % (test_event['pid'])
174 else:
175 line = ''
176 self.job_panel.set_line(worker_index, line)
177 elif event == 'job_end':
178 self.jobs[worker_index] = ''
179 self.job_panel.set_line(worker_index, '')
Todd Fiala46a4e342015-12-02 18:48:38 +0000180 elif event == 'initialize':
Greg Clayton825ab582015-09-22 16:29:15 +0000181 self.initialize_event = test_event
182 num_jobs = test_event['worker_count']
183 job_frame = self.main_window.get_contained_rect(height=num_jobs+2)
184 results_frame = self.main_window.get_contained_rect(top_inset=num_jobs+2, bottom_inset=1)
185 status_frame = self.main_window.get_contained_rect(height=1, top_inset=self.main_window.get_size().h-1)
Greg Clayton414dba52015-09-24 00:19:42 +0000186 self.job_panel = lldbcurses.BoxedPanel(frame=job_frame, title="Jobs")
187 self.results_panel = lldbcurses.BoxedPanel(frame=results_frame, title="Results")
Todd Fiala46a4e342015-12-02 18:48:38 +0000188
Greg Clayton414dba52015-09-24 00:19:42 +0000189 self.results_panel.add_key_action(curses.KEY_UP, self.results_panel.select_prev , "Select the previous list entry")
190 self.results_panel.add_key_action(curses.KEY_DOWN, self.results_panel.select_next , "Select the next list entry")
191 self.results_panel.add_key_action(curses.KEY_HOME, self.results_panel.scroll_begin , "Scroll to the start of the list")
192 self.results_panel.add_key_action(curses.KEY_END, self.results_panel.scroll_end , "Scroll to the end of the list")
193 self.results_panel.add_key_action(curses.KEY_ENTER, self.show_info_panel , "Display info for the selected result item")
Todd Fiala46a4e342015-12-02 18:48:38 +0000194 self.results_panel.add_key_action('.', lambda : self.toggle_status(EventBuilder.STATUS_SUCCESS) , "Toggle showing/hiding tests whose status is 'success'")
195 self.results_panel.add_key_action('e', lambda : self.toggle_status(EventBuilder.STATUS_ERROR) , "Toggle showing/hiding tests whose status is 'error'")
196 self.results_panel.add_key_action('f', lambda : self.toggle_status(EventBuilder.STATUS_FAILURE) , "Toggle showing/hiding tests whose status is 'failure'")
197 self.results_panel.add_key_action('s', lambda : self.toggle_status(EventBuilder.STATUS_SKIP) , "Toggle showing/hiding tests whose status is 'skip'")
198 self.results_panel.add_key_action('x', lambda : self.toggle_status(EventBuilder.STATUS_EXPECTED_FAILURE) , "Toggle showing/hiding tests whose status is 'expected_failure'")
199 self.results_panel.add_key_action('?', lambda : self.toggle_status(EventBuilder.STATUS_UNEXPECTED_SUCCESS), "Toggle showing/hiding tests whose status is 'unexpected_success'")
Greg Clayton825ab582015-09-22 16:29:15 +0000200 self.status_panel = lldbcurses.StatusPanel(frame=status_frame)
Todd Fiala46a4e342015-12-02 18:48:38 +0000201
Greg Clayton825ab582015-09-22 16:29:15 +0000202 self.main_window.add_child(self.job_panel)
203 self.main_window.add_child(self.results_panel)
204 self.main_window.add_child(self.status_panel)
205 self.main_window.set_first_responder(self.results_panel)
Todd Fiala46a4e342015-12-02 18:48:38 +0000206
Greg Clayton414dba52015-09-24 00:19:42 +0000207 self.status_panel.add_status_item(name="time", title="Elapsed", format="%s", width=20, value="0:00:00", update=False)
Todd Fiala46a4e342015-12-02 18:48:38 +0000208 self.status_panel.add_status_item(name=EventBuilder.STATUS_SUCCESS, title="Success", format="%u", width=20, value=0, update=False)
209 self.status_panel.add_status_item(name=EventBuilder.STATUS_FAILURE, title="Failure", format="%u", width=20, value=0, update=False)
210 self.status_panel.add_status_item(name=EventBuilder.STATUS_ERROR, title="Error", format="%u", width=20, value=0, update=False)
211 self.status_panel.add_status_item(name=EventBuilder.STATUS_SKIP, title="Skipped", format="%u", width=20, value=0, update=True)
212 self.status_panel.add_status_item(name=EventBuilder.STATUS_EXPECTED_FAILURE, title="Expected Failure", format="%u", width=30, value=0, update=False)
213 self.status_panel.add_status_item(name=EventBuilder.STATUS_UNEXPECTED_SUCCESS, title="Unexpected Success", format="%u", width=30, value=0, update=False)
Greg Clayton825ab582015-09-22 16:29:15 +0000214 self.main_window.refresh()
215 elif event == 'terminate':
Greg Claytonb877a202015-09-24 16:37:41 +0000216 #self.main_window.key_event_loop()
Greg Clayton825ab582015-09-22 16:29:15 +0000217 lldbcurses.terminate_curses()
218 check_for_one_key = False
219 self.using_terminal = False
Todd Fiala46a4e342015-12-02 18:48:38 +0000220 # Check for 1 keypress with no delay
221
Greg Clayton825ab582015-09-22 16:29:15 +0000222 # Check for 1 keypress with no delay
223 if check_for_one_key:
Todd Fiala46a4e342015-12-02 18:48:38 +0000224 self.main_window.key_event_loop(0, 1)