import Websock from './websock'; import * as message from './message.js'; import * as rendezvous from './rendezvous.js'; import { loadVp9, loadOpus } from './codec'; import * as sha256 from "fast-sha256"; import * as globals from './globals'; const PORT = 21116; const HOST = 'rs-sg.rustdesk.com'; const licenceKey = ''; const SCHEMA = 'ws://'; type MsgboxCallback = (type: string, title: string, text: string) => void; export default class Connection { _msgs: any[]; _ws: Websock | undefined; _interval: any; _id: string; _hash: message.Hash | undefined; _msgbox: MsgboxCallback | undefined; _peerInfo: message.PeerInfo | undefined; _firstFrame: Boolean | undefined; constructor() { this._msgs = []; this._id = ''; this._interval = setInterval(() => { while (this._msgs.length) { this._ws?.sendMessage(this._msgs[0]); this._msgs.splice(0, 1); } }, 1); } async start(id: string) { const uri = getDefaultUri(); const ws = new Websock(uri); this._ws = ws; this._id = id; console.log(new Date() + ': Conntecting to rendezvoous server: ' + uri); await ws.open(); console.log(new Date() + ': Connected to rendezvoous server'); const connType = rendezvous.ConnType.DEFAULT_CONN; const natType = rendezvous.NatType.SYMMETRIC; const punchHoleRequest = rendezvous.PunchHoleRequest.fromPartial({ id, licenceKey, connType, natType, }); ws.sendRendezvous({ punchHoleRequest }); const msg = ws.parseRendezvous(await ws.next()); ws.close(); console.log(new Date() + ': Got relay response'); const phr = msg.punchHoleResponse; const rr = msg.relayResponse; if (phr) { if (phr.failure != rendezvous.PunchHoleResponse_Failure.UNKNOWN) { switch (phr?.failure) { case rendezvous.PunchHoleResponse_Failure.ID_NOT_EXIST: this.msgbox('error', 'Error', 'ID does not exist'); break; case rendezvous.PunchHoleResponse_Failure.OFFLINE: this.msgbox('error', 'Error', 'Remote desktop is offline'); break; case rendezvous.PunchHoleResponse_Failure.LICENSE_MISMATCH: this.msgbox('error', 'Error', 'Key mismatch'); break; default: if (phr?.otherFailure) { this.msgbox('error', 'Error', phr?.otherFailure); } } } } else if (rr) { await this.connectRelay(rr); } } async connectRelay(rr: rendezvous.RelayResponse) { const pk = rr.pk; let uri = rr.relayServer; if (uri) { uri = getrUriFromRs(uri); } else { uri = getDefaultUri(true); } const uuid = rr.uuid; console.log(new Date() + ': Connecting to relay server: ' + uri); const ws = new Websock(uri); await ws.open(); console.log(new Date() + ': Connected to relay server'); this._ws = ws; const requestRelay = rendezvous.RequestRelay.fromPartial({ licenceKey, uuid, }); ws.sendRendezvous({ requestRelay }); await this.secure(pk); await this.msgLoop(); } async secure(pk: Uint8Array | undefined) { if (pk) { const RS_PK = 'OeVuKk5nlHiXp+APNn0Y3pC1Iwpwn44JGqrQCsWqmBw='; try { pk = await globals.verify(pk, RS_PK).catch(); if (pk?.length != 32) { pk = undefined; } } catch (e) { console.error(e); pk = undefined; } if (!pk) console.error('Handshake failed: invalid public key from rendezvous server'); } if (!pk) { // send an empty message out in case server is setting up secure and waiting for first message await this._ws?.sendMessage({}); return; } const msg = this._ws?.parseMessage(await this._ws?.next()); let signedId: any = msg?.signedId; if (!signedId) { console.error("Handshake failed: invalid message type"); await this._ws?.sendMessage({}); return; } try { signedId = await globals.verify(signedId.id, Uint8Array.from(pk!)); } catch (e) { console.error(e); // fall back to non-secure connection in case pk mismatch console.error("pk mismatch, fall back to non-secure"); const publicKey = message.PublicKey.fromPartial({}); await this._ws?.sendMessage({ publicKey }); return; } signedId = new TextDecoder().decode(signedId!); const tmp = signedId.split('\0'); const id = tmp[0]; let theirPk = tmp[1]; if (id != this._id!) { console.error("Handshake failed: sign failure"); await this._ws?.sendMessage({}); return; } theirPk = globals.decodeBase64(theirPk); if (theirPk.length != 32) { console.error("Handshake failed: invalid public box key length from peer"); await this._ws?.sendMessage({}); return; } const [mySk, asymmetricValue] = globals.genBoxKeyPair(); const secretKey = globals.genSecretKey(); const symmetricValue = globals.seal(secretKey, theirPk, mySk); const publicKey = message.PublicKey.fromPartial({ asymmetricValue, symmetricValue }); await this._ws?.sendMessage({ publicKey }); this._ws?.setSecretKey(secretKey) } async msgLoop() { while (true) { const msg = this._ws?.parseMessage(await this._ws?.next()); if (msg?.hash) { this._hash = msg?.hash; await this.handleHash(); this.msgbox("input-password", "Password Required", ""); } else if (msg?.testDelay) { const testDelay = msg?.testDelay; if (!testDelay.fromClient) { await this._ws?.sendMessage({ testDelay }); } } else if (msg?.loginResponse) { const r = msg?.loginResponse; if (r.error) { this.msgbox('error', 'Error', r.error); } else if (r.peerInfo) { this._peerInfo = r.peerInfo; this.msgbox('success', 'Successful', 'Connected, waiting for image...'); } } else if (msg?.videoFrame) { this.handleVideoFrame(); } } } async handleHash() { await this._sendLoginMessage(); } msgbox(type_: string, title: string, text: string) { this._msgbox?.(type_, title, text); } close() { clearInterval(this._interval); this._ws?.close(); } setMsgbox(callback: MsgboxCallback) { this._msgbox = callback; } async login(password: string) { this.msgbox('connecting', 'Connecting...', 'Logging in...'); let salt = this._hash?.salt; if (salt) { let p = hash([password, salt]); let challenge = this._hash?.challenge; if (challenge) { p = hash([p, challenge]); await this._sendLoginMessage(p); } } } async _sendLoginMessage(password: Uint8Array | undefined = undefined) { const loginRequest = message.LoginRequest.fromPartial({ username: this._id!, myId: 'web', // to-do myName: 'web', // to-do password, }); await this._ws?.sendMessage({ loginRequest }); } handleVideoFrame() { if (!this._firstFrame) { this.msgbox('', '', ''); this._firstFrame = true; } } } async function testDelay() { const ws = new Websock(getDefaultUri(false)); await ws.open(); console.log(ws.latency()); } function getDefaultUri(isRelay: Boolean = false): string { const host = localStorage.getItem('host'); return SCHEMA + (host || HOST) + ':' + (PORT + (isRelay ? 3 : 2)); } function getrUriFromRs(uri: string): string { if (uri.indexOf(':') > 0) { const tmp = uri.split(':'); const port = parseInt(tmp[1]); uri = tmp[0] + ':' + (port + 2); } else { uri += ':' + (PORT + 3); } return SCHEMA + uri; } function hash(datas: (string | Uint8Array)[]): Uint8Array { const hasher = new sha256.Hash(); datas.forEach((data) => { if (typeof data == 'string') { data = new TextEncoder().encode(data); } return hasher.update(data); }); return hasher.digest(); }