artiq/artiq/frontend/artiq_dashboard.py

260 lines
9.6 KiB
Python
Raw Normal View History

#!/usr/bin/env python3
2014-12-29 12:48:14 +08:00
import argparse
import asyncio
2015-01-02 14:47:09 +08:00
import atexit
import importlib
2015-07-17 02:52:53 +08:00
import os
import logging
import sys
2014-12-29 12:48:14 +08:00
2016-02-15 07:23:47 +08:00
from PyQt5 import QtCore, QtGui, QtWidgets
2020-12-12 21:50:11 +08:00
from qasync import QEventLoop
2016-02-09 02:32:40 +08:00
2019-11-10 15:55:17 +08:00
from sipyco.pc_rpc import AsyncioClient, Client
from sipyco.broadcast import Receiver
from sipyco import common_args
2019-11-14 15:21:51 +08:00
from sipyco.asyncio_tools import atexit_register_coroutine
2019-11-10 15:55:17 +08:00
from artiq import __artiq_dir__ as artiq_dir, __version__ as artiq_version
2019-11-14 15:21:51 +08:00
from artiq.tools import get_user_config_dir
2015-11-11 12:13:19 +08:00
from artiq.gui.models import ModelSubscriber
from artiq.gui import state, log
from artiq.dashboard import (experiments, shortcuts, explorer,
moninj, datasets, schedule, applets_ccb)
2015-07-14 04:08:20 +08:00
2015-01-23 00:52:13 +08:00
def get_argparser():
2016-04-04 22:12:45 +08:00
parser = argparse.ArgumentParser(description="ARTIQ Dashboard")
parser.add_argument("--version", action="version",
version="ARTIQ v{}".format(artiq_version),
help="print the ARTIQ version number")
2014-12-29 12:48:14 +08:00
parser.add_argument(
"-s", "--server", default="::1",
help="hostname or IP of the master to connect to")
parser.add_argument(
"--port-notify", default=3250, type=int,
2014-12-31 20:13:10 +08:00
help="TCP port to connect to for notifications")
2014-12-29 12:48:14 +08:00
parser.add_argument(
"--port-control", default=3251, type=int,
2014-12-31 20:13:10 +08:00
help="TCP port to connect to for control")
parser.add_argument(
"--port-broadcast", default=1067, type=int,
help="TCP port to connect to for broadcasts")
2015-01-23 19:00:09 +08:00
parser.add_argument(
"--db-file", default=None,
help="database file for local GUI settings")
parser.add_argument(
"-p", "--load-plugin", dest="plugin_modules", action="append",
help="Python module to load on startup")
2019-11-10 15:55:17 +08:00
common_args.verbosity_args(parser)
2015-01-23 00:52:13 +08:00
return parser
2014-12-29 12:48:14 +08:00
2016-02-15 07:23:47 +08:00
class MainWindow(QtWidgets.QMainWindow):
2016-01-03 00:47:43 +08:00
def __init__(self, server):
2016-02-15 07:23:47 +08:00
QtWidgets.QMainWindow.__init__(self)
2016-03-29 16:59:43 +08:00
2016-02-21 08:06:52 +08:00
icon = QtGui.QIcon(os.path.join(artiq_dir, "gui", "logo.svg"))
2015-11-04 00:35:03 +08:00
self.setWindowIcon(icon)
2016-04-04 22:12:45 +08:00
self.setWindowTitle("ARTIQ Dashboard - {}".format(server))
2016-03-29 16:59:43 +08:00
qfm = QtGui.QFontMetrics(self.font())
self.resize(140*qfm.averageCharWidth(), 38*qfm.lineSpacing())
2015-07-20 00:27:41 +08:00
self.exit_request = asyncio.Event()
def closeEvent(self, event):
event.ignore()
2015-07-20 00:27:41 +08:00
self.exit_request.set()
2015-08-06 22:27:46 +08:00
def save_state(self):
2016-02-15 06:08:14 +08:00
return {
"state": bytes(self.saveState()),
"geometry": bytes(self.saveGeometry())
}
2015-08-06 22:27:46 +08:00
def restore_state(self, state):
2016-02-15 06:08:14 +08:00
self.restoreGeometry(QtCore.QByteArray(state["geometry"]))
self.restoreState(QtCore.QByteArray(state["state"]))
2015-08-06 22:27:46 +08:00
2015-01-23 19:00:09 +08:00
2016-02-21 08:06:52 +08:00
class MdiArea(QtWidgets.QMdiArea):
def __init__(self):
QtWidgets.QMdiArea.__init__(self)
2016-07-18 22:51:17 +08:00
self.pixmap = QtGui.QPixmap(os.path.join(
2016-09-05 23:04:44 +08:00
artiq_dir, "gui", "logo_ver.svg"))
2016-02-21 08:06:52 +08:00
def paintEvent(self, event):
QtWidgets.QMdiArea.paintEvent(self, event)
painter = QtGui.QPainter(self.viewport())
x = (self.width() - self.pixmap.width())//2
y = (self.height() - self.pixmap.height())//2
painter.setOpacity(0.5)
2016-02-21 08:06:52 +08:00
painter.drawPixmap(x, y, self.pixmap)
2015-08-01 16:48:44 +08:00
def main():
2015-11-12 01:13:57 +08:00
# initialize application
2015-08-05 11:41:43 +08:00
args = get_argparser().parse_args()
widget_log_handler = log.init_log(args, "dashboard")
2015-08-05 11:41:43 +08:00
# load any plugin modules first (to register argument_ui classes, etc.)
if args.plugin_modules:
for mod in args.plugin_modules:
importlib.import_module(mod)
if args.db_file is None:
args.db_file = os.path.join(get_user_config_dir(),
"artiq_dashboard_{server}_{port}.pyon".format(
server=args.server.replace(":","."),
port=args.port_notify))
app = QtWidgets.QApplication(["ARTIQ Dashboard"])
2015-05-22 17:05:15 +08:00
loop = QEventLoop(app)
asyncio.set_event_loop(loop)
2015-11-01 00:03:46 +08:00
atexit.register(loop.close)
2015-11-12 01:13:57 +08:00
smgr = state.StateManager(args.db_file)
2015-01-02 14:47:09 +08:00
2015-11-12 01:13:57 +08:00
# create connections to master
rpc_clients = dict()
for target in "schedule", "experiment_db", "dataset_db", "device_db":
client = AsyncioClient()
loop.run_until_complete(client.connect_rpc(
args.server, args.port_control, "master_" + target))
2015-11-01 00:03:46 +08:00
atexit.register(client.close_rpc)
rpc_clients[target] = client
2015-08-01 16:48:44 +08:00
config = Client(args.server, args.port_control, "master_config")
2017-11-28 06:00:30 +08:00
try:
server_name = config.get_name()
finally:
config.close_rpc()
disconnect_reported = False
def report_disconnect():
nonlocal disconnect_reported
if not disconnect_reported:
logging.error("connection to master lost, "
"restart dashboard to reconnect")
disconnect_reported = True
2015-11-11 12:13:19 +08:00
sub_clients = dict()
for notifier_name, modelf in (("explist", explorer.Model),
("explist_status", explorer.StatusUpdater),
("datasets", datasets.Model),
("schedule", schedule.Model)):
subscriber = ModelSubscriber(notifier_name, modelf,
report_disconnect)
2015-11-11 12:13:19 +08:00
loop.run_until_complete(subscriber.connect(
args.server, args.port_notify))
atexit_register_coroutine(subscriber.close)
sub_clients[notifier_name] = subscriber
broadcast_clients = dict()
for target in "log", "ccb":
client = Receiver(target, [], report_disconnect)
loop.run_until_complete(client.connect(
args.server, args.port_broadcast))
atexit_register_coroutine(client.close)
broadcast_clients[target] = client
2015-11-12 01:13:57 +08:00
# initialize main window
2017-11-28 06:00:30 +08:00
main_window = MainWindow(args.server if server_name is None else server_name)
smgr.register(main_window)
2016-02-21 08:06:52 +08:00
mdi_area = MdiArea()
2016-02-16 07:19:50 +08:00
mdi_area.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded)
mdi_area.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded)
main_window.setCentralWidget(mdi_area)
2015-01-05 19:52:58 +08:00
2015-11-12 01:13:57 +08:00
# create UI components
expmgr = experiments.ExperimentManager(main_window,
sub_clients["datasets"],
2015-11-27 19:30:05 +08:00
sub_clients["explist"],
sub_clients["schedule"],
rpc_clients["schedule"],
2015-12-06 18:39:27 +08:00
rpc_clients["experiment_db"])
2015-11-30 11:40:50 +08:00
smgr.register(expmgr)
d_shortcuts = shortcuts.ShortcutsDock(main_window, expmgr)
smgr.register(d_shortcuts)
d_explorer = explorer.ExplorerDock(expmgr, d_shortcuts,
sub_clients["explist"],
sub_clients["explist_status"],
rpc_clients["schedule"],
rpc_clients["experiment_db"],
rpc_clients["device_db"])
smgr.register(d_explorer)
d_datasets = datasets.DatasetsDock(sub_clients["datasets"],
rpc_clients["dataset_db"])
smgr.register(d_datasets)
2015-07-14 23:31:18 +08:00
d_applets = applets_ccb.AppletsCCBDock(main_window,
sub_clients["datasets"],
extra_substitutes={
"server": args.server,
"port_notify": args.port_notify,
"port_control": args.port_control,
})
2016-02-08 16:59:15 +08:00
atexit_register_coroutine(d_applets.stop)
smgr.register(d_applets)
broadcast_clients["ccb"].notify_cbs.append(d_applets.ccb_notify)
2015-07-14 23:31:18 +08:00
d_ttl_dds = moninj.MonInj(rpc_clients["schedule"])
loop.run_until_complete(d_ttl_dds.start(args.server, args.port_notify))
atexit_register_coroutine(d_ttl_dds.stop)
2015-11-11 12:13:19 +08:00
d_schedule = schedule.ScheduleDock(
rpc_clients["schedule"], sub_clients["schedule"])
smgr.register(d_schedule)
2015-01-14 22:22:33 +08:00
logmgr = log.LogDockManager(main_window)
2015-11-12 01:13:57 +08:00
smgr.register(logmgr)
broadcast_clients["log"].notify_cbs.append(logmgr.append_message)
widget_log_handler.callback = logmgr.append_message
2015-11-12 01:13:57 +08:00
# lay out docks
right_docks = [
d_explorer, d_shortcuts,
2017-10-26 02:26:59 +08:00
d_ttl_dds.ttl_dock, d_ttl_dds.dds_dock, d_ttl_dds.dac_dock,
d_datasets, d_applets
]
2016-03-29 16:59:43 +08:00
main_window.addDockWidget(QtCore.Qt.RightDockWidgetArea, right_docks[0])
for d1, d2 in zip(right_docks, right_docks[1:]):
main_window.tabifyDockWidget(d1, d2)
2016-02-15 06:15:18 +08:00
main_window.addDockWidget(QtCore.Qt.BottomDockWidgetArea, d_schedule)
2015-11-12 01:13:57 +08:00
# load/initialize state
if os.name == "nt":
# HACK: show the main window before creating applets.
# Otherwise, the windows of those applets that are in detached
# QDockWidgets fail to be embedded.
main_window.show()
2015-08-01 16:48:44 +08:00
smgr.load()
smgr.start()
2015-11-11 12:13:19 +08:00
atexit_register_coroutine(smgr.stop)
2015-11-12 01:13:57 +08:00
# work around for https://github.com/m-labs/artiq/issues/1307
d_ttl_dds.ttl_dock.show()
d_ttl_dds.dds_dock.show()
2015-11-12 01:13:57 +08:00
# create first log dock if not already in state
d_log0 = logmgr.first_log_dock()
if d_log0 is not None:
main_window.tabifyDockWidget(d_schedule, d_log0)
2015-11-12 01:13:57 +08:00
2017-11-28 06:00:30 +08:00
if server_name is not None:
server_description = server_name + " ({})".format(args.server)
else:
server_description = args.server
2022-06-23 19:09:00 +08:00
logging.info("ARTIQ dashboard version: %s",
artiq_version)
logging.info("ARTIQ dashboard connected to moninj_proxy (%s)", server_description)
2015-11-12 01:13:57 +08:00
# run
main_window.show()
loop.run_until_complete(main_window.exit_request.wait())
2014-12-29 12:48:14 +08:00
if __name__ == "__main__":
main()