mirror of
https://github.com/kataras/iris.git
synced 2025-01-25 03:31:04 +01:00
4e1d64c5c0
Former-commit-id: 662bdd01a7cd403d1f7b8f0d17bed16ed1f06562
262 lines
8.6 KiB
TypeScript
262 lines
8.6 KiB
TypeScript
// -------------------------------------------------------------------------------------
|
|
// -------------------------------------------------------------------------------------
|
|
// ----------------Client side websocket commented typescript source code --------------
|
|
// -------------------------------------------------------------------------------------
|
|
// -------------------------------------------------------------------------------------
|
|
|
|
// export to client.go:clientSource []byte
|
|
|
|
const websocketStringMessageType = 0;
|
|
const websocketIntMessageType = 1;
|
|
const websocketBoolMessageType = 2;
|
|
// bytes is missing here for reasons I will explain somewhen
|
|
const websocketJSONMessageType = 4;
|
|
|
|
const websocketMessagePrefix = "iris-websocket-message:";
|
|
const websocketMessageSeparator = ";";
|
|
|
|
const websocketMessagePrefixLen = websocketMessagePrefix.length;
|
|
var websocketMessageSeparatorLen = websocketMessageSeparator.length;
|
|
var websocketMessagePrefixAndSepIdx = websocketMessagePrefixLen + websocketMessageSeparatorLen - 1;
|
|
var websocketMessagePrefixIdx = websocketMessagePrefixLen - 1;
|
|
var websocketMessageSeparatorIdx = websocketMessageSeparatorLen - 1;
|
|
|
|
type onConnectFunc = () => void;
|
|
type onWebsocketDisconnectFunc = () => void;
|
|
type onWebsocketNativeMessageFunc = (websocketMessage: string) => void;
|
|
type onMessageFunc = (message: any) => void;
|
|
|
|
class Ws {
|
|
private conn: WebSocket;
|
|
private isReady: boolean;
|
|
|
|
// events listeners
|
|
|
|
private connectListeners: onConnectFunc[] = [];
|
|
private disconnectListeners: onWebsocketDisconnectFunc[] = [];
|
|
private nativeMessageListeners: onWebsocketNativeMessageFunc[] = [];
|
|
private messageListeners: { [event: string]: onMessageFunc[] } = {};
|
|
|
|
//
|
|
|
|
constructor(endpoint: string, protocols?: string[]) {
|
|
if (!window["WebSocket"]) {
|
|
return;
|
|
}
|
|
|
|
if (endpoint.indexOf("ws") == -1) {
|
|
endpoint = "ws://" + endpoint;
|
|
}
|
|
if (protocols != null && protocols.length > 0) {
|
|
this.conn = new WebSocket(endpoint, protocols);
|
|
} else {
|
|
this.conn = new WebSocket(endpoint);
|
|
}
|
|
|
|
this.conn.onopen = ((evt: Event): any => {
|
|
this.fireConnect();
|
|
this.isReady = true;
|
|
return null;
|
|
});
|
|
|
|
this.conn.onclose = ((evt: Event): any => {
|
|
this.fireDisconnect();
|
|
return null;
|
|
});
|
|
|
|
this.conn.onmessage = ((evt: MessageEvent) => {
|
|
this.messageReceivedFromConn(evt);
|
|
});
|
|
}
|
|
|
|
//utils
|
|
|
|
private isNumber(obj: any): boolean {
|
|
return !isNaN(obj - 0) && obj !== null && obj !== "" && obj !== false;
|
|
}
|
|
|
|
private isString(obj: any): boolean {
|
|
return Object.prototype.toString.call(obj) == "[object String]";
|
|
}
|
|
|
|
private isBoolean(obj: any): boolean {
|
|
return typeof obj === 'boolean' ||
|
|
(typeof obj === 'object' && typeof obj.valueOf() === 'boolean');
|
|
}
|
|
|
|
private isJSON(obj: any): boolean {
|
|
return typeof obj === 'object';
|
|
}
|
|
|
|
//
|
|
|
|
// messages
|
|
private _msg(event: string, websocketMessageType: number, dataMessage: string): string {
|
|
|
|
return websocketMessagePrefix + event + websocketMessageSeparator + String(websocketMessageType) + websocketMessageSeparator + dataMessage;
|
|
}
|
|
|
|
private encodeMessage(event: string, data: any): string {
|
|
let m = "";
|
|
let t = 0;
|
|
if (this.isNumber(data)) {
|
|
t = websocketIntMessageType;
|
|
m = data.toString();
|
|
} else if (this.isBoolean(data)) {
|
|
t = websocketBoolMessageType;
|
|
m = data.toString();
|
|
} else if (this.isString(data)) {
|
|
t = websocketStringMessageType;
|
|
m = data.toString();
|
|
} else if (this.isJSON(data)) {
|
|
//propably json-object
|
|
t = websocketJSONMessageType;
|
|
m = JSON.stringify(data);
|
|
} else {
|
|
console.log("Invalid, javascript-side should contains an empty second parameter.");
|
|
}
|
|
|
|
return this._msg(event, t, m);
|
|
}
|
|
|
|
private decodeMessage<T>(event: string, websocketMessage: string): T | any {
|
|
//q-websocket-message;user;4;themarshaledstringfromajsonstruct
|
|
let skipLen = websocketMessagePrefixLen + websocketMessageSeparatorLen + event.length + 2;
|
|
if (websocketMessage.length < skipLen + 1) {
|
|
return null;
|
|
}
|
|
let websocketMessageType = parseInt(websocketMessage.charAt(skipLen - 2));
|
|
let theMessage = websocketMessage.substring(skipLen, websocketMessage.length);
|
|
if (websocketMessageType == websocketIntMessageType) {
|
|
return parseInt(theMessage);
|
|
} else if (websocketMessageType == websocketBoolMessageType) {
|
|
return Boolean(theMessage);
|
|
} else if (websocketMessageType == websocketStringMessageType) {
|
|
return theMessage;
|
|
} else if (websocketMessageType == websocketJSONMessageType) {
|
|
return JSON.parse(theMessage);
|
|
} else {
|
|
return null; // invalid
|
|
}
|
|
}
|
|
|
|
private getWebsocketCustomEvent(websocketMessage: string): string {
|
|
if (websocketMessage.length < websocketMessagePrefixAndSepIdx) {
|
|
return "";
|
|
}
|
|
let s = websocketMessage.substring(websocketMessagePrefixAndSepIdx, websocketMessage.length);
|
|
let evt = s.substring(0, s.indexOf(websocketMessageSeparator));
|
|
|
|
return evt;
|
|
}
|
|
|
|
private getCustomMessage(event: string, websocketMessage: string): string {
|
|
let eventIdx = websocketMessage.indexOf(event + websocketMessageSeparator);
|
|
let s = websocketMessage.substring(eventIdx + event.length + websocketMessageSeparator.length + 2, websocketMessage.length);
|
|
return s;
|
|
}
|
|
|
|
//
|
|
|
|
// Ws Events
|
|
|
|
// messageReceivedFromConn this is the func which decides
|
|
// if it's a native websocket message or a custom qws message
|
|
// if native message then calls the fireNativeMessage
|
|
// else calls the fireMessage
|
|
//
|
|
// remember q gives you the freedom of native websocket messages if you don't want to use this client side at all.
|
|
private messageReceivedFromConn(evt: MessageEvent): void {
|
|
//check if qws message
|
|
let message = <string>evt.data;
|
|
if (message.indexOf(websocketMessagePrefix) != -1) {
|
|
let event = this.getWebsocketCustomEvent(message);
|
|
if (event != "") {
|
|
// it's a custom message
|
|
this.fireMessage(event, this.getCustomMessage(event, message));
|
|
return;
|
|
}
|
|
}
|
|
|
|
// it's a native websocket message
|
|
this.fireNativeMessage(message);
|
|
}
|
|
|
|
OnConnect(fn: onConnectFunc): void {
|
|
if (this.isReady) {
|
|
fn();
|
|
}
|
|
this.connectListeners.push(fn);
|
|
}
|
|
|
|
fireConnect(): void {
|
|
for (let i = 0; i < this.connectListeners.length; i++) {
|
|
this.connectListeners[i]();
|
|
}
|
|
}
|
|
|
|
OnDisconnect(fn: onWebsocketDisconnectFunc): void {
|
|
this.disconnectListeners.push(fn);
|
|
}
|
|
|
|
fireDisconnect(): void {
|
|
for (let i = 0; i < this.disconnectListeners.length; i++) {
|
|
this.disconnectListeners[i]();
|
|
}
|
|
}
|
|
|
|
OnMessage(cb: onWebsocketNativeMessageFunc): void {
|
|
this.nativeMessageListeners.push(cb);
|
|
}
|
|
|
|
fireNativeMessage(websocketMessage: string): void {
|
|
for (let i = 0; i < this.nativeMessageListeners.length; i++) {
|
|
this.nativeMessageListeners[i](websocketMessage);
|
|
}
|
|
}
|
|
|
|
On(event: string, cb: onMessageFunc): void {
|
|
if (this.messageListeners[event] == null || this.messageListeners[event] == undefined) {
|
|
this.messageListeners[event] = [];
|
|
}
|
|
this.messageListeners[event].push(cb);
|
|
}
|
|
|
|
fireMessage(event: string, message: any): void {
|
|
for (let key in this.messageListeners) {
|
|
if (this.messageListeners.hasOwnProperty(key)) {
|
|
if (key == event) {
|
|
for (let i = 0; i < this.messageListeners[key].length; i++) {
|
|
this.messageListeners[key][i](message);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
|
|
// Ws Actions
|
|
|
|
Disconnect(): void {
|
|
this.conn.close();
|
|
}
|
|
|
|
// EmitMessage sends a native websocket message
|
|
EmitMessage(websocketMessage: string): void {
|
|
this.conn.send(websocketMessage);
|
|
}
|
|
|
|
// Emit sends an q-custom websocket message
|
|
Emit(event: string, data: any): void {
|
|
let messageStr = this.encodeMessage(event, data);
|
|
this.EmitMessage(messageStr);
|
|
}
|
|
|
|
//
|
|
|
|
}
|
|
|
|
// node-modules export {Ws};
|