summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--kvmd/apps/kvmd/api/ugpio.py27
-rw-r--r--web/kvm/index.html2
-rw-r--r--web/kvm/navbar-macro.pug2
-rw-r--r--web/share/js/kvm/gpio.js14
-rw-r--r--web/share/js/kvm/recorder.js35
-rw-r--r--web/share/js/kvm/session.js2
6 files changed, 40 insertions, 42 deletions
diff --git a/kvmd/apps/kvmd/api/ugpio.py b/kvmd/apps/kvmd/api/ugpio.py
index ddff00f7..444ec83e 100644
--- a/kvmd/apps/kvmd/api/ugpio.py
+++ b/kvmd/apps/kvmd/api/ugpio.py
@@ -19,11 +19,9 @@
# #
# ========================================================================== #
-from typing import Dict
from aiohttp.web import Request
from aiohttp.web import Response
-from aiohttp.web import WebSocketResponse
from ....validators.basic import valid_bool
from ....validators.basic import valid_float_f0
@@ -32,7 +30,6 @@ from ....validators.ugpio import valid_ugpio_channel
from ..ugpio import UserGpio
from ..http import exposed_http
-from ..http import exposed_ws
from ..http import make_json_response
@@ -41,7 +38,7 @@ class UserGpioApi:
def __init__(self, user_gpio: UserGpio) -> None:
self.__user_gpio = user_gpio
- # ===== Http
+ # =====
@exposed_http("GET", "/gpio")
async def __state_handler(self, _: Request) -> Response:
@@ -65,25 +62,3 @@ class UserGpioApi:
wait = valid_bool(request.query.get("wait", "0"))
await self.__user_gpio.pulse(channel, delay, wait)
return make_json_response()
-
- # ===== Websocket
-
- @exposed_ws("gpio_switch")
- async def __ws_gpio_switch_handler(self, _: WebSocketResponse, event: Dict) -> None:
- try:
- channel = valid_ugpio_channel(event["channel"])
- state = valid_bool(event["state"])
- wait = valid_bool(event["wait"])
- except Exception:
- return
- await self.__user_gpio.switch(channel, state, wait)
-
- @exposed_ws("gpio_pulse")
- async def __ws_gpio_pulse_handler(self, _: WebSocketResponse, event: Dict) -> None:
- try:
- channel = valid_ugpio_channel(event["channel"])
- delay = valid_float_f0(event["delay"])
- wait = valid_bool(event["wait"])
- except Exception:
- return
- await self.__user_gpio.pulse(channel, delay, wait)
diff --git a/web/kvm/index.html b/web/kvm/index.html
index 2ec36fd7..7d6a097f 100644
--- a/web/kvm/index.html
+++ b/web/kvm/index.html
@@ -437,7 +437,7 @@
</li>
<li class="right"><a class="menu-button" href="#"><img class="led-gray" data-dont-hide-menu id="hid-recorder-led" src="/share/svg/led-gear.svg">Macro</a>
<div class="menu" data-dont-hide-menu>
- <div class="text"><b>Record and play keyboard &amp; mouse actions<br></b><sub>For security reasons, the record will not be saved on the PiKVM</sub></div>
+ <div class="text"><b>Record and play HID/GPIO actions<br></b><sub>For security reasons, the record will not be saved on the PiKVM</sub></div>
<hr>
<div class="buttons buttons-row">
<button class="row25" disabled data-force-hide-menu id="hid-recorder-record">&bull; Rec</button>
diff --git a/web/kvm/navbar-macro.pug b/web/kvm/navbar-macro.pug
index c9dfc2dc..8880550d 100644
--- a/web/kvm/navbar-macro.pug
+++ b/web/kvm/navbar-macro.pug
@@ -4,7 +4,7 @@ li(class="right")
| Macro
div(data-dont-hide-menu class="menu")
div(class="text")
- b Record and play keyboard &amp; mouse actions#[br]
+ b Record and play HID/GPIO actions#[br]
sub For security reasons, the record will not be saved on the PiKVM
hr
div(class="buttons buttons-row")
diff --git a/web/share/js/kvm/gpio.js b/web/share/js/kvm/gpio.js
index bd9e5dd7..2319f2ac 100644
--- a/web/share/js/kvm/gpio.js
+++ b/web/share/js/kvm/gpio.js
@@ -27,7 +27,7 @@ import {tools, $, $$$} from "../tools.js";
import {wm} from "../wm.js";
-export function Gpio(__recordWsEvent) {
+export function Gpio(__recorder) {
var self = this;
/************************************************************************/
@@ -167,13 +167,9 @@ export function Gpio(__recordWsEvent) {
if (to === "0" && el.hasAttribute("data-confirm-off")) {
confirm = el.getAttribute("data-confirm-off");
}
- let event = {
- "event_type": "gpio_switch",
- "event": {"channel": channel, "state": to, "wait": 0},
- };
let act = () => {
__sendPost(`/api/gpio/switch?channel=${channel}&state=${to}`);
- __recordWsEvent(event);
+ __recorder.recordGpioSwitchEvent(channel, to);
};
if (confirm) {
wm.confirm(confirm).then(function(ok) {
@@ -191,13 +187,9 @@ export function Gpio(__recordWsEvent) {
var __pulseChannel = function(el) {
let channel = el.getAttribute("data-channel");
let confirm = el.getAttribute("data-confirm");
- let event = {
- "event_type": "gpio_pulse",
- "event": {"channel": channel, "delay": 0, "wait": 0},
- };
let act = () => {
__sendPost(`/api/gpio/pulse?channel=${channel}`);
- __recordWsEvent(event);
+ __recorder.recordGpioPulseEvent(channel);
};
if (confirm) {
wm.confirm(confirm).then(function(ok) { if (ok) act(); });
diff --git a/web/share/js/kvm/recorder.js b/web/share/js/kvm/recorder.js
index b1992eb7..f2ca6d5c 100644
--- a/web/share/js/kvm/recorder.js
+++ b/web/share/js/kvm/recorder.js
@@ -71,6 +71,14 @@ export function Recorder() {
__recordEvent({"event_type": "print", "event": {"text": text}});
};
+ self.recordGpioSwitchEvent = function(channel, to) {
+ __recordEvent({"event_type": "gpio_switch", "event": {"channel": channel, "state": to}});
+ };
+
+ self.recordGpioPulseEvent = function(channel) {
+ __recordEvent({"event_type": "gpio_pulse", "event": {"channel": channel}});
+ };
+
var __recordEvent = function(event) {
if (__recording) {
let now = new Date().getTime();
@@ -163,8 +171,13 @@ export function Recorder() {
__checkType(event.event.delta, "object", "Non-object mouse wheel delta");
__checkInt(event.event.delta.x, "Non-int mouse delta X");
__checkInt(event.event.delta.y, "Non-int mouse delta Y");
+ } else if (event.event_type === "gpio_switch") {
+ __checkType(event.event.channel, "string", "Non-string GPIO channel");
+ __checkType(event.event.state, "boolean", "Non-bool GPIO state");
+ } else if (event.event_type === "gpio_pulse") {
+ __checkType(event.event.channel, "string", "Non-string GPIO channel");
} else {
- throw "Unknown event type";
+ throw `Unknown event type: ${event.event_type}`;
}
events.push(event);
@@ -217,7 +230,25 @@ export function Recorder() {
}
}, event.event.text, "text/plain");
return;
- } else if (["key", "mouse_button", "mouse_move", "mouse_wheel", "gpio_switch", "gpio_pulse"].includes(event.event_type)) {
+ } else if (["gpio_switch", "gpio_pulse"].includes(event.event_type)) {
+ let path = "/api/gpio";
+ if (event.event_type === "gpio_switch") {
+ path += `/switch?channel=${event.event.channel}&state=${event.event.to}`;
+ } else { // gpio_pulse
+ path += `/pulse?channel=${event.event.channel}`;
+ }
+ let http = tools.makeRequest("POST", path, function() {
+ if (http.readyState === 4) {
+ if (http.status !== 200) {
+ wm.error("GPIO error:<br>", http.responseText);
+ __stopProcess();
+ } else if (http.status === 200) {
+ __play_timer = setTimeout(() => __runEvents(index + 1, time), 0);
+ }
+ }
+ });
+ return;
+ } else if (["key", "mouse_button", "mouse_move", "mouse_wheel"].includes(event.event_type)) {
__ws.send(JSON.stringify(event));
}
index += 1;
diff --git a/web/share/js/kvm/session.js b/web/share/js/kvm/session.js
index b183730a..3b2f18c1 100644
--- a/web/share/js/kvm/session.js
+++ b/web/share/js/kvm/session.js
@@ -49,7 +49,7 @@ export function Session() {
var __hid = new Hid(__streamer.getResolution, __recorder);
var __atx = new Atx();
var __msd = new Msd();
- var __gpio = new Gpio(__recorder.recordWsEvent);
+ var __gpio = new Gpio(__recorder);
var __init__ = function() {
__startSession();