2023-05-19 11:23:39 +08:00
|
|
|
from PyQt6 import QtWidgets, uic
|
2023-05-19 13:45:01 +08:00
|
|
|
from PyQt6.QtCore import QThread, QThreadPool, pyqtSignal, QRunnable, QObject, QSignalBlocker, pyqtSlot, QDeadlineTimer
|
2023-05-19 11:23:39 +08:00
|
|
|
from pyqtgraph import PlotWidget
|
|
|
|
from pyqtgraph.parametertree import Parameter, ParameterTree, ParameterItem, registerParameterType
|
|
|
|
import pyqtgraph as pg
|
|
|
|
import sys
|
|
|
|
import argparse
|
|
|
|
import logging
|
2023-06-28 15:01:47 +08:00
|
|
|
import asyncio
|
|
|
|
import atexit
|
|
|
|
from qasync import asyncSlot, QEventLoop
|
|
|
|
import qasync
|
2023-05-19 11:23:39 +08:00
|
|
|
from pytec.client import Client
|
|
|
|
|
|
|
|
# pyuic6 -x tec_qt.ui -o ui_tec_qt.py
|
|
|
|
from ui_tec_qt import Ui_MainWindow
|
|
|
|
|
|
|
|
tec_client: Client = None
|
|
|
|
|
|
|
|
# ui = None
|
|
|
|
ui: Ui_MainWindow = None
|
|
|
|
|
2023-06-28 14:59:54 +08:00
|
|
|
thread_pool = QThreadPool.globalInstance()
|
2023-05-19 11:23:39 +08:00
|
|
|
connection_watcher = None
|
2023-05-19 13:45:01 +08:00
|
|
|
client_watcher = None
|
|
|
|
app: QtWidgets.QApplication = None
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
|
2023-06-28 15:01:47 +08:00
|
|
|
class CommandsParameter(Parameter):
|
|
|
|
def __init__(self, **opts):
|
|
|
|
super().__init__()
|
|
|
|
self.opts["commands"] = opts.get("commands", None)
|
|
|
|
self.opts["payload"] = opts.get("payload", None)
|
|
|
|
|
|
|
|
|
|
|
|
ThermostatParams = [[
|
|
|
|
{'name': 'Constant Current', 'type': 'float', 'value': 0, 'step': 0.1, 'limits': (-3, 3), 'siPrefix': True,
|
|
|
|
'suffix': 'A', 'commands': [f'pwm {ch} i_set {{value}}']},
|
|
|
|
{'name': 'Temperature PID', 'type': 'bool', 'value': False, 'commands': [f'pwm {ch} pid'], 'payload': ch,
|
|
|
|
'children': [
|
|
|
|
{'name': 'Set Temperature', 'type': 'float', 'value': 25, 'step': 0.1, 'limits': (-273, 300), 'siPrefix': True,
|
|
|
|
'suffix': '°C', 'commands': [f'pid {ch} target {{value}}']},
|
|
|
|
]},
|
|
|
|
{'name': 'Output Config', 'expanded': False, 'type': 'group', 'children': [
|
|
|
|
{'name': 'Max Current', 'type': 'float', 'value': 0, 'step': 0.1, 'limits': (0, 3), 'siPrefix': True,
|
|
|
|
'suffix': 'A', 'commands': [f'pwm {ch} max_i_pos {{value}}', f'pwm {ch} max_i_neg {{value}}',
|
|
|
|
f'pid {ch} output_min -{{value}}', f'pid {ch} output_max {{value}}']},
|
|
|
|
{'name': 'Max Voltage', 'type': 'float', 'value': 0, 'step': 0.1, 'limits': (0, 5), 'siPrefix': True,
|
|
|
|
'suffix': 'V', 'commands': [f'pwm {ch} max_v {{value}}']},
|
|
|
|
]},
|
|
|
|
{'name': 'Thermistor Config', 'expanded': False, 'type': 'group', 'children': [
|
|
|
|
{'name': 'T0', 'type': 'float', 'value': 25, 'step': 0.1, 'limits': (-100, 100), 'siPrefix': True,
|
|
|
|
'suffix': 'C', 'commands': [f's-h {ch} t0 {{value}}']},
|
|
|
|
{'name': 'R0', 'type': 'float', 'value': 10000, 'step': 1, 'siPrefix': True, 'suffix': 'Ohm',
|
|
|
|
'commands': [f's-h {ch} r0 {{value}}']},
|
|
|
|
{'name': 'Beta', 'type': 'float', 'value': 3950, 'step': 1, 'commands': [f's-h {ch} b {{value}}']},
|
|
|
|
]},
|
|
|
|
{'name': 'PID Config', 'expanded': False, 'type': 'group', 'children': [
|
|
|
|
{'name': 'kP', 'type': 'float', 'value': 0, 'step': 0.1, 'commands': [f'pid {ch} kp {{value}}']},
|
|
|
|
{'name': 'kI', 'type': 'float', 'value': 0, 'step': 0.1, 'commands': [f'pid {ch} ki {{value}}']},
|
|
|
|
{'name': 'kD', 'type': 'float', 'value': 0, 'step': 0.1, 'commands': [f'pid {ch} kd {{value}}']},
|
|
|
|
{'name': 'PID Auto Tune', 'expanded': False, 'type': 'group', 'children': [
|
|
|
|
{'name': 'Target Temperature', 'type': 'float', 'value': 20, 'step': 0.1, 'siPrefix': True, 'suffix': 'C'},
|
|
|
|
{'name': 'Test Current', 'type': 'float', 'value': 1, 'step': 0.1, 'siPrefix': True, 'suffix': 'A'},
|
|
|
|
{'name': 'Temperature Swing', 'type': 'float', 'value': 1.5, 'step': 0.1, 'siPrefix': True, 'suffix': 'C'},
|
|
|
|
{'name': 'Run', 'type': 'action', 'tip': 'Run'},
|
|
|
|
]},
|
|
|
|
]}
|
|
|
|
] for ch in range(2)]
|
|
|
|
|
|
|
|
params = [CommandsParameter.create(name='Thermostat Params 0', type='group', children=ThermostatParams[0]),
|
|
|
|
CommandsParameter.create(name='Thermostat Params 1', type='group', children=ThermostatParams[1])]
|
|
|
|
|
|
|
|
|
2023-05-19 11:23:39 +08:00
|
|
|
def get_argparser():
|
|
|
|
parser = argparse.ArgumentParser(description="ARTIQ master")
|
|
|
|
|
|
|
|
parser.add_argument("--connect", default=None, action="store_true",
|
|
|
|
help="Automatically connect to the specified Thermostat in IP:port format")
|
|
|
|
parser.add_argument('IP', metavar="ip", default=None, nargs='?')
|
|
|
|
parser.add_argument('PORT', metavar="port", default=None, nargs='?')
|
|
|
|
parser.add_argument("-l", "--log", dest="logLevel", choices=['DEBUG', 'INFO', 'WARNING', 'ERROR', 'CRITICAL'],
|
|
|
|
help="Set the logging level")
|
|
|
|
|
|
|
|
return parser
|
|
|
|
|
|
|
|
|
2023-06-28 14:59:54 +08:00
|
|
|
class WatchConnectTask(QThread):
|
2023-05-19 11:23:39 +08:00
|
|
|
connected = pyqtSignal(bool)
|
|
|
|
hw_rev = pyqtSignal(dict)
|
|
|
|
connecting = pyqtSignal()
|
|
|
|
fan_update = pyqtSignal(object)
|
|
|
|
|
2023-05-19 13:45:01 +08:00
|
|
|
def __init__(self, parent, ip, port):
|
2023-05-19 11:23:39 +08:00
|
|
|
self.ip = ip
|
|
|
|
self.port = port
|
|
|
|
super().__init__(parent)
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
global tec_client
|
|
|
|
try:
|
|
|
|
if tec_client:
|
|
|
|
tec_client.disconnect()
|
|
|
|
tec_client = None
|
|
|
|
self.connected.emit(False)
|
|
|
|
else:
|
|
|
|
self.connecting.emit()
|
|
|
|
tec_client = Client(host=self.ip, port=self.port, timeout=30)
|
|
|
|
self.connected.emit(True)
|
2023-06-28 14:59:54 +08:00
|
|
|
thread_pool.start(ClientTask(lambda: self.hw_rev.emit(tec_client.hw_rev())))
|
2023-06-28 15:01:47 +08:00
|
|
|
thread_pool.start(ClientTask(lambda: self.fan_update.emit(tec_client.fan())))
|
2023-05-19 11:23:39 +08:00
|
|
|
except Exception as e:
|
|
|
|
logging.error(f"Failed communicating to the {self.ip}:{self.port}: {e}")
|
|
|
|
self.connected.emit(False)
|
|
|
|
|
|
|
|
@pyqtSlot()
|
|
|
|
def client_disconnected(self):
|
|
|
|
global tec_client
|
|
|
|
if tec_client:
|
|
|
|
tec_client.disconnect()
|
|
|
|
tec_client = None
|
2023-05-19 13:45:01 +08:00
|
|
|
self.connected.emit(False)
|
|
|
|
|
|
|
|
|
2023-06-28 14:59:54 +08:00
|
|
|
class ClientWatcher(QThread):
|
2023-05-19 13:45:01 +08:00
|
|
|
fan_update = pyqtSignal(object)
|
|
|
|
pwm_update = pyqtSignal(object)
|
|
|
|
report_update = pyqtSignal(object)
|
|
|
|
pid_update = pyqtSignal(object)
|
|
|
|
|
|
|
|
def __init__(self, parent, update_s):
|
|
|
|
self.update_s = update_s
|
|
|
|
self.running = True
|
|
|
|
super().__init__(parent)
|
|
|
|
|
2023-06-28 14:59:54 +08:00
|
|
|
def run(self):
|
2023-05-19 13:45:01 +08:00
|
|
|
while self.running:
|
2023-06-28 14:59:54 +08:00
|
|
|
thread_pool.start(ClientTask(lambda: self.update_params()))
|
|
|
|
self.msleep(int(self.update_s * 1000))
|
2023-05-19 13:45:01 +08:00
|
|
|
|
|
|
|
def update_params(self):
|
|
|
|
self.fan_update.emit(tec_client.fan())
|
2023-06-28 15:01:47 +08:00
|
|
|
self.pwm_update.emit(tec_client.get_pwm())
|
|
|
|
self.report_update.emit(tec_client._command("report"))
|
|
|
|
self.pid_update.emit(tec_client.get_pid())
|
2023-05-19 13:45:01 +08:00
|
|
|
|
|
|
|
@pyqtSlot()
|
|
|
|
def stop_watching(self):
|
|
|
|
self.running = False
|
2023-06-28 14:59:54 +08:00
|
|
|
deadline = QDeadlineTimer()
|
|
|
|
deadline.setDeadline(100)
|
|
|
|
self.wait(deadline)
|
|
|
|
self.terminate()
|
2023-05-19 13:45:01 +08:00
|
|
|
|
|
|
|
@pyqtSlot()
|
|
|
|
def set_update_s(self):
|
|
|
|
self.update_s = ui.report_refresh_spin.value()
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
|
2023-06-28 14:59:54 +08:00
|
|
|
class ClientTask(QRunnable):
|
|
|
|
def __init__(self, func, *args, **kwargs):
|
|
|
|
self.func = func
|
|
|
|
self.args = args
|
|
|
|
self.kwargs = kwargs
|
|
|
|
super().__init__()
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
try:
|
|
|
|
self.func(*self.args, **self.kwargs)
|
|
|
|
except (TimeoutError, OSError):
|
|
|
|
logging.warning("Client connection error, disconnecting", exc_info=True)
|
|
|
|
if connection_watcher:
|
|
|
|
thread_pool.clear() # clearing all next requests
|
|
|
|
connection_watcher.client_disconnected()
|
|
|
|
|
|
|
|
|
|
|
|
def connected(result):
|
2023-05-19 13:45:01 +08:00
|
|
|
global client_watcher, connection_watcher
|
2023-05-19 11:23:39 +08:00
|
|
|
ui.graph_group.setEnabled(result)
|
|
|
|
ui.hw_rev_lbl.setEnabled(result)
|
|
|
|
ui.fan_group.setEnabled(result)
|
|
|
|
ui.report_group.setEnabled(result)
|
|
|
|
|
|
|
|
ui.ip_set_line.setEnabled(not result)
|
|
|
|
ui.port_set_spin.setEnabled(not result)
|
|
|
|
ui.status_lbl.setText("Connected" if result else "Disconnected")
|
|
|
|
ui.connect_btn.setText("Disconnect" if result else "Connect")
|
|
|
|
if not result:
|
|
|
|
ui.hw_rev_lbl.setText("Thermostat vX.Y")
|
|
|
|
ui.fan_group.setStyleSheet("")
|
2023-05-19 13:45:01 +08:00
|
|
|
if client_watcher:
|
|
|
|
client_watcher.stop_watching()
|
|
|
|
client_watcher = None
|
|
|
|
elif client_watcher is None:
|
|
|
|
client_watcher = ClientWatcher(ui.main_widget, ui.report_refresh_spin.value())
|
|
|
|
client_watcher.fan_update.connect(fan_update)
|
|
|
|
ui.report_apply_btn.clicked.connect(client_watcher.set_update_s)
|
|
|
|
app.aboutToQuit.connect(client_watcher.stop_watching)
|
2023-06-28 14:59:54 +08:00
|
|
|
client_watcher.start()
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
|
|
|
|
def hw_rev(hw_rev_d: dict):
|
|
|
|
logging.debug(hw_rev_d)
|
|
|
|
ui.hw_rev_lbl.setText(f"Thermostat v{hw_rev_d['rev']['major']}.{hw_rev_d['rev']['major']}")
|
|
|
|
ui.fan_group.setEnabled(hw_rev_d["settings"]["fan_available"])
|
|
|
|
if hw_rev_d["settings"]["fan_pwm_recommended"]:
|
|
|
|
ui.fan_group.setStyleSheet("")
|
|
|
|
ui.fan_group.setToolTip("")
|
|
|
|
else:
|
|
|
|
ui.fan_group.setStyleSheet("background-color: yellow")
|
|
|
|
ui.fan_group.setToolTip("Changing the fan settings of not recommended")
|
|
|
|
|
|
|
|
|
|
|
|
def fan_update(fan_settings):
|
|
|
|
logging.debug(fan_settings)
|
|
|
|
if fan_settings is None:
|
|
|
|
return
|
|
|
|
with QSignalBlocker(ui.fan_power_slider) as _:
|
|
|
|
ui.fan_power_slider.setValue(fan_settings["fan_pwm"])
|
|
|
|
ui.fan_power_slider.setEnabled(not fan_settings["auto_mode"])
|
|
|
|
with QSignalBlocker(ui.fan_auto_box) as _:
|
|
|
|
ui.fan_auto_box.setChecked(fan_settings["auto_mode"])
|
|
|
|
|
|
|
|
|
|
|
|
def fan_set():
|
|
|
|
global tec_client
|
|
|
|
if tec_client is None or ui.fan_auto_box.isChecked():
|
|
|
|
return
|
2023-06-28 14:59:54 +08:00
|
|
|
thread_pool.start(ClientTask(lambda: tec_client.set_param("fan", ui.fan_power_slider.value())))
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
|
|
|
|
def fan_auto_set(enabled):
|
|
|
|
global tec_client
|
|
|
|
if tec_client is None:
|
|
|
|
return
|
|
|
|
ui.fan_power_slider.setEnabled(not enabled)
|
|
|
|
if enabled:
|
2023-06-28 14:59:54 +08:00
|
|
|
thread_pool.start(ClientTask(lambda: tec_client.set_param("fan", "auto")))
|
2023-05-19 11:23:39 +08:00
|
|
|
else:
|
2023-06-28 14:59:54 +08:00
|
|
|
thread_pool.start(ClientTask(lambda: tec_client.set_param("fan", ui.fan_power_slider.value())))
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
|
|
|
|
def connect():
|
|
|
|
global connection_watcher
|
2023-05-19 13:45:01 +08:00
|
|
|
connection_watcher = WatchConnectTask(ui.main_widget, ui.ip_set_line.text(), ui.port_set_spin.value())
|
2023-06-28 14:59:54 +08:00
|
|
|
connection_watcher.connected.connect(connected)
|
2023-05-19 11:23:39 +08:00
|
|
|
connection_watcher.connecting.connect(lambda: ui.status_lbl.setText("Connecting..."))
|
|
|
|
connection_watcher.hw_rev.connect(hw_rev)
|
|
|
|
connection_watcher.fan_update.connect(fan_update)
|
2023-06-28 14:59:54 +08:00
|
|
|
connection_watcher.start()
|
|
|
|
app.aboutToQuit.connect(connection_watcher.terminate)
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
|
2023-06-28 15:01:47 +08:00
|
|
|
def update_pid(pid_settings):
|
|
|
|
for settings in pid_settings:
|
|
|
|
channel = settings["channel"]
|
|
|
|
with QSignalBlocker(params[channel].sigTreeStateChanged) as _:
|
|
|
|
params[channel].child("PID Config", "kP").setValue(settings["parameters"]["kp"])
|
|
|
|
params[channel].child("PID Config", "kI").setValue(settings["parameters"]["ki"])
|
|
|
|
params[channel].child("PID Config", "kD").setValue(settings["parameters"]["kd"])
|
|
|
|
if params[channel].child("Temperature PID").value():
|
|
|
|
params[channel].child("Temperature PID", "Set Temperature").setValue(settings["target"])
|
|
|
|
|
|
|
|
def update_report(report_data):
|
|
|
|
for settings in report_data:
|
|
|
|
channel = settings["channel"]
|
|
|
|
with QSignalBlocker(params[channel].sigTreeStateChanged) as _:
|
|
|
|
params[channel].child("Temperature PID").setValue(settings["pid_engaged"])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def send_command(param, changes):
|
|
|
|
for param, change, data in changes:
|
|
|
|
if param.name() == 'Temperature PID' and not data:
|
|
|
|
ch = param.opts["payload"]
|
|
|
|
thread_pool.start(ClientTask(
|
|
|
|
lambda: tec_client.set_param('pwm', ch, 'i_set', params[ch].child('Constant Current').value())))
|
|
|
|
elif param.opts.get("commands", None) is not None:
|
|
|
|
thread_pool.start(ClientTask(lambda: [tec_client._command(x.format(value=data)) for x in param.opts["commands"]]))
|
|
|
|
|
|
|
|
|
|
|
|
def set_param_tree():
|
|
|
|
ui.ch0_tree.setParameters(params[0], showTop=False)
|
|
|
|
ui.ch1_tree.setParameters(params[1], showTop=False)
|
|
|
|
params[0].sigTreeStateChanged.connect(send_command)
|
|
|
|
params[1].sigTreeStateChanged.connect(send_command)
|
|
|
|
|
|
|
|
|
2023-05-19 11:23:39 +08:00
|
|
|
def main():
|
2023-06-28 14:59:54 +08:00
|
|
|
global ui, thread_pool, app
|
2023-05-19 11:23:39 +08:00
|
|
|
args = get_argparser().parse_args()
|
|
|
|
if args.logLevel:
|
|
|
|
logging.basicConfig(level=getattr(logging, args.logLevel))
|
|
|
|
|
|
|
|
app = QtWidgets.QApplication(sys.argv)
|
2023-06-28 15:01:47 +08:00
|
|
|
|
|
|
|
loop = QEventLoop(app)
|
|
|
|
asyncio.set_event_loop(loop)
|
|
|
|
atexit.register(loop.close)
|
|
|
|
|
2023-05-19 11:23:39 +08:00
|
|
|
main_window = QtWidgets.QMainWindow()
|
2023-05-19 13:45:01 +08:00
|
|
|
ui = Ui_MainWindow()
|
|
|
|
ui.setupUi(main_window)
|
|
|
|
# ui = uic.loadUi('tec_qt.ui', main_window)
|
|
|
|
|
2023-06-28 14:59:54 +08:00
|
|
|
thread_pool = QThreadPool(parent=ui.main_widget)
|
|
|
|
thread_pool.setMaxThreadCount(1) # avoid concurrent requests
|
|
|
|
|
2023-05-19 11:23:39 +08:00
|
|
|
ui.connect_btn.clicked.connect(lambda _checked: connect())
|
|
|
|
ui.fan_power_slider.valueChanged.connect(fan_set)
|
|
|
|
ui.fan_auto_box.stateChanged.connect(fan_auto_set)
|
|
|
|
|
2023-06-28 15:01:47 +08:00
|
|
|
set_param_tree()
|
|
|
|
|
2023-05-19 11:23:39 +08:00
|
|
|
if args.connect:
|
|
|
|
if args.IP:
|
|
|
|
ui.ip_set_line.setText(args.IP)
|
|
|
|
if args.PORT:
|
|
|
|
ui.port_set_spin.setValue(int(args.PORT))
|
|
|
|
ui.connect_btn.click()
|
|
|
|
|
|
|
|
main_window.show()
|
2023-06-28 14:59:54 +08:00
|
|
|
sys.exit(app.exec())
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|