forked from M-Labs/artiq
348 lines
11 KiB
Python
348 lines
11 KiB
Python
import asyncio
|
|
|
|
from quamash import QtGui, QtCore
|
|
from pyqtgraph import dockarea
|
|
from pyqtgraph import LayoutWidget
|
|
|
|
from artiq.protocols.sync_struct import Subscriber
|
|
from artiq.protocols import pyon
|
|
from artiq.gui.tools import DictSyncModel
|
|
from artiq.gui.scan import ScanController
|
|
|
|
|
|
class _ExplistModel(DictSyncModel):
|
|
def __init__(self, explorer, parent, init):
|
|
self.explorer = explorer
|
|
DictSyncModel.__init__(self,
|
|
["Experiment"],
|
|
parent, init)
|
|
|
|
def sort_key(self, k, v):
|
|
return k
|
|
|
|
def convert(self, k, v, column):
|
|
return k
|
|
|
|
def __setitem__(self, k, v):
|
|
DictSyncModel.__setitem__(self, k, v)
|
|
if k == self.explorer.selected_key:
|
|
self.explorer.update_selection(k, k)
|
|
|
|
|
|
class _FreeValueEntry(QtGui.QLineEdit):
|
|
def __init__(self, procdesc):
|
|
QtGui.QLineEdit.__init__(self)
|
|
if "default" in procdesc:
|
|
self.set_argument_value(procdesc["default"])
|
|
|
|
def get_argument_value(self):
|
|
return pyon.decode(self.text())
|
|
|
|
def set_argument_value(self, value):
|
|
self.setText(pyon.encode(value))
|
|
|
|
|
|
class _BooleanEntry(QtGui.QCheckBox):
|
|
def __init__(self, procdesc):
|
|
QtGui.QCheckBox.__init__(self)
|
|
if "default" in procdesc:
|
|
self.set_argument_value(procdesc["default"])
|
|
|
|
def get_argument_value(self):
|
|
return self.isChecked()
|
|
|
|
def set_argument_value(self, value):
|
|
self.setChecked(value)
|
|
|
|
|
|
class _EnumerationEntry(QtGui.QComboBox):
|
|
def __init__(self, procdesc):
|
|
QtGui.QComboBox.__init__(self)
|
|
self.choices = procdesc["choices"]
|
|
self.addItems(self.choices)
|
|
if "default" in procdesc:
|
|
self.set_argument_value(procdesc["default"])
|
|
|
|
def get_argument_value(self):
|
|
return self.choices[self.currentIndex()]
|
|
|
|
def set_argument_value(self, value):
|
|
idx = self.choices.index(value)
|
|
self.setCurrentIndex(idx)
|
|
|
|
|
|
class _NumberEntry(QtGui.QDoubleSpinBox):
|
|
def __init__(self, procdesc):
|
|
QtGui.QDoubleSpinBox.__init__(self)
|
|
self.setDecimals(procdesc["ndecimals"])
|
|
self.setSingleStep(procdesc["step"])
|
|
if procdesc["min"] is not None:
|
|
self.setMinimum(procdesc["min"])
|
|
else:
|
|
self.setMinimum(float("-inf"))
|
|
if procdesc["max"] is not None:
|
|
self.setMaximum(procdesc["max"])
|
|
else:
|
|
self.setMaximum(float("inf"))
|
|
if procdesc["unit"]:
|
|
self.setSuffix(" " + procdesc["unit"])
|
|
if "default" in procdesc:
|
|
self.set_argument_value(procdesc["default"])
|
|
|
|
def get_argument_value(self):
|
|
return self.value()
|
|
|
|
def set_argument_value(self, value):
|
|
self.setValue(value)
|
|
|
|
|
|
class _StringEntry(QtGui.QLineEdit):
|
|
def __init__(self, procdesc):
|
|
QtGui.QLineEdit.__init__(self)
|
|
if "default" in procdesc:
|
|
self.set_argument_value(procdesc["default"])
|
|
|
|
def get_argument_value(self):
|
|
return self.text()
|
|
|
|
def set_argument_value(self, value):
|
|
self.setText(value)
|
|
|
|
|
|
_procty_to_entry = {
|
|
"FreeValue": _FreeValueEntry,
|
|
"BooleanValue": _BooleanEntry,
|
|
"EnumerationValue": _EnumerationEntry,
|
|
"NumberValue": _NumberEntry,
|
|
"StringValue": _StringEntry,
|
|
"Scannable": ScanController
|
|
}
|
|
|
|
|
|
class _ArgumentEditor(QtGui.QTreeWidget):
|
|
def __init__(self, dialog_parent):
|
|
QtGui.QTreeWidget.__init__(self)
|
|
self.setColumnCount(2)
|
|
self.header().setResizeMode(QtGui.QHeaderView.ResizeToContents)
|
|
self.header().setVisible(False)
|
|
self.setSelectionMode(QtGui.QAbstractItemView.NoSelection)
|
|
|
|
self.dialog_parent = dialog_parent
|
|
self._groups = dict()
|
|
self.set_arguments([])
|
|
|
|
def clear(self):
|
|
QtGui.QTreeWidget.clear(self)
|
|
self._groups.clear()
|
|
|
|
def _get_group(self, name):
|
|
if name in self._groups:
|
|
return self._groups[name]
|
|
group = QtGui.QTreeWidgetItem([name, ""])
|
|
for c in 0, 1:
|
|
group.setBackground(c, QtGui.QBrush(QtGui.QColor(100, 100, 100)))
|
|
group.setForeground(c, QtGui.QBrush(QtGui.QColor(220, 220, 255)))
|
|
font = group.font(c)
|
|
font.setBold(True)
|
|
group.setFont(c, font)
|
|
self.addTopLevelItem(group)
|
|
self._groups[name] = group
|
|
return group
|
|
|
|
def set_arguments(self, arguments):
|
|
self.clear()
|
|
|
|
if not arguments:
|
|
self.addTopLevelItem(QtGui.QTreeWidgetItem(["No arguments", ""]))
|
|
|
|
self._args_to_entries = dict()
|
|
for n, (name, (procdesc, group)) in enumerate(arguments):
|
|
entry = _procty_to_entry[procdesc["ty"]](procdesc)
|
|
self._args_to_entries[name] = entry
|
|
|
|
widget_item = QtGui.QTreeWidgetItem([name, ""])
|
|
if group is None:
|
|
self.addTopLevelItem(widget_item)
|
|
else:
|
|
self._get_group(group).addChild(widget_item)
|
|
self.setItemWidget(widget_item, 1, entry)
|
|
|
|
def get_argument_values(self, show_error_message):
|
|
r = dict()
|
|
for arg, entry in self._args_to_entries.items():
|
|
try:
|
|
r[arg] = entry.get_argument_value()
|
|
except Exception as e:
|
|
if show_error_message:
|
|
msgbox = QtGui.QMessageBox(self.dialog_parent)
|
|
msgbox.setWindowTitle("Error")
|
|
msgbox.setText("Failed to obtain value for argument '{}':\n{}"
|
|
.format(arg, str(e)))
|
|
msgbox.setStandardButtons(QtGui.QMessageBox.Ok)
|
|
msgbox.show()
|
|
return None
|
|
return r
|
|
|
|
def set_argument_values(self, arguments, ignore_errors):
|
|
for arg, value in arguments.items():
|
|
try:
|
|
entry = self._args_to_entries[arg]
|
|
entry.set_argument_value(value)
|
|
except:
|
|
if not ignore_errors:
|
|
raise
|
|
|
|
def save_state(self):
|
|
expanded = []
|
|
for k, v in self._groups.items():
|
|
if v.isExpanded():
|
|
expanded.append(k)
|
|
argument_values = self.get_argument_values(False)
|
|
return {
|
|
"expanded": expanded,
|
|
"argument_values": argument_values
|
|
}
|
|
|
|
def restore_state(self, state):
|
|
self.set_argument_values(state["argument_values"], True)
|
|
for e in state["expanded"]:
|
|
try:
|
|
self._groups[e].setExpanded(True)
|
|
except KeyError:
|
|
pass
|
|
|
|
|
|
class ExplorerDock(dockarea.Dock):
|
|
def __init__(self, dialog_parent, status_bar, schedule_ctl):
|
|
dockarea.Dock.__init__(self, "Explorer", size=(1500, 500))
|
|
|
|
self.dialog_parent = dialog_parent
|
|
self.status_bar = status_bar
|
|
self.schedule_ctl = schedule_ctl
|
|
|
|
self.splitter = QtGui.QSplitter(QtCore.Qt.Horizontal)
|
|
self.addWidget(self.splitter)
|
|
|
|
grid = LayoutWidget()
|
|
self.splitter.addWidget(grid)
|
|
|
|
self.el = QtGui.QListView()
|
|
self.el.selectionChanged = self._selection_changed
|
|
self.selected_key = None
|
|
grid.addWidget(self.el, 0, 0, colspan=4)
|
|
|
|
self.datetime = QtGui.QDateTimeEdit()
|
|
self.datetime.setDisplayFormat("MMM d yyyy hh:mm:ss")
|
|
self.datetime.setCalendarPopup(True)
|
|
self.datetime.setDate(QtCore.QDate.currentDate())
|
|
self.datetime.dateTimeChanged.connect(self.enable_duedate)
|
|
self.datetime_en = QtGui.QCheckBox("Due date:")
|
|
grid.addWidget(self.datetime_en, 1, 0)
|
|
grid.addWidget(self.datetime, 1, 1)
|
|
|
|
self.priority = QtGui.QSpinBox()
|
|
self.priority.setRange(-99, 99)
|
|
grid.addWidget(QtGui.QLabel("Priority:"), 1, 2)
|
|
grid.addWidget(self.priority, 1, 3)
|
|
|
|
self.pipeline = QtGui.QLineEdit()
|
|
self.pipeline.setText("main")
|
|
grid.addWidget(QtGui.QLabel("Pipeline:"), 2, 0)
|
|
grid.addWidget(self.pipeline, 2, 1)
|
|
|
|
self.flush = QtGui.QCheckBox("Flush")
|
|
grid.addWidget(self.flush, 2, 2, colspan=2)
|
|
|
|
submit = QtGui.QPushButton("Submit")
|
|
grid.addWidget(submit, 3, 0, colspan=4)
|
|
submit.clicked.connect(self.submit_clicked)
|
|
|
|
self.argeditor = _ArgumentEditor(self.dialog_parent)
|
|
self.splitter.addWidget(self.argeditor)
|
|
self.splitter.setSizes([grid.minimumSizeHint().width(), 1000])
|
|
self.state = dict()
|
|
|
|
def update_selection(self, selected, deselected):
|
|
if deselected:
|
|
self.state[deselected] = self.argeditor.save_state()
|
|
|
|
if selected:
|
|
expinfo = self.explist_model.backing_store[selected]
|
|
self.argeditor.set_arguments(expinfo["arguments"])
|
|
if selected in self.state:
|
|
self.argeditor.restore_state(self.state[selected])
|
|
self.splitter.insertWidget(1, self.argeditor)
|
|
self.selected_key = selected
|
|
|
|
def _sel_to_key(self, selection):
|
|
selection = selection.indexes()
|
|
if selection:
|
|
row = selection[0].row()
|
|
return self.explist_model.row_to_key[row]
|
|
else:
|
|
return None
|
|
|
|
def _selection_changed(self, selected, deselected):
|
|
self.update_selection(self._sel_to_key(selected),
|
|
self._sel_to_key(deselected))
|
|
|
|
def save_state(self):
|
|
idx = self.el.selectedIndexes()
|
|
if idx:
|
|
row = idx[0].row()
|
|
key = self.explist_model.row_to_key[row]
|
|
self.state[key] = self.argeditor.save_state()
|
|
return self.state
|
|
|
|
def restore_state(self, state):
|
|
self.state = state
|
|
|
|
def enable_duedate(self):
|
|
self.datetime_en.setChecked(True)
|
|
|
|
@asyncio.coroutine
|
|
def sub_connect(self, host, port):
|
|
self.explist_subscriber = Subscriber("explist",
|
|
self.init_explist_model)
|
|
yield from self.explist_subscriber.connect(host, port)
|
|
|
|
@asyncio.coroutine
|
|
def sub_close(self):
|
|
yield from self.explist_subscriber.close()
|
|
|
|
def init_explist_model(self, init):
|
|
self.explist_model = _ExplistModel(self, self.el, init)
|
|
self.el.setModel(self.explist_model)
|
|
return self.explist_model
|
|
|
|
@asyncio.coroutine
|
|
def submit(self, pipeline_name, file, class_name, arguments,
|
|
priority, due_date, flush):
|
|
expid = {
|
|
"repo_rev": None,
|
|
"file": file,
|
|
"class_name": class_name,
|
|
"arguments": arguments,
|
|
}
|
|
rid = yield from self.schedule_ctl.submit(pipeline_name, expid,
|
|
priority, due_date, flush)
|
|
self.status_bar.showMessage("Submitted RID {}".format(rid))
|
|
|
|
def submit_clicked(self):
|
|
if self.selected_key is not None:
|
|
expinfo = self.explist_model.backing_store[self.selected_key]
|
|
if self.datetime_en.isChecked():
|
|
due_date = self.datetime.dateTime().toMSecsSinceEpoch()/1000
|
|
else:
|
|
due_date = None
|
|
arguments = self.argeditor.get_argument_values(True)
|
|
if arguments is None:
|
|
return
|
|
asyncio.ensure_future(self.submit(self.pipeline.text(),
|
|
expinfo["file"],
|
|
expinfo["class_name"],
|
|
arguments,
|
|
self.priority.value(),
|
|
due_date,
|
|
self.flush.isChecked()))
|