Files
sexy.pivoine.art/packages/buttplug/src/utils/ButtplugBrowserWebsocketConnector.ts

89 lines
2.8 KiB
TypeScript
Raw Normal View History

2025-10-25 22:04:41 +02:00
/*!
* Buttplug JS Source Code File - Visit https://buttplug.io for more info about
* the project. Licensed under the BSD 3-Clause license. See LICENSE file in the
* project root for full license information.
*
* @copyright Copyright (c) Nonpolynomial Labs LLC. All rights reserved.
*/
'use strict';
2025-10-25 22:04:41 +02:00
import { EventEmitter } from 'eventemitter3';
import { ButtplugMessage } from '../core/Messages';
2025-10-25 22:04:41 +02:00
export class ButtplugBrowserWebsocketConnector extends EventEmitter {
protected _ws: WebSocket | undefined;
protected _websocketConstructor: typeof WebSocket | null = null;
2025-10-25 22:04:41 +02:00
public constructor(private _url: string) {
super();
}
2025-10-25 22:04:41 +02:00
public get Connected(): boolean {
return this._ws !== undefined;
}
2025-10-25 22:04:41 +02:00
public connect = async (): Promise<void> => {
return new Promise<void>((resolve, reject) => {
const ws = new (this._websocketConstructor ?? WebSocket)(this._url);
const onErrorCallback = (event: Event) => {reject(event)}
const onCloseCallback = (event: CloseEvent) => reject(event.reason)
ws.addEventListener('open', async () => {
this._ws = ws;
try {
await this.initialize();
this._ws.addEventListener('message', (msg) => {
this.parseIncomingMessage(msg);
});
this._ws.removeEventListener('close', onCloseCallback);
this._ws.removeEventListener('error', onErrorCallback);
this._ws.addEventListener('close', this.disconnect);
resolve();
} catch (e) {
reject(e);
}
});
// In websockets, our error rarely tells us much, as for security reasons
// browsers usually only throw Error Code 1006. It's up to those using this
// library to state what the problem might be.
2025-10-25 22:04:41 +02:00
ws.addEventListener('error', onErrorCallback)
ws.addEventListener('close', onCloseCallback);
});
};
2025-10-25 22:04:41 +02:00
public disconnect = async (): Promise<void> => {
if (!this.Connected) {
return;
}
this._ws!.close();
this._ws = undefined;
this.emit('disconnect');
};
2025-10-25 22:04:41 +02:00
public sendMessage(msg: ButtplugMessage) {
if (!this.Connected) {
throw new Error('ButtplugBrowserWebsocketConnector not connected');
}
this._ws!.send('[' + JSON.stringify(msg) + ']');
}
2025-10-25 22:04:41 +02:00
public initialize = async (): Promise<void> => {
return Promise.resolve();
};
2025-10-25 22:04:41 +02:00
protected parseIncomingMessage(event: MessageEvent) {
if (typeof event.data === 'string') {
const msgs: ButtplugMessage[] = JSON.parse(event.data);
this.emit('message', msgs);
} else if (event.data instanceof Blob) {
// No-op, we only use text message types.
}
}
2025-10-25 22:04:41 +02:00
protected onReaderLoad(event: Event) {
const msgs: ButtplugMessage[] = JSON.parse((event.target as FileReader).result as string);
this.emit('message', msgs);
}
2025-10-25 22:04:41 +02:00
}