artiq/artiq/gui/log.py

248 lines
8.6 KiB
Python
Raw Normal View History

2015-07-22 05:13:50 +08:00
import asyncio
import logging
import time
2015-07-22 05:13:50 +08:00
2015-07-23 23:06:37 +08:00
from quamash import QtGui, QtCore
2015-10-14 21:21:19 +08:00
from pyqtgraph import dockarea, LayoutWidget
2015-05-24 00:38:30 +08:00
2015-07-22 05:13:50 +08:00
from artiq.protocols.sync_struct import Subscriber
try:
QSortFilterProxyModel = QtCore.QSortFilterProxyModel
except AttributeError:
QSortFilterProxyModel = QtGui.QSortFilterProxyModel
2015-07-22 05:13:50 +08:00
def _level_to_name(level):
if level >= logging.CRITICAL:
return "CRITICAL"
if level >= logging.ERROR:
return "ERROR"
if level >= logging.WARNING:
return "WARNING"
if level >= logging.INFO:
return "INFO"
return "DEBUG"
2015-10-14 21:21:19 +08:00
2015-10-31 00:34:14 +08:00
class _LogModel(QtCore.QAbstractTableModel):
2015-07-22 05:13:50 +08:00
def __init__(self, parent, init):
2015-10-31 00:34:14 +08:00
QtCore.QAbstractTableModel.__init__(self, parent)
self.headers = ["Level", "Source", "Time", "Message"]
self.entries = init
self.pending_entries = []
self.depth = 1000
timer = QtCore.QTimer(self)
timer.timeout.connect(self.timer_tick)
timer.start(100)
2015-07-25 11:38:26 +08:00
self.fixed_font = QtGui.QFont()
self.fixed_font.setFamily("Monospace")
2015-10-14 16:30:16 +08:00
self.white = QtGui.QBrush(QtGui.QColor(255, 255, 255))
self.black = QtGui.QBrush(QtGui.QColor(0, 0, 0))
self.debug_fg = QtGui.QBrush(QtGui.QColor(55, 55, 55))
self.warning_bg = QtGui.QBrush(QtGui.QColor(255, 255, 180))
self.error_bg = QtGui.QBrush(QtGui.QColor(255, 150, 150))
2015-10-31 00:34:14 +08:00
def headerData(self, col, orientation, role):
if (orientation == QtCore.Qt.Horizontal
and role == QtCore.Qt.DisplayRole):
return self.headers[col]
return None
def rowCount(self, parent):
return len(self.entries)
def columnCount(self, parent):
return len(self.headers)
def __delitem__(self, k):
pass
def append(self, v):
self.pending_entries.append(v)
def insertRows(self, position, rows=1, index=QtCore.QModelIndex()):
self.beginInsertRows(QtCore.QModelIndex(), position, position+rows-1)
self.endInsertRows()
def removeRows(self, position, rows=1, index=QtCore.QModelIndex()):
self.beginRemoveRows(QtCore.QModelIndex(), position, position+rows-1)
self.endRemoveRows()
def timer_tick(self):
if not self.pending_entries:
return
nrows = len(self.entries)
records = self.pending_entries
self.pending_entries = []
self.entries.extend(records)
self.insertRows(nrows, len(records))
if len(self.entries) > self.depth:
start = len(self.entries) - self.depth
self.entries = self.entries[start:]
self.removeRows(0, start)
2015-07-25 11:38:26 +08:00
def data(self, index, role):
2015-10-31 00:34:14 +08:00
if index.isValid():
if (role == QtCore.Qt.FontRole
and index.column() == 3):
return self.fixed_font
elif role == QtCore.Qt.BackgroundRole:
level = self.entries[index.row()][0]
if level >= logging.ERROR:
return self.error_bg
elif level >= logging.WARNING:
return self.warning_bg
else:
return self.white
elif role == QtCore.Qt.ForegroundRole:
level = self.entries[index.row()][0]
if level <= logging.DEBUG:
return self.debug_fg
else:
return self.black
elif role == QtCore.Qt.DisplayRole:
v = self.entries[index.row()]
column = index.column()
if column == 0:
return _level_to_name(v[0])
elif column == 1:
return v[1]
elif column == 2:
return time.strftime("%m/%d %H:%M:%S", time.localtime(v[2]))
else:
return v[3]
2015-07-22 05:13:50 +08:00
2015-05-24 00:38:30 +08:00
2015-10-30 19:58:33 +08:00
class _LogFilterProxyModel(QSortFilterProxyModel):
def __init__(self, min_level, freetext):
QSortFilterProxyModel.__init__(self)
2015-10-14 21:21:19 +08:00
self.min_level = min_level
2015-10-30 19:58:33 +08:00
self.freetext = freetext
2015-10-14 21:21:19 +08:00
def filterAcceptsRow(self, sourceRow, sourceParent):
model = self.sourceModel()
2015-10-30 19:58:33 +08:00
2015-10-14 21:21:19 +08:00
index = model.index(sourceRow, 0, sourceParent)
data = model.data(index, QtCore.Qt.DisplayRole)
2015-10-30 19:58:33 +08:00
accepted_level = getattr(logging, data) >= self.min_level
if self.freetext:
index = model.index(sourceRow, 1, sourceParent)
data_source = model.data(index, QtCore.Qt.DisplayRole)
index = model.index(sourceRow, 3, sourceParent)
data_message = model.data(index, QtCore.Qt.DisplayRole)
accepted_freetext = (self.freetext in data_source
or self.freetext in data_message)
else:
accepted_freetext = True
return accepted_level and accepted_freetext
2015-10-14 21:21:19 +08:00
def set_min_level(self, min_level):
self.min_level = min_level
self.invalidateFilter()
2015-10-30 19:58:33 +08:00
def set_freetext(self, freetext):
self.freetext = freetext
self.invalidateFilter()
2015-10-14 21:21:19 +08:00
2015-05-24 00:38:30 +08:00
class LogDock(dockarea.Dock):
def __init__(self):
dockarea.Dock.__init__(self, "Log", size=(1000, 300))
2015-07-22 05:13:50 +08:00
2015-10-14 21:21:19 +08:00
grid = LayoutWidget()
self.addWidget(grid)
grid.addWidget(QtGui.QLabel("Minimum level: "), 0, 0)
2015-10-30 19:58:33 +08:00
self.filter_level = QtGui.QComboBox()
self.filter_level.addItems(["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"])
self.filter_level.setToolTip("Display entries at or above this level")
grid.addWidget(self.filter_level, 0, 1)
self.filter_level.currentIndexChanged.connect(
self.filter_level_changed)
self.filter_freetext = QtGui.QLineEdit()
self.filter_freetext.setPlaceholderText("freetext filter...")
self.filter_freetext.editingFinished.connect(
self.filter_freetext_changed)
grid.addWidget(self.filter_freetext, 0, 2)
2015-10-14 21:21:19 +08:00
2015-07-22 05:13:50 +08:00
self.log = QtGui.QTableView()
self.log.setSelectionMode(QtGui.QAbstractItemView.NoSelection)
2015-07-23 22:36:52 +08:00
self.log.horizontalHeader().setResizeMode(
QtGui.QHeaderView.ResizeToContents)
2015-07-23 23:06:37 +08:00
self.log.setHorizontalScrollMode(
QtGui.QAbstractItemView.ScrollPerPixel)
self.log.setShowGrid(False)
self.log.setTextElideMode(QtCore.Qt.ElideNone)
2015-10-30 19:58:33 +08:00
grid.addWidget(self.log, 1, 0, colspan=4)
2015-07-24 00:11:05 +08:00
self.scroll_at_bottom = False
2015-07-22 05:13:50 +08:00
2015-10-03 19:28:57 +08:00
async def sub_connect(self, host, port):
2015-07-22 05:13:50 +08:00
self.subscriber = Subscriber("log", self.init_log_model)
2015-10-03 19:28:57 +08:00
await self.subscriber.connect(host, port)
2015-07-22 05:13:50 +08:00
2015-10-03 19:28:57 +08:00
async def sub_close(self):
await self.subscriber.close()
2015-07-22 05:13:50 +08:00
2015-10-30 19:58:33 +08:00
def filter_level_changed(self):
if not hasattr(self, "table_model_filter"):
return
2015-10-14 21:21:19 +08:00
self.table_model_filter.set_min_level(
2015-10-30 19:58:33 +08:00
getattr(logging, self.filter_level.currentText()))
def filter_freetext_changed(self):
if not hasattr(self, "table_model_filter"):
return
2015-10-30 19:58:33 +08:00
self.table_model_filter.set_freetext(self.filter_freetext.text())
2015-10-14 21:21:19 +08:00
2015-07-24 00:11:05 +08:00
def rows_inserted_before(self):
scrollbar = self.log.verticalScrollBar()
self.scroll_value = scrollbar.value()
self.scroll_at_bottom = self.scroll_value == scrollbar.maximum()
2015-07-24 00:11:05 +08:00
def rows_inserted_after(self):
if self.scroll_at_bottom:
self.log.scrollToBottom()
# HACK:
# Qt intermittently likes to scroll back to the top when rows are removed.
# Work around this by restoring the scrollbar to the previously memorized
# position, after the removal.
# Note that this works because _LogModel always does the insertion right
# before the removal.
def rows_removed(self):
if self.scroll_at_bottom:
self.log.scrollToBottom()
else:
scrollbar = self.log.verticalScrollBar()
scrollbar.setValue(self.scroll_value)
2015-07-22 05:13:50 +08:00
def init_log_model(self, init):
self.table_model = _LogModel(self.log, init)
2015-10-30 19:58:33 +08:00
self.table_model_filter = _LogFilterProxyModel(
getattr(logging, self.filter_level.currentText()),
self.filter_freetext.text())
self.table_model_filter.setSourceModel(self.table_model)
2015-10-14 21:21:19 +08:00
self.log.setModel(self.table_model_filter)
self.table_model_filter.rowsAboutToBeInserted.connect(self.rows_inserted_before)
self.table_model_filter.rowsInserted.connect(self.rows_inserted_after)
self.table_model_filter.rowsRemoved.connect(self.rows_removed)
return self.table_model
2015-10-14 21:21:19 +08:00
def save_state(self):
2015-10-30 19:58:33 +08:00
return {"min_level_idx": self.filter_level.currentIndex()}
2015-10-14 21:21:19 +08:00
def restore_state(self, state):
try:
idx = state["min_level_idx"]
except KeyError:
pass
else:
2015-10-30 19:58:33 +08:00
self.filter_level.setCurrentIndex(idx)