// 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 if (data !== null && typeof (data) !== "undefined") { // if it has a second parameter but it's not a type we know, then fire this: console.log("unsupported type of input argument passed, try to not include this argument to the 'Emit'"); } return this._msg(event, t, m); } private decodeMessage(event: string, websocketMessage: string): T | any { //iris-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 iris 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 = 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 iris-custom websocket message Emit(event: string, data: any): void { let messageStr = this.encodeMessage(event, data); this.EmitMessage(messageStr); } // } // node-modules export {Ws};