diff --git a/src/components/port-picker/PortOverrideOption.vue b/src/components/port-picker/PortOverrideOption.vue index e3f0c95fe86..7480e4ac404 100644 --- a/src/components/port-picker/PortOverrideOption.vue +++ b/src/components/port-picker/PortOverrideOption.vue @@ -1,8 +1,7 @@ diff --git a/src/js/data_storage.js b/src/js/data_storage.js index efb37b3498e..38b391d387f 100644 --- a/src/js/data_storage.js +++ b/src/js/data_storage.js @@ -11,6 +11,7 @@ const CONFIGURATOR = { connectionValid: false, connectionValidCliOnly: false, + manualMode: false, virtualMode: false, virtualApiVersion: '0.0.1', cliActive: false, diff --git a/src/js/protocols/websocket.js b/src/js/protocols/websocket.js new file mode 100644 index 00000000000..fb8e7bf8fdf --- /dev/null +++ b/src/js/protocols/websocket.js @@ -0,0 +1,147 @@ +import PortHandler from "../port_handler"; + +class WebsocketSerial extends EventTarget { + constructor() { + super(); + + this.connected = false; + this.connectionInfo = null; + + this.bitrate = 0; + this.bytesSent = 0; + this.bytesReceived = 0; + this.failed = 0; + + this.logHead = "[WEBSOCKET] "; + + this.address = "ws://localhost:5761"; + + this.ws = null; + + this.connect = this.connect.bind(this); + } + + handleReceiveBytes(info) { + this.bytesReceived += info.detail.byteLength; + } + + handleDisconnect() { + this.disconnect(); + } + + createPort(url) { + this.address = url; + return { + path: url, + displayName: `Betaflight SITL`, + vendorId: 0, + productId: 0, + port: 0, + }; + } + + getConnectedPort() { + return { + path: this.address, + displayName: `Betaflight SITL`, + vendorId: 0, + productId: 0, + port: 0, + }; + } + + async getDevices() { + return []; + } + + async blob2uint(blob) { + const buffer = await new Response(blob).arrayBuffer(); + return new Uint8Array(buffer); + } + + waitForConnection(socket) { + return new Promise((resolve) => { + const interval = setInterval(() => { + if (socket.connected) { + clearInterval(interval); // Stop checking + resolve(); // Resolve the promise + } + }, 100); // Check every 100ms, adjust as needed + }); + } + + async connect(path, options) { + this.address = path; + console.log(`${this.logHead} Connecting to ${this.address}`); + + this.ws = new WebSocket(this.address, "wsSerial"); + let socket = this; + + this.ws.onopen = function(e) { + console.log(`${socket.logHead} Connected: `, e); + socket.connected = true; + socket.dispatchEvent( + new CustomEvent("connect", { detail: { + socketId: socket.address, + }}), + ); + }; + + await this.waitForConnection(socket); + + this.ws.onclose = function(e) { + console.log(`${socket.logHead} Connection closed: `, e); + + socket.disconnect(() => { + socket.dispatchEvent(new CustomEvent("disconnect", this.disconnect.bind(this))); + }); + }; + + this.ws.onerror = function(e) { + console.error(`${socket.logHead} Connection error: `, e); + + socket.disconnect(() => { + socket.dispatchEvent(new CustomEvent("disconnect", this.disconnect.bind(this))); + }); + }; + + this.ws.onmessage = async function(msg) { + let uint8Chunk = await socket.blob2uint(msg.data); + socket.dispatchEvent( + new CustomEvent("receive", { detail: uint8Chunk }), + ); + }; + } + + async disconnect() { + this.connected = false; + this.bytesReceived = 0; + this.bytesSent = 0; + + if (this.ws) { + try { + this.ws.close(); + } catch (e) { + console.error(`${this.logHead}Failed to close socket: ${e}`); + } + } + } + + async send(data, cb) { + if (this.ws) { + try { + this.ws.send(data); + this.bytesSent += data.byteLength; + } + catch(e) { + console.error(`${this.logHead}Failed to send data e: ${e}`); + } + } + + return { + bytesSent: data.byteLength, + }; + } +} + +export default new WebsocketSerial(); diff --git a/src/js/serial_backend.js b/src/js/serial_backend.js index bc95ca4ffc1..bc96fdd6868 100644 --- a/src/js/serial_backend.js +++ b/src/js/serial_backend.js @@ -107,7 +107,6 @@ function connectDisconnect() { GUI.configuration_loaded = false; const baudRate = PortHandler.portPicker.selectedBauds; - const selectedPort = portName; if (!isConnected) { // prevent connection when we do not have permission @@ -124,6 +123,7 @@ function connectDisconnect() { CONFIGURATOR.virtualMode = selectedPort === 'virtual'; CONFIGURATOR.bluetoothMode = selectedPort.startsWith('bluetooth'); + CONFIGURATOR.manualMode = selectedPort === 'manual'; if (CONFIGURATOR.virtualMode) { CONFIGURATOR.virtualApiVersion = PortHandler.portPicker.virtualMspVersion; @@ -131,6 +131,16 @@ function connectDisconnect() { // Hack to get virtual working on the web serial = serialShim(); serial.connect(onOpenVirtual); + } else if (selectedPort === 'manual') { + serial = serialShim(); + // Explicitly disconnect the event listeners before attaching the new ones. + serial.removeEventListener('connect', connectHandler); + serial.addEventListener('connect', connectHandler); + + serial.removeEventListener('disconnect', disconnectHandler); + serial.addEventListener('disconnect', disconnectHandler); + + serial.connect(portName, { baudRate }); } else { CONFIGURATOR.virtualMode = false; serial = serialShim(); diff --git a/src/js/serial_shim.js b/src/js/serial_shim.js index dee494a6392..0a52ce56b2c 100644 --- a/src/js/serial_shim.js +++ b/src/js/serial_shim.js @@ -1,6 +1,14 @@ import CONFIGURATOR from "./data_storage"; import serialWeb from "./webSerial.js"; import BT from "./protocols/bluetooth.js"; +import websocketSerial from "./protocols/websocket.js"; import virtualSerial from "./virtualSerial.js"; -export let serialShim = () => CONFIGURATOR.virtualMode ? virtualSerial: CONFIGURATOR.bluetoothMode ? BT : serialWeb; +export let serialShim = () => + CONFIGURATOR.virtualMode + ? virtualSerial + : CONFIGURATOR.manualMode + ? websocketSerial + : CONFIGURATOR.bluetoothMode + ? BT + : serialWeb;