forked from M-Labs/thermostat
Finish moving over to qasync
Also: -Add aioclient The old client is synchronous and blocking, and the only way to achieve true asynchronous IO is to create a new client that interfaces with asyncio. -Make the GUI `nix run`-able
This commit is contained in:
parent
0ad77047f1
commit
f546a3c61b
44
flake.nix
44
flake.nix
|
@ -58,42 +58,37 @@
|
||||||
|
|
||||||
qasync = pkgs.python3Packages.buildPythonPackage rec {
|
qasync = pkgs.python3Packages.buildPythonPackage rec {
|
||||||
pname = "qasync";
|
pname = "qasync";
|
||||||
version = "0.24.0";
|
version = "0.27.1";
|
||||||
src = pkgs.fetchFromGitHub {
|
format = "pyproject";
|
||||||
owner = "CabbageDevelopment";
|
src = pkgs.fetchPypi {
|
||||||
repo = "qasync";
|
inherit pname version;
|
||||||
rev = "v${version}";
|
sha256 = "sha256-jcdo/R7l3hBEx8MF7M8tOdJNh4A+pxGJ1AJPtHX0mF8=";
|
||||||
sha256 = "sha256-ls5F+VntXXa3n+dULaYWK9sAmwly1nk/5+RGWLrcf2Y=";
|
|
||||||
};
|
};
|
||||||
|
buildInputs = [ pkgs.python3Packages.poetry-core ];
|
||||||
propagatedBuildInputs = [ pkgs.python3Packages.pyqt6 ];
|
propagatedBuildInputs = [ pkgs.python3Packages.pyqt6 ];
|
||||||
nativeCheckInputs = [ pkgs.python3Packages.pytest ];
|
|
||||||
checkPhase = ''
|
|
||||||
pytest -k 'test_qthreadexec.py' # the others cause the test execution to be aborted, I think because of asyncio
|
|
||||||
'';
|
|
||||||
};
|
};
|
||||||
thermostat_gui = pkgs.python3Packages.buildPythonPackage rec {
|
|
||||||
|
thermostat_gui = pkgs.python3Packages.buildPythonPackage {
|
||||||
pname = "thermostat_gui";
|
pname = "thermostat_gui";
|
||||||
version = "0.0.0";
|
version = "0.0.0";
|
||||||
src = self;
|
src = "${self}/pytec";
|
||||||
|
|
||||||
preBuild =
|
|
||||||
''
|
|
||||||
export VERSIONEER_OVERRIDE=${version}
|
|
||||||
export VERSIONEER_REV=v0.0.0
|
|
||||||
'';
|
|
||||||
|
|
||||||
nativeBuildInputs = [ pkgs.qt6.wrapQtAppsHook ];
|
nativeBuildInputs = [ pkgs.qt6.wrapQtAppsHook ];
|
||||||
propagatedBuildInputs = (with pkgs.python3Packages; [ pyqtgraph pyqt6 qasync]);
|
propagatedBuildInputs = [ pkgs.qt6.qtbase ] ++ (with pkgs.python3Packages; [ pyqtgraph pyqt6 qasync ]);
|
||||||
|
|
||||||
dontWrapQtApps = true;
|
dontWrapQtApps = true;
|
||||||
postFixup = ''
|
postFixup = ''
|
||||||
ls -al $out/
|
wrapQtApp "$out/bin/tec_qt"
|
||||||
wrapQtApp "$out/pytec/tec_qt"
|
|
||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
in {
|
in {
|
||||||
packages.x86_64-linux = {
|
packages.x86_64-linux = {
|
||||||
inherit thermostat qasync thermostat_gui;
|
inherit thermostat thermostat_gui;
|
||||||
|
};
|
||||||
|
|
||||||
|
apps.x86_64-linux.thermostat_gui = {
|
||||||
|
type = "app";
|
||||||
|
program = "${self.packages.x86_64-linux.thermostat_gui}/bin/tec_qt";
|
||||||
};
|
};
|
||||||
|
|
||||||
hydraJobs = {
|
hydraJobs = {
|
||||||
|
@ -107,11 +102,6 @@
|
||||||
] ++ (with python3Packages; [
|
] ++ (with python3Packages; [
|
||||||
numpy matplotlib pyqtgraph setuptools pyqt6 qasync
|
numpy matplotlib pyqtgraph setuptools pyqt6 qasync
|
||||||
]);
|
]);
|
||||||
shellHook=
|
|
||||||
''
|
|
||||||
export QT_PLUGIN_PATH=${pkgs.qt6.qtbase}/${pkgs.qt6.qtbase.dev.qtPluginPrefix}
|
|
||||||
export QML2_IMPORT_PATH=${pkgs.qt6.qtbase}/${pkgs.qt6.qtbase.dev.qtQmlPrefix}
|
|
||||||
'';
|
|
||||||
};
|
};
|
||||||
defaultPackage.x86_64-linux = thermostat;
|
defaultPackage.x86_64-linux = thermostat;
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
import asyncio
|
||||||
|
from pytec.aioclient import Client
|
||||||
|
|
||||||
|
async def main():
|
||||||
|
tec = Client()
|
||||||
|
await tec.connect() #(host="192.168.1.26", port=23)
|
||||||
|
await tec.set_param("s-h", 1, "t0", 20)
|
||||||
|
print(await tec.get_pwm())
|
||||||
|
print(await tec.get_pid())
|
||||||
|
print(await tec.get_pwm())
|
||||||
|
print(await tec.get_postfilter())
|
||||||
|
print(await tec.get_steinhart_hart())
|
||||||
|
async for data in tec.report_mode():
|
||||||
|
print(data)
|
||||||
|
|
||||||
|
asyncio.run(main())
|
|
@ -0,0 +1,181 @@
|
||||||
|
import asyncio
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
|
||||||
|
class CommandError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
class Client:
|
||||||
|
def __init__(self):
|
||||||
|
self._reader = None
|
||||||
|
self._writer = None
|
||||||
|
self._command_lock = asyncio.Lock()
|
||||||
|
|
||||||
|
async def connect(self, host='192.168.1.26', port=23, timeout=None):
|
||||||
|
self._reader, self._writer = await asyncio.open_connection(host, port)
|
||||||
|
await self._check_zero_limits()
|
||||||
|
|
||||||
|
async def disconnect(self):
|
||||||
|
self._writer.close()
|
||||||
|
await self._writer.wait_closed()
|
||||||
|
|
||||||
|
async def _check_zero_limits(self):
|
||||||
|
pwm_report = await self.get_pwm()
|
||||||
|
for pwm_channel in pwm_report:
|
||||||
|
for limit in ["max_i_neg", "max_i_pos", "max_v"]:
|
||||||
|
if pwm_channel[limit]["value"] == 0.0:
|
||||||
|
logging.warning("`{}` limit is set to zero on channel {}".format(limit, pwm_channel["channel"]))
|
||||||
|
|
||||||
|
async def _read_line(self):
|
||||||
|
# read 1 line
|
||||||
|
chunk = await self._reader.readline()
|
||||||
|
return chunk.decode('utf-8', errors='ignore')
|
||||||
|
|
||||||
|
async def _command(self, *command):
|
||||||
|
async with self._command_lock:
|
||||||
|
self._writer.write(((" ".join(command)).strip() + "\n").encode('utf-8'))
|
||||||
|
await self._writer.drain()
|
||||||
|
|
||||||
|
line = await self._read_line()
|
||||||
|
|
||||||
|
response = json.loads(line)
|
||||||
|
logging.debug(f"{command}: {response}")
|
||||||
|
if "error" in response:
|
||||||
|
raise CommandError(response["error"])
|
||||||
|
return response
|
||||||
|
|
||||||
|
async def _get_conf(self, topic):
|
||||||
|
result = [None, None]
|
||||||
|
for item in await self._command(topic):
|
||||||
|
result[int(item["channel"])] = item
|
||||||
|
return result
|
||||||
|
|
||||||
|
async def get_pwm(self):
|
||||||
|
"""Retrieve PWM limits for the TEC
|
||||||
|
|
||||||
|
Example::
|
||||||
|
[{'channel': 0,
|
||||||
|
'center': 'vref',
|
||||||
|
'i_set': {'max': 2.9802790335151985, 'value': -0.02002179650216762},
|
||||||
|
'max_i_neg': {'max': 3.0, 'value': 3.0},
|
||||||
|
'max_v': {'max': 5.988, 'value': 5.988},
|
||||||
|
'max_i_pos': {'max': 3.0, 'value': 3.0}},
|
||||||
|
{'channel': 1,
|
||||||
|
'center': 'vref',
|
||||||
|
'i_set': {'max': 2.9802790335151985, 'value': -0.02002179650216762},
|
||||||
|
'max_i_neg': {'max': 3.0, 'value': 3.0},
|
||||||
|
'max_v': {'max': 5.988, 'value': 5.988},
|
||||||
|
'max_i_pos': {'max': 3.0, 'value': 3.0}}
|
||||||
|
]
|
||||||
|
"""
|
||||||
|
return await self._get_conf("pwm")
|
||||||
|
|
||||||
|
async def get_pid(self):
|
||||||
|
"""Retrieve PID control state
|
||||||
|
|
||||||
|
Example::
|
||||||
|
[{'channel': 0,
|
||||||
|
'parameters': {
|
||||||
|
'kp': 10.0,
|
||||||
|
'ki': 0.02,
|
||||||
|
'kd': 0.0,
|
||||||
|
'output_min': 0.0,
|
||||||
|
'output_max': 3.0},
|
||||||
|
'target': 37.0},
|
||||||
|
{'channel': 1,
|
||||||
|
'parameters': {
|
||||||
|
'kp': 10.0,
|
||||||
|
'ki': 0.02,
|
||||||
|
'kd': 0.0,
|
||||||
|
'output_min': 0.0,
|
||||||
|
'output_max': 3.0},
|
||||||
|
'target': 36.5}]
|
||||||
|
"""
|
||||||
|
return await self._get_conf("pid")
|
||||||
|
|
||||||
|
async def get_steinhart_hart(self):
|
||||||
|
"""Retrieve Steinhart-Hart parameters for resistance to temperature conversion
|
||||||
|
|
||||||
|
Example::
|
||||||
|
[{'params': {'b': 3800.0, 'r0': 10000.0, 't0': 298.15}, 'channel': 0},
|
||||||
|
{'params': {'b': 3800.0, 'r0': 10000.0, 't0': 298.15}, 'channel': 1}]
|
||||||
|
"""
|
||||||
|
return await self._get_conf("s-h")
|
||||||
|
|
||||||
|
async def get_postfilter(self):
|
||||||
|
"""Retrieve DAC postfilter configuration
|
||||||
|
|
||||||
|
Example::
|
||||||
|
[{'rate': None, 'channel': 0},
|
||||||
|
{'rate': 21.25, 'channel': 1}]
|
||||||
|
"""
|
||||||
|
return await self._get_conf("postfilter")
|
||||||
|
|
||||||
|
async def report_mode(self):
|
||||||
|
"""Start reporting measurement values
|
||||||
|
|
||||||
|
Example of yielded data::
|
||||||
|
{'channel': 0,
|
||||||
|
'time': 2302524,
|
||||||
|
'adc': 0.6199188965423515,
|
||||||
|
'sens': 6138.519310282602,
|
||||||
|
'temperature': 36.87032392655527,
|
||||||
|
'pid_engaged': True,
|
||||||
|
'i_set': 2.0635816680889123,
|
||||||
|
'vref': 1.494,
|
||||||
|
'dac_value': 2.527790834044456,
|
||||||
|
'dac_feedback': 2.523,
|
||||||
|
'i_tec': 2.331,
|
||||||
|
'tec_i': 2.0925,
|
||||||
|
'tec_u_meas': 2.5340000000000003,
|
||||||
|
'pid_output': 2.067581958092247}
|
||||||
|
"""
|
||||||
|
await self._command("report mode", "on")
|
||||||
|
|
||||||
|
while True:
|
||||||
|
line = await self._read_line()
|
||||||
|
if not line:
|
||||||
|
break
|
||||||
|
try:
|
||||||
|
yield json.loads(line)
|
||||||
|
except json.decoder.JSONDecodeError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
async def set_param(self, topic, channel, field="", value=""):
|
||||||
|
"""Set configuration parameters
|
||||||
|
|
||||||
|
Examples::
|
||||||
|
tec.set_param("pwm", 0, "max_v", 2.0)
|
||||||
|
tec.set_param("pid", 1, "output_max", 2.5)
|
||||||
|
tec.set_param("s-h", 0, "t0", 20.0)
|
||||||
|
tec.set_param("center", 0, "vref")
|
||||||
|
tec.set_param("postfilter", 1, 21)
|
||||||
|
|
||||||
|
See the firmware's README.md for a full list.
|
||||||
|
"""
|
||||||
|
if type(value) is float:
|
||||||
|
value = "{:f}".format(value)
|
||||||
|
if type(value) is not str:
|
||||||
|
value = str(value)
|
||||||
|
await self._command(topic, str(channel), field, value)
|
||||||
|
|
||||||
|
async def power_up(self, channel, target):
|
||||||
|
"""Start closed-loop mode"""
|
||||||
|
await self.set_param("pid", channel, "target", value=target)
|
||||||
|
await self.set_param("pwm", channel, "pid")
|
||||||
|
|
||||||
|
async def save_config(self):
|
||||||
|
"""Save current configuration to EEPROM"""
|
||||||
|
await self._command("save")
|
||||||
|
|
||||||
|
async def load_config(self):
|
||||||
|
"""Load current configuration from EEPROM"""
|
||||||
|
await self._command("load")
|
||||||
|
|
||||||
|
async def hw_rev(self):
|
||||||
|
"""Get Thermostat hardware revision"""
|
||||||
|
return await self._command("hwrev")
|
||||||
|
|
||||||
|
async def fan(self):
|
||||||
|
"""Get Thermostat current fan settings"""
|
||||||
|
return await self._command("fan")
|
|
@ -9,4 +9,10 @@ setup(
|
||||||
license="GPLv3",
|
license="GPLv3",
|
||||||
install_requires=["setuptools"],
|
install_requires=["setuptools"],
|
||||||
packages=find_packages(),
|
packages=find_packages(),
|
||||||
|
entry_points={
|
||||||
|
"gui_scripts": [
|
||||||
|
"tec_qt = tec_qt:main",
|
||||||
|
]
|
||||||
|
},
|
||||||
|
py_modules=['tec_qt', 'ui_tec_qt'],
|
||||||
)
|
)
|
||||||
|
|
176
pytec/tec_qt.py
176
pytec/tec_qt.py
|
@ -1,5 +1,5 @@
|
||||||
from PyQt6 import QtWidgets, uic
|
from PyQt6 import QtWidgets, uic
|
||||||
from PyQt6.QtCore import QThread, QThreadPool, pyqtSignal, QRunnable, QObject, QSignalBlocker, pyqtSlot, QDeadlineTimer
|
from PyQt6.QtCore import pyqtSignal, QObject, QSignalBlocker, pyqtSlot
|
||||||
from pyqtgraph import PlotWidget
|
from pyqtgraph import PlotWidget
|
||||||
from pyqtgraph.parametertree import Parameter, ParameterTree, ParameterItem, registerParameterType
|
from pyqtgraph.parametertree import Parameter, ParameterTree, ParameterItem, registerParameterType
|
||||||
import pyqtgraph as pg
|
import pyqtgraph as pg
|
||||||
|
@ -7,9 +7,9 @@ import sys
|
||||||
import argparse
|
import argparse
|
||||||
import logging
|
import logging
|
||||||
import asyncio
|
import asyncio
|
||||||
import atexit
|
from pytec.aioclient import Client
|
||||||
from pytec.client import Client
|
import qasync
|
||||||
from qasync import QEventLoop
|
from qasync import asyncSlot, asyncClose
|
||||||
|
|
||||||
# pyuic6 -x tec_qt.ui -o ui_tec_qt.py
|
# pyuic6 -x tec_qt.ui -o ui_tec_qt.py
|
||||||
from ui_tec_qt import Ui_MainWindow
|
from ui_tec_qt import Ui_MainWindow
|
||||||
|
@ -19,9 +19,8 @@ tec_client: Client = None
|
||||||
# ui = None
|
# ui = None
|
||||||
ui: Ui_MainWindow = None
|
ui: Ui_MainWindow = None
|
||||||
|
|
||||||
queue = None
|
|
||||||
connection_watcher = None
|
|
||||||
client_watcher = None
|
client_watcher = None
|
||||||
|
client_watcher_task = None
|
||||||
app: QtWidgets.QApplication = None
|
app: QtWidgets.QApplication = None
|
||||||
|
|
||||||
|
|
||||||
|
@ -38,80 +37,6 @@ def get_argparser():
|
||||||
return parser
|
return parser
|
||||||
|
|
||||||
|
|
||||||
def wrap_client_task(func, *args, **kwargs):
|
|
||||||
loop = asyncio.get_event_loop()
|
|
||||||
task = ClientTask(func, *args, **kwargs)
|
|
||||||
asyncio.ensure_future(queue.put(task), loop=loop)
|
|
||||||
|
|
||||||
|
|
||||||
async def process_client_tasks():
|
|
||||||
global queue
|
|
||||||
if queue is None:
|
|
||||||
queue = asyncio.Queue()
|
|
||||||
loop = asyncio.get_event_loop()
|
|
||||||
while True:
|
|
||||||
task = await queue.get()
|
|
||||||
await task.run()
|
|
||||||
queue.task_done()
|
|
||||||
|
|
||||||
|
|
||||||
class ClientTask:
|
|
||||||
def __init__(self, func, *args, **kwargs):
|
|
||||||
self.func = func
|
|
||||||
self.args = args
|
|
||||||
self.kwargs = kwargs
|
|
||||||
super().__init__()
|
|
||||||
|
|
||||||
async def run(self):
|
|
||||||
try:
|
|
||||||
lock = asyncio.Lock()
|
|
||||||
async with lock:
|
|
||||||
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()
|
|
||||||
|
|
||||||
|
|
||||||
class WatchConnectTask(QObject):
|
|
||||||
connected = pyqtSignal(bool)
|
|
||||||
hw_rev = pyqtSignal(dict)
|
|
||||||
connecting = pyqtSignal()
|
|
||||||
fan_update = pyqtSignal(object)
|
|
||||||
|
|
||||||
def __init__(self, parent, ip, port):
|
|
||||||
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)
|
|
||||||
wrap_client_task(lambda: self.hw_rev.emit(tec_client.hw_rev()))
|
|
||||||
# wrap_client_task(lambda: self.fan_update.emit(tec_client.fan()))
|
|
||||||
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
|
|
||||||
self.connected.emit(False)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class ClientWatcher(QObject):
|
class ClientWatcher(QObject):
|
||||||
fan_update = pyqtSignal(object)
|
fan_update = pyqtSignal(object)
|
||||||
pwm_update = pyqtSignal(object)
|
pwm_update = pyqtSignal(object)
|
||||||
|
@ -125,19 +50,15 @@ class ClientWatcher(QObject):
|
||||||
|
|
||||||
async def run(self):
|
async def run(self):
|
||||||
while self.running:
|
while self.running:
|
||||||
wrap_client_task(lambda: self.update_params())
|
await self.update_params()
|
||||||
await asyncio.sleep(int(self.update_s * 1000))
|
await asyncio.sleep(self.update_s)
|
||||||
|
|
||||||
def update_params(self):
|
async def update_params(self):
|
||||||
self.fan_update.emit(tec_client.fan())
|
self.fan_update.emit(await tec_client.fan())
|
||||||
|
|
||||||
@pyqtSlot()
|
@pyqtSlot()
|
||||||
def stop_watching(self):
|
def stop_watching(self):
|
||||||
self.running = False
|
self.running = False
|
||||||
#deadline = QDeadlineTimer()
|
|
||||||
#deadline.setDeadline(100)
|
|
||||||
#self.wait(deadline)
|
|
||||||
#self.terminate()
|
|
||||||
|
|
||||||
@pyqtSlot()
|
@pyqtSlot()
|
||||||
def set_update_s(self):
|
def set_update_s(self):
|
||||||
|
@ -145,7 +66,7 @@ class ClientWatcher(QObject):
|
||||||
|
|
||||||
|
|
||||||
def on_connection_changed(result):
|
def on_connection_changed(result):
|
||||||
global client_watcher, connection_watcher
|
global client_watcher, client_watcher_task
|
||||||
ui.graph_group.setEnabled(result)
|
ui.graph_group.setEnabled(result)
|
||||||
ui.hw_rev_lbl.setEnabled(result)
|
ui.hw_rev_lbl.setEnabled(result)
|
||||||
ui.fan_group.setEnabled(result)
|
ui.fan_group.setEnabled(result)
|
||||||
|
@ -161,12 +82,7 @@ def on_connection_changed(result):
|
||||||
if client_watcher:
|
if client_watcher:
|
||||||
client_watcher.stop_watching()
|
client_watcher.stop_watching()
|
||||||
client_watcher = None
|
client_watcher = None
|
||||||
elif client_watcher is None:
|
client_watcher_task = 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)
|
|
||||||
wrap_client_task(client_watcher.run)
|
|
||||||
|
|
||||||
|
|
||||||
def hw_rev(hw_rev_d: dict):
|
def hw_rev(hw_rev_d: dict):
|
||||||
|
@ -192,55 +108,73 @@ def fan_update(fan_settings):
|
||||||
ui.fan_auto_box.setChecked(fan_settings["auto_mode"])
|
ui.fan_auto_box.setChecked(fan_settings["auto_mode"])
|
||||||
|
|
||||||
|
|
||||||
def fan_set():
|
@asyncSlot()
|
||||||
|
async def fan_set(_):
|
||||||
global tec_client
|
global tec_client
|
||||||
if tec_client is None or ui.fan_auto_box.isChecked():
|
if tec_client is None or ui.fan_auto_box.isChecked():
|
||||||
return
|
return
|
||||||
wrap_client_task(lambda: tec_client.set_param("fan", ui.fan_power_slider.value()))
|
await tec_client.set_param("fan", ui.fan_power_slider.value())
|
||||||
|
|
||||||
|
|
||||||
def fan_auto_set(enabled):
|
@asyncSlot()
|
||||||
|
async def fan_auto_set(enabled):
|
||||||
global tec_client
|
global tec_client
|
||||||
if tec_client is None:
|
if tec_client is None:
|
||||||
return
|
return
|
||||||
ui.fan_power_slider.setEnabled(not enabled)
|
ui.fan_power_slider.setEnabled(not enabled)
|
||||||
if enabled:
|
if enabled:
|
||||||
wrap_client_task(lambda: tec_client.set_param("fan", "auto"))
|
await tec_client.set_param("fan", "auto")
|
||||||
else:
|
else:
|
||||||
wrap_client_task(lambda: tec_client.set_param("fan", ui.fan_power_slider.value()))
|
await tec_client.set_param("fan", ui.fan_power_slider.value())
|
||||||
|
|
||||||
|
|
||||||
def connect():
|
@asyncSlot()
|
||||||
global connection_watcher
|
async def connect(_):
|
||||||
connection_watcher = WatchConnectTask(ui.main_widget, ui.ip_set_line.text(), ui.port_set_spin.value())
|
global tec_client, client_watcher, client_watcher_task
|
||||||
connection_watcher.connected.connect(on_connection_changed)
|
ip, port = ui.ip_set_line.text(), ui.port_set_spin.value()
|
||||||
connection_watcher.connecting.connect(lambda: ui.status_lbl.setText("Connecting..."))
|
try:
|
||||||
connection_watcher.hw_rev.connect(hw_rev)
|
if tec_client:
|
||||||
connection_watcher.fan_update.connect(fan_update)
|
await tec_client.disconnect()
|
||||||
wrap_client_task(connection_watcher.run)
|
tec_client = None
|
||||||
#app.aboutToQuit.connect(connection_watcher.terminate)
|
on_connection_changed(False)
|
||||||
|
else:
|
||||||
|
ui.status_lbl.setText("Connecting...")
|
||||||
|
tec_client = Client()
|
||||||
|
await tec_client.connect(host=ip, port=port, timeout=30)
|
||||||
|
on_connection_changed(True)
|
||||||
|
hw_rev(await tec_client.hw_rev())
|
||||||
|
# fan_update(await tec_client.fan())
|
||||||
|
if 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(
|
||||||
|
lambda: client_watcher.set_update_s(ui.report_refresh_spin.value())
|
||||||
|
)
|
||||||
|
app.aboutToQuit.connect(client_watcher.stop_watching)
|
||||||
|
client_watcher_task = asyncio.create_task(client_watcher.run())
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"Failed communicating to the {ip}:{port}: {e}")
|
||||||
|
on_connection_changed(False)
|
||||||
|
|
||||||
|
|
||||||
def main():
|
async def coro_main():
|
||||||
global ui, app, queue
|
global ui, app
|
||||||
|
|
||||||
args = get_argparser().parse_args()
|
args = get_argparser().parse_args()
|
||||||
if args.logLevel:
|
if args.logLevel:
|
||||||
logging.basicConfig(level=getattr(logging, args.logLevel))
|
logging.basicConfig(level=getattr(logging, args.logLevel))
|
||||||
|
|
||||||
app = QtWidgets.QApplication(sys.argv)
|
app_quit_event = asyncio.Event()
|
||||||
|
|
||||||
loop = QEventLoop(app)
|
app = QtWidgets.QApplication.instance()
|
||||||
asyncio.set_event_loop(loop)
|
app.aboutToQuit.connect(app_quit_event.set)
|
||||||
atexit.register(loop.close)
|
|
||||||
|
|
||||||
loop.create_task(process_client_tasks())
|
|
||||||
|
|
||||||
main_window = QtWidgets.QMainWindow()
|
main_window = QtWidgets.QMainWindow()
|
||||||
ui = Ui_MainWindow()
|
ui = Ui_MainWindow()
|
||||||
ui.setupUi(main_window)
|
ui.setupUi(main_window)
|
||||||
# ui = uic.loadUi('tec_qt.ui', main_window)
|
# ui = uic.loadUi('tec_qt.ui', main_window)
|
||||||
|
|
||||||
ui.connect_btn.clicked.connect(lambda _checked: connect())
|
ui.connect_btn.clicked.connect(connect)
|
||||||
ui.fan_power_slider.valueChanged.connect(fan_set)
|
ui.fan_power_slider.valueChanged.connect(fan_set)
|
||||||
ui.fan_auto_box.stateChanged.connect(fan_auto_set)
|
ui.fan_auto_box.stateChanged.connect(fan_auto_set)
|
||||||
|
|
||||||
|
@ -253,7 +187,11 @@ def main():
|
||||||
|
|
||||||
main_window.show()
|
main_window.show()
|
||||||
|
|
||||||
loop.run_until_complete(app.exec())
|
await app_quit_event.wait()
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
qasync.run(coro_main())
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
|
Loading…
Reference in New Issue