blob: f415575ded8ac29d3a8b0f2f98f44e24fb4c5cdd [file] [log] [blame]
"""
The LLVM Compiler Infrastructure
This file is distributed under the University of Illinois Open Source
License. See LICENSE.TXT for details.
"""
from __future__ import absolute_import
from __future__ import print_function
# System modules
import curses
import datetime
import math
import sys
import time
# Third-party modules
# LLDB modules
from lldbsuite.test import lldbcurses
from . import results_formatter
from ..event_builder import EventBuilder
class Curses(results_formatter.ResultsFormatter):
"""Receives live results from tests that are running and reports them to the terminal in a curses GUI"""
def __init__(self, out_file, options, file_is_stream):
# Initialize the parent
super(Curses, self).__init__(out_file, options, file_is_stream)
self.using_terminal = True
self.have_curses = True
self.initialize_event = None
self.jobs = [None] * 64
self.job_tests = [None] * 64
self.results = list()
try:
self.main_window = lldbcurses.intialize_curses()
self.main_window.add_key_action(
'\t',
self.main_window.select_next_first_responder,
"Switch between views that can respond to keyboard input")
self.main_window.refresh()
self.job_panel = None
self.results_panel = None
self.status_panel = None
self.info_panel = None
self.hide_status_list = list()
self.start_time = time.time()
except:
self.have_curses = False
lldbcurses.terminate_curses()
self.using_terminal = False
print("Unexpected error:", sys.exc_info()[0])
raise
self.line_dict = dict()
# self.events_file = open("/tmp/events.txt", "w")
# self.formatters = list()
# if tee_results_formatter:
# self.formatters.append(tee_results_formatter)
def status_to_short_str(self, status, test_event):
if status == EventBuilder.STATUS_SUCCESS:
return '.'
elif status == EventBuilder.STATUS_FAILURE:
return 'F'
elif status == EventBuilder.STATUS_UNEXPECTED_SUCCESS:
return '?'
elif status == EventBuilder.STATUS_EXPECTED_FAILURE:
return 'X'
elif status == EventBuilder.STATUS_SKIP:
return 'S'
elif status == EventBuilder.STATUS_ERROR:
if test_event.get("issue_phase", None) == "build":
# Build failure
return 'B'
else:
return 'E'
elif status == EventBuilder.STATUS_TIMEOUT:
return 'T'
elif status == EventBuilder.STATUS_EXPECTED_TIMEOUT:
return 't'
else:
return status
def show_info_panel(self):
selected_idx = self.results_panel.get_selected_idx()
if selected_idx >= 0 and selected_idx < len(self.results):
if self.info_panel is None:
info_frame = self.results_panel.get_contained_rect(
top_inset=10, left_inset=10, right_inset=10, height=30)
self.info_panel = lldbcurses.BoxedPanel(
info_frame, "Result Details")
# Add a key action for any key that will hide this panel when
# any key is pressed
self.info_panel.add_key_action(-1,
self.hide_info_panel,
'Hide the info panel')
self.info_panel.top()
else:
self.info_panel.show()
self.main_window.push_first_responder(self.info_panel)
test_start = self.results[selected_idx][0]
test_result = self.results[selected_idx][1]
self.info_panel.set_line(
0, "File: %s" %
(test_start['test_filename']))
self.info_panel.set_line(
1, "Test: %s.%s" %
(test_start['test_class'], test_start['test_name']))
self.info_panel.set_line(
2, "Time: %s" %
(test_result['elapsed_time']))
self.info_panel.set_line(3, "Status: %s" % (test_result['status']))
def hide_info_panel(self):
self.main_window.pop_first_responder(self.info_panel)
self.info_panel.hide()
self.main_window.refresh()
def toggle_status(self, status):
if status:
# Toggle showing and hiding results whose status matches "status"
# in "Results" window
if status in self.hide_status_list:
self.hide_status_list.remove(status)
else:
self.hide_status_list.append(status)
self.update_results()
def update_results(self, update=True):
'''Called after a category of test have been show/hidden to update the results list with
what the user desires to see.'''
self.results_panel.clear(update=False)
for result in self.results:
test_result = result[1]
status = test_result['status']
if status in self.hide_status_list:
continue
name = test_result['test_class'] + '.' + test_result['test_name']
self.results_panel.append_line(
'%s (%6.2f sec) %s' %
(self.status_to_short_str(
status,
test_result),
test_result['elapsed_time'],
name))
if update:
self.main_window.refresh()
def handle_event(self, test_event):
with self.lock:
super(Curses, self).handle_event(test_event)
# for formatter in self.formatters:
# formatter.process_event(test_event)
if self.have_curses:
worker_index = -1
if 'worker_index' in test_event:
worker_index = test_event['worker_index']
if 'event' in test_event:
check_for_one_key = True
#print(str(test_event), file=self.events_file)
event = test_event['event']
if self.status_panel:
self.status_panel.update_status(
'time', str(
datetime.timedelta(
seconds=math.floor(
time.time() - self.start_time))))
if event == 'test_start':
name = test_event['test_class'] + \
'.' + test_event['test_name']
self.job_tests[worker_index] = test_event
if 'pid' in test_event:
line = 'pid: %5d ' % (test_event['pid']) + name
else:
line = name
self.job_panel.set_line(worker_index, line)
self.main_window.refresh()
elif event == 'test_result':
status = test_event['status']
self.status_panel.increment_status(status)
if 'pid' in test_event:
line = 'pid: %5d ' % (test_event['pid'])
else:
line = ''
self.job_panel.set_line(worker_index, line)
name = test_event['test_class'] + \
'.' + test_event['test_name']
elapsed_time = test_event[
'event_time'] - self.job_tests[worker_index]['event_time']
if status not in self.hide_status_list:
self.results_panel.append_line(
'%s (%6.2f sec) %s' %
(self.status_to_short_str(
status, test_event), elapsed_time, name))
self.main_window.refresh()
# Append the result pairs
test_event['elapsed_time'] = elapsed_time
self.results.append(
[self.job_tests[worker_index], test_event])
self.job_tests[worker_index] = ''
elif event == 'job_begin':
self.jobs[worker_index] = test_event
if 'pid' in test_event:
line = 'pid: %5d ' % (test_event['pid'])
else:
line = ''
self.job_panel.set_line(worker_index, line)
elif event == 'job_end':
self.jobs[worker_index] = ''
self.job_panel.set_line(worker_index, '')
elif event == 'initialize':
self.initialize_event = test_event
num_jobs = test_event['worker_count']
job_frame = self.main_window.get_contained_rect(
height=num_jobs + 2)
results_frame = self.main_window.get_contained_rect(
top_inset=num_jobs + 2, bottom_inset=1)
status_frame = self.main_window.get_contained_rect(
height=1, top_inset=self.main_window.get_size().h - 1)
self.job_panel = lldbcurses.BoxedPanel(
frame=job_frame, title="Jobs")
self.results_panel = lldbcurses.BoxedPanel(
frame=results_frame, title="Results")
self.results_panel.add_key_action(
curses.KEY_UP,
self.results_panel.select_prev,
"Select the previous list entry")
self.results_panel.add_key_action(
curses.KEY_DOWN, self.results_panel.select_next, "Select the next list entry")
self.results_panel.add_key_action(
curses.KEY_HOME,
self.results_panel.scroll_begin,
"Scroll to the start of the list")
self.results_panel.add_key_action(
curses.KEY_END, self.results_panel.scroll_end, "Scroll to the end of the list")
self.results_panel.add_key_action(
curses.KEY_ENTER,
self.show_info_panel,
"Display info for the selected result item")
self.results_panel.add_key_action(
'.',
lambda: self.toggle_status(
EventBuilder.STATUS_SUCCESS),
"Toggle showing/hiding tests whose status is 'success'")
self.results_panel.add_key_action(
'e',
lambda: self.toggle_status(
EventBuilder.STATUS_ERROR),
"Toggle showing/hiding tests whose status is 'error'")
self.results_panel.add_key_action(
'f',
lambda: self.toggle_status(
EventBuilder.STATUS_FAILURE),
"Toggle showing/hiding tests whose status is 'failure'")
self.results_panel.add_key_action('s', lambda: self.toggle_status(
EventBuilder.STATUS_SKIP), "Toggle showing/hiding tests whose status is 'skip'")
self.results_panel.add_key_action(
'x',
lambda: self.toggle_status(
EventBuilder.STATUS_EXPECTED_FAILURE),
"Toggle showing/hiding tests whose status is 'expected_failure'")
self.results_panel.add_key_action(
'?',
lambda: self.toggle_status(
EventBuilder.STATUS_UNEXPECTED_SUCCESS),
"Toggle showing/hiding tests whose status is 'unexpected_success'")
self.status_panel = lldbcurses.StatusPanel(
frame=status_frame)
self.main_window.add_child(self.job_panel)
self.main_window.add_child(self.results_panel)
self.main_window.add_child(self.status_panel)
self.main_window.set_first_responder(
self.results_panel)
self.status_panel.add_status_item(
name="time",
title="Elapsed",
format="%s",
width=20,
value="0:00:00",
update=False)
self.status_panel.add_status_item(
name=EventBuilder.STATUS_SUCCESS,
title="Success",
format="%u",
width=20,
value=0,
update=False)
self.status_panel.add_status_item(
name=EventBuilder.STATUS_FAILURE,
title="Failure",
format="%u",
width=20,
value=0,
update=False)
self.status_panel.add_status_item(
name=EventBuilder.STATUS_ERROR,
title="Error",
format="%u",
width=20,
value=0,
update=False)
self.status_panel.add_status_item(
name=EventBuilder.STATUS_SKIP,
title="Skipped",
format="%u",
width=20,
value=0,
update=True)
self.status_panel.add_status_item(
name=EventBuilder.STATUS_EXPECTED_FAILURE,
title="Expected Failure",
format="%u",
width=30,
value=0,
update=False)
self.status_panel.add_status_item(
name=EventBuilder.STATUS_UNEXPECTED_SUCCESS,
title="Unexpected Success",
format="%u",
width=30,
value=0,
update=False)
self.main_window.refresh()
elif event == 'terminate':
# self.main_window.key_event_loop()
lldbcurses.terminate_curses()
check_for_one_key = False
self.using_terminal = False
# Check for 1 keypress with no delay
# Check for 1 keypress with no delay
if check_for_one_key:
self.main_window.key_event_loop(0, 1)