summaryrefslogtreecommitdiff
path: root/kvmd/apps/vnc
diff options
context:
space:
mode:
authorDevaev Maxim <[email protected]>2020-05-24 11:41:38 +0300
committerDevaev Maxim <[email protected]>2020-05-24 11:41:38 +0300
commitcf47e0c8805a27ef024f5024a75957ea159b0328 (patch)
tree93a6684b9432e219b36b6ac387d3b0bee53d98c7 /kvmd/apps/vnc
parent6d7351502e268155c00cd0eb6b3bba37054f761d (diff)
commond kvmd ws client
Diffstat (limited to 'kvmd/apps/vnc')
-rw-r--r--kvmd/apps/vnc/server.py94
1 files changed, 28 insertions, 66 deletions
diff --git a/kvmd/apps/vnc/server.py b/kvmd/apps/vnc/server.py
index 270072db..c7d376d8 100644
--- a/kvmd/apps/vnc/server.py
+++ b/kvmd/apps/vnc/server.py
@@ -22,11 +22,9 @@
import os
import asyncio
-import asyncio.queues
import socket
import dataclasses
import contextlib
-import json
from typing import Dict
from typing import Optional
@@ -38,14 +36,13 @@ from ...logging import get_logger
from ...keyboard.keysym import SymmapWebKey
from ...keyboard.keysym import build_symmap
+from ...clients.kvmd import KvmdClientWs
from ...clients.kvmd import KvmdClientSession
from ...clients.kvmd import KvmdClient
from ...clients.streamer import StreamerError
from ...clients.streamer import StreamerClient
-from ... import aiotools
-
from .rfb import RfbClient
from .rfb.stream import rfb_format_remote
from .rfb.stream import rfb_close_writer
@@ -106,10 +103,10 @@ class _Client(RfbClient): # pylint: disable=too-many-instance-attributes
self.__shared_params = shared_params
- self.__kvmd_session: Optional[KvmdClientSession] = None
self.__authorized = asyncio.Future() # type: ignore
self.__ws_connected = asyncio.Future() # type: ignore
- self.__ws_writer_queue: asyncio.queues.Queue = asyncio.Queue()
+ self.__kvmd_session: Optional[KvmdClientSession] = None
+ self.__kvmd_ws: Optional[KvmdClientWs] = None
self.__fb_requested = False
self.__fb_stub_text = ""
@@ -133,48 +130,23 @@ class _Client(RfbClient): # pylint: disable=too-many-instance-attributes
finally:
if self.__kvmd_session:
await self.__kvmd_session.close()
+ self.__kvmd_session = None
# =====
async def __kvmd_task_loop(self) -> None:
logger = get_logger(0)
-
await self.__authorized
assert self.__kvmd_session
-
- async with self.__kvmd_session.ws() as ws:
- logger.info("[kvmd] Client %s: Connected to KVMD websocket", self._remote)
- self.__ws_connected.set_result(None)
-
- receive_task: Optional[asyncio.Task] = None
- writer_task: Optional[asyncio.Task] = None
- try:
- while True:
- if receive_task is None:
- receive_task = asyncio.create_task(ws.receive())
- if writer_task is None:
- writer_task = asyncio.create_task(self.__ws_writer_queue.get())
-
- done = (await aiotools.wait_first(receive_task, writer_task))[0]
-
- if receive_task in done:
- msg = receive_task.result()
- if msg.type == aiohttp.WSMsgType.TEXT:
- await self.__process_ws_event(json.loads(msg.data))
- elif msg.type == aiohttp.WSMsgType.CLOSE:
- raise RfbError("KVMD closed the wesocket (it may have been stopped)")
- else:
- raise RuntimeError(f"Unhandled WS message type: {msg!r}")
- receive_task = None
-
- if writer_task in done:
- await ws.send_str(json.dumps(writer_task.result()))
- writer_task = None
- finally:
- if receive_task:
- receive_task.cancel()
- if writer_task:
- writer_task.cancel()
+ try:
+ async with self.__kvmd_session.ws() as self.__kvmd_ws:
+ logger.info("[kvmd] Client %s: Connected to KVMD websocket", self._remote)
+ self.__ws_connected.set_result(None)
+ async for event in self.__kvmd_ws.communicate():
+ await self.__process_ws_event(event)
+ raise RfbError("KVMD closes the websocket (the server may have been stopped)")
+ finally:
+ self.__kvmd_ws = None
async def __process_ws_event(self, event: Dict) -> None:
if event["event_type"] == "info_state":
@@ -262,33 +234,23 @@ class _Client(RfbClient): # pylint: disable=too-many-instance-attributes
# =====
async def _on_key_event(self, code: int, state: bool) -> None:
- if (web_key := self.__symmap.get(code)) is not None:
- await self.__ws_writer_queue.put({
- "event_type": "key",
- "event": {"key": web_key.name, "state": state},
- })
+ if self.__kvmd_ws:
+ if (web_key := self.__symmap.get(code)) is not None:
+ await self.__kvmd_ws.send_key_event(web_key.name, state)
async def _on_pointer_event(self, buttons: Dict[str, bool], wheel: Dict[str, int], move: Dict[str, int]) -> None:
- for (button, state) in buttons.items():
- if self.__mouse_buttons[button] != state:
- await self.__ws_writer_queue.put({
- "event_type": "mouse_button",
- "event": {"button": button, "state": state},
- })
- self.__mouse_buttons[button] = state
-
- if wheel["x"] or wheel["y"]:
- await self.__ws_writer_queue.put({
- "event_type": "mouse_wheel",
- "event": {"delta": wheel},
- })
-
- if self.__mouse_move != move:
- await self.__ws_writer_queue.put({
- "event_type": "mouse_move",
- "event": {"to": move},
- })
- self.__mouse_move = move
+ if self.__kvmd_ws:
+ for (button, state) in buttons.items():
+ if self.__mouse_buttons[button] != state:
+ await self.__kvmd_ws.send_mouse_button_event(button, state)
+ self.__mouse_buttons[button] = state
+
+ if wheel["x"] or wheel["y"]:
+ await self.__kvmd_ws.send_mouse_wheel_event(wheel["x"], wheel["y"])
+
+ if self.__mouse_move != move:
+ await self.__kvmd_ws.send_mouse_move_event(move["x"], move["y"])
+ self.__mouse_move = move
async def _on_cut_event(self, text: str) -> None:
assert self.__authorized.done()