// Copyright 2020-2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { LoggerType } from '../types/Logging'; import { maybeParseUrl } from './url'; import { isValidE164 } from './isValidE164'; const SIGNAL_HOSTS = new Set(['signal.group', 'signal.art', 'signal.me']); const SIGNAL_DOT_ME_E164_PREFIX = 'p/'; const SIGNAL_DOT_ME_USERNAME_PREFIX = 'u/'; function parseUrl(value: string | URL, logger: LoggerType): undefined | URL { if (value instanceof URL) { return value; } if (typeof value === 'string') { return maybeParseUrl(value); } logger.warn('Tried to parse a sgnl:// URL but got an unexpected type'); return undefined; } export function isSgnlHref(value: string | URL, logger: LoggerType): boolean { const url = parseUrl(value, logger); return Boolean(url?.protocol === 'sgnl:'); } export function isCaptchaHref( value: string | URL, logger: LoggerType ): boolean { const url = parseUrl(value, logger); return Boolean(url?.protocol === 'signalcaptcha:'); } // A link to a signal 'action' domain with private data in path/hash/query. We could // open a browser, but it will just link back to us. We will parse it locally instead. export function isSignalHttpsLink( value: string | URL, logger: LoggerType ): boolean { const url = parseUrl(value, logger); return Boolean( url && !url.username && !url.password && !url.port && url.protocol === 'https:' && SIGNAL_HOSTS.has(url.host) && (url.hash || url.pathname !== '/' || url.search) ); } type ParsedSgnlHref = | { command: null; args: Map; hash: undefined } | { command: string; args: Map; hash: string | undefined }; export function parseSgnlHref( href: string, logger: LoggerType ): ParsedSgnlHref { const url = parseUrl(href, logger); if (!url || !isSgnlHref(url, logger)) { return { command: null, args: new Map(), hash: undefined }; } const args = new Map(); url.searchParams.forEach((value, key) => { if (!args.has(key)) { args.set(key, value); } }); return { command: url.host, args, hash: url.hash ? url.hash.slice(1) : undefined, }; } type ParsedCaptchaHref = { readonly captcha: string; }; export function parseCaptchaHref( href: URL | string, logger: LoggerType ): ParsedCaptchaHref { const url = parseUrl(href, logger); if (!url || !isCaptchaHref(url, logger)) { throw new Error('Not a captcha href'); } return { captcha: url.host, }; } export function parseSignalHttpsLink( href: string, logger: LoggerType ): ParsedSgnlHref { const url = parseUrl(href, logger); if (!url || !isSignalHttpsLink(url, logger)) { return { command: null, args: new Map(), hash: undefined }; } if (url.host === 'signal.art') { const hash = url.hash.slice(1); const hashParams = new URLSearchParams(hash); const args = new Map(); hashParams.forEach((value, key) => { if (!args.has(key)) { args.set(key, value); } }); if (!args.get('pack_id') || !args.get('pack_key')) { return { command: null, args: new Map(), hash: undefined }; } return { command: url.pathname.replace(/\//g, ''), args, hash: url.hash ? url.hash.slice(1) : undefined, }; } if (url.host === 'signal.group' || url.host === 'signal.me') { return { command: url.host, args: new Map(), hash: url.hash ? url.hash.slice(1) : undefined, }; } return { command: null, args: new Map(), hash: undefined }; } export function parseE164FromSignalDotMeHash(hash: string): undefined | string { if (!hash.startsWith(SIGNAL_DOT_ME_E164_PREFIX)) { return; } const maybeE164 = hash.slice(SIGNAL_DOT_ME_E164_PREFIX.length); return isValidE164(maybeE164, true) ? maybeE164 : undefined; } export function parseUsernameFromSignalDotMeHash( hash: string ): undefined | string { if (!hash.startsWith(SIGNAL_DOT_ME_USERNAME_PREFIX)) { return; } return decodeURIComponent(hash.slice(SIGNAL_DOT_ME_USERNAME_PREFIX.length)); } /** * Converts `http://signal.group/#abc` to `https://signal.group/#abc`. Does the same for * other Signal hosts, like signal.me. Does nothing to other URLs. Expects a valid href. */ export function rewriteSignalHrefsIfNecessary(href: string): string { const resultUrl = new URL(href); const isHttp = resultUrl.protocol === 'http:'; const isHttpOrHttps = isHttp || resultUrl.protocol === 'https:'; if (SIGNAL_HOSTS.has(resultUrl.host) && isHttpOrHttps) { if (isHttp) { resultUrl.protocol = 'https:'; } resultUrl.username = ''; resultUrl.password = ''; return resultUrl.href; } return href; } export type GenerateUsernameLinkOptionsType = Readonly<{ short?: boolean; }>; export function generateUsernameLink( username: string, { short = false }: GenerateUsernameLinkOptionsType = {} ): string { const shortVersion = `signal.me/#u/${encodeURIComponent(username)}`; if (short) { return shortVersion; } return `https://${shortVersion}`; }