import _, { isEmpty } from 'lodash'; import { OpenGroupV2Room } from '../../../../data/opengroups'; import { OpenGroupRequestCommonType } from '../opengroupV2/ApiUtil'; const protocolRegex = new RegExp('https?://'); const dot = '\\.'; const qMark = '\\?'; const hostSegment = '[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?'; const hostnameRegex = new RegExp(`(?:${hostSegment}${dot})+${hostSegment}`); const portRegex = ':[1-9][0-9]{0,4}'; // roomIds allow up to 64 ascii numbers, letters, '_', or '-' chars const roomIdV2Regex = '[0-9a-zA-Z_-]{1,64}'; const publicKeyRegex = '[0-9a-fA-F]{64}'; export const publicKeyParam = 'public_key='; const openGroupV2ServerUrlRegex = new RegExp( `(?:${protocolRegex.source})?${hostnameRegex.source}(?:${portRegex})?` ); /** * Regex to use to check if a string is a v2open completeURL with pubkey. * Be aware that the /g flag is not set as .test() will otherwise return alternating result * * see https://stackoverflow.com/a/9275499/1680951 */ export const openGroupV2CompleteURLRegex = new RegExp( `^${openGroupV2ServerUrlRegex.source}\/${roomIdV2Regex}${qMark}${publicKeyParam}${publicKeyRegex}$` ); /** * Just a constant to have less 'http' everywhere. * This is the prefix used to identify our open groups in the conversation database (v1 or v2) */ // tslint:disable-next-line: no-http-string const openGroupPrefix = 'http'; // can be http:// or https:// /** * This function returns a full url on an open group v2 room used for sync messages for instance. * This is basically what the QRcode encodes * */ export function getCompleteUrlFromRoom(roomInfos: OpenGroupV2Room) { if ( isEmpty(roomInfos.serverUrl) || isEmpty(roomInfos.roomId) || isEmpty(roomInfos.serverPublicKey) ) { throw new Error('getCompleteUrlFromRoom needs serverPublicKey, roomid and serverUrl to be set'); } // serverUrl has the port and protocol already return `${roomInfos.serverUrl}/${roomInfos.roomId}?${publicKeyParam}${roomInfos.serverPublicKey}`; } /** * Prefix server with https:// if it's not already prefixed with http or https. */ export function prefixify(server: string): string { const hasPrefix = server.match('^https?://'); if (hasPrefix) { return server; } // tslint:disable-next-line: no-http-string return `http://${server}`; } /** * No sql access. Just how our open groupv2 url looks like. * ServerUrl can have the protocol and port included, or not * @returns `${openGroupPrefix}${roomId}@${serverUrl}` */ export function getOpenGroupV2ConversationId(serverUrl: string, roomId: string) { // TODOLATER we should probably make this force the serverURL to be our sogs with https when it matches pubkey or domain name if (!roomId.match(`^${roomIdV2Regex}$`)) { throw new Error('getOpenGroupV2ConversationId: Invalid roomId'); } if (!serverUrl.match(openGroupV2ServerUrlRegex)) { throw new Error('getOpenGroupV2ConversationId: Invalid serverUrl'); } return `${serverUrl}/${roomId}`; } /** * No sql access. Just plain string logic */ export function getOpenGroupV2FromConversationId( conversationId: string ): OpenGroupRequestCommonType { if (isOpenGroupV2(conversationId)) { const endProtocolStr = '://'; const startOfDoubleSlashes = conversationId.indexOf(endProtocolStr); // works for both http or https if (startOfDoubleSlashes < 0) { throw new Error('We need :// to be present in an opengroup URL'); } const firstSlashAfterProtocol = conversationId.indexOf( '/', startOfDoubleSlashes + endProtocolStr.length + 1 ); const baseUrlWithProtocol = conversationId.substring(0, firstSlashAfterProtocol); const lastSlash = conversationId.lastIndexOf('/'); const roomId = conversationId.slice(lastSlash + 1); return { serverUrl: baseUrlWithProtocol, roomId, }; } throw new Error('Not a v2 open group convo id'); } /** * Check if this conversation id corresponds to an OpenGroupV2 conversation. */ export function isOpenGroupV2(conversationId: string) { return Boolean(conversationId?.startsWith(openGroupPrefix)); }