// Copyright 2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { debounce, omit } from 'lodash'; import { CallLinkRootKey } from '@signalapp/ringrtc'; import type { LinkPreviewWithHydratedData } from '../types/message/LinkPreviews'; import type { LinkPreviewImage, LinkPreviewResult, LinkPreviewSourceType, MaybeGrabLinkPreviewOptionsType, AddLinkPreviewOptionsType, } from '../types/LinkPreview'; import * as Errors from '../types/errors'; import type { StickerPackType as StickerPackDBType } from '../sql/Interface'; import type { MIMEType } from '../types/MIME'; import * as Bytes from '../Bytes'; import { sha256 } from '../Crypto'; import * as LinkPreview from '../types/LinkPreview'; import * as Stickers from '../types/Stickers'; import * as VisualAttachment from '../types/VisualAttachment'; import * as log from '../logging/log'; import { IMAGE_JPEG, IMAGE_WEBP, stringToMIMEType } from '../types/MIME'; import { SECOND } from '../util/durations'; import { autoScale } from '../util/handleImageAttachment'; import { dropNull } from '../util/dropNull'; import { fileToBytes } from '../util/fileToBytes'; import { imageToBlurHash } from '../util/imageToBlurHash'; import { maybeParseUrl } from '../util/url'; import { sniffImageMimeType } from '../util/sniffImageMimeType'; import { drop } from '../util/drop'; import { linkCallRoute } from '../util/signalRoutes'; import { calling } from './calling'; const LINK_PREVIEW_TIMEOUT = 60 * SECOND; let currentlyMatchedLink: string | undefined; let disableLinkPreviews = false; let excludedPreviewUrls: Array = []; let linkPreviewAbortController: AbortController | undefined; let linkPreviewResult: Array | undefined; export function suspendLinkPreviews(): void { disableLinkPreviews = true; } export function hasLinkPreviewLoaded(): boolean { return Boolean(linkPreviewResult); } export const maybeGrabLinkPreview = debounce(_maybeGrabLinkPreview, 200); function _maybeGrabLinkPreview( message: string, source: LinkPreviewSourceType, { caretLocation, conversationId, mode = 'conversation', }: MaybeGrabLinkPreviewOptionsType = {} ): void { // Don't generate link previews if user has turned them off. When posting a // story we should return minimal (url-only) link previews. if (!window.Events.getLinkPreviewSetting() && mode === 'conversation') { return; } // Do nothing if we're offline const { messaging } = window.textsecure; if (!messaging) { return; } if (!message) { resetLinkPreview(conversationId); return; } if (disableLinkPreviews) { return; } const links = LinkPreview.findLinks(message, caretLocation); if (currentlyMatchedLink && links.includes(currentlyMatchedLink)) { return; } currentlyMatchedLink = undefined; excludedPreviewUrls = excludedPreviewUrls || []; const link = links.find( item => LinkPreview.shouldPreviewHref(item) && !excludedPreviewUrls.includes(item) ); if (!link) { removeLinkPreview(conversationId); return; } drop( addLinkPreview(link, source, { conversationId, disableFetch: !window.Events.getLinkPreviewSetting(), }) ); } export function resetLinkPreview(conversationId?: string): void { disableLinkPreviews = false; excludedPreviewUrls = []; removeLinkPreview(conversationId); } export function removeLinkPreview(conversationId?: string): void { (linkPreviewResult || []).forEach((item: LinkPreviewResult) => { if (item.url) { URL.revokeObjectURL(item.url); } }); linkPreviewResult = undefined; currentlyMatchedLink = undefined; linkPreviewAbortController?.abort(); linkPreviewAbortController = undefined; window.reduxActions.linkPreviews.removeLinkPreview(conversationId); } export async function addLinkPreview( url: string, source: LinkPreviewSourceType, { conversationId, disableFetch }: AddLinkPreviewOptionsType = {} ): Promise { if (currentlyMatchedLink === url) { log.warn('addLinkPreview should not be called with the same URL like this'); return; } (linkPreviewResult || []).forEach((item: LinkPreviewResult) => { if (item.url) { URL.revokeObjectURL(item.url); } }); window.reduxActions.linkPreviews.removeLinkPreview(conversationId); linkPreviewResult = undefined; // Cancel other in-flight link preview requests. if (linkPreviewAbortController) { log.info( 'addLinkPreview: canceling another in-flight link preview request' ); linkPreviewAbortController.abort(); } const thisRequestAbortController = new AbortController(); linkPreviewAbortController = thisRequestAbortController; const timeout = setTimeout(() => { thisRequestAbortController.abort(); }, LINK_PREVIEW_TIMEOUT); currentlyMatchedLink = url; // Adding just the URL so that we get into a "loading" state window.reduxActions.linkPreviews.addLinkPreview( { url, isCallLink: false, }, source, conversationId ); try { let result: LinkPreviewResult | null; if (disableFetch) { result = { title: null, url, description: null, date: null, }; } else { result = await getPreview(url, thisRequestAbortController.signal); } if (!result) { log.info( 'addLinkPreview: failed to load preview (not necessarily a problem)' ); // This helps us disambiguate between two kinds of failure: // // 1. We failed to fetch the preview because of (1) a network failure (2) an // invalid response (3) a timeout // 2. We failed to fetch the preview because we aborted the request because the // user changed the link (e.g., by continuing to type the URL) const failedToFetch = currentlyMatchedLink === url; if (failedToFetch) { excludedPreviewUrls.push(url); removeLinkPreview(conversationId); } return; } if (result.image && result.image.data) { const blob = new Blob([result.image.data], { type: result.image.contentType, }); result.image.url = URL.createObjectURL(blob); } else if (!result.title && !disableFetch) { // A link preview isn't worth showing unless we have either a title or an image removeLinkPreview(conversationId); return; } window.reduxActions.linkPreviews.addLinkPreview( { ...result, title: dropNull(result.title), description: dropNull(result.description), date: dropNull(result.date), domain: LinkPreview.getDomain(result.url), isStickerPack: LinkPreview.isStickerPack(result.url), isCallLink: LinkPreview.isCallLink(result.url), }, source, conversationId ); linkPreviewResult = [result]; } catch (error) { log.error( 'Problem loading link preview, disabling.', Errors.toLogFormat(error) ); disableLinkPreviews = true; removeLinkPreview(conversationId); } finally { clearTimeout(timeout); } } export function getLinkPreviewForSend( message: string ): Array { // Don't generate link previews if user has turned them off if (!window.storage.get('linkPreviews', false)) { return []; } if (!linkPreviewResult) { return []; } const urlsInMessage = new Set(LinkPreview.findLinks(message)); return ( linkPreviewResult // This bullet-proofs against sending link previews for URLs that are no longer in // the message. This can happen if you have a link preview, then quickly delete // the link and send the message. .filter(({ url }: Readonly<{ url: string }>) => urlsInMessage.has(url)) .map(sanitizeLinkPreview) ); } export function sanitizeLinkPreview( item: LinkPreviewResult | LinkPreviewWithHydratedData ): LinkPreviewWithHydratedData { if (item.image) { // We eliminate the ObjectURL here, unneeded for send or save return { ...item, image: omit(item.image, 'url'), title: dropNull(item.title), description: dropNull(item.description), date: dropNull(item.date), domain: LinkPreview.getDomain(item.url), isStickerPack: LinkPreview.isStickerPack(item.url), isCallLink: LinkPreview.isCallLink(item.url), }; } return { ...item, title: dropNull(item.title), description: dropNull(item.description), date: dropNull(item.date), domain: LinkPreview.getDomain(item.url), isStickerPack: LinkPreview.isStickerPack(item.url), isCallLink: LinkPreview.isCallLink(item.url), }; } async function getPreview( url: string, abortSignal: Readonly ): Promise { const { messaging } = window.textsecure; if (!messaging) { throw new Error('messaging is not available!'); } if (LinkPreview.isStickerPack(url)) { return getStickerPackPreview(url, abortSignal); } if (LinkPreview.isGroupLink(url)) { return getGroupPreview(url, abortSignal); } if (LinkPreview.isCallLink(url)) { return getCallLinkPreview(url, abortSignal); } // This is already checked elsewhere, but we want to be extra-careful. if (!LinkPreview.shouldPreviewHref(url)) { return null; } const linkPreviewMetadata = await messaging.fetchLinkPreviewMetadata( url, abortSignal ); if (!linkPreviewMetadata || abortSignal.aborted) { return null; } const { title, imageHref, description, date } = linkPreviewMetadata; let image; if (imageHref && LinkPreview.shouldPreviewHref(imageHref)) { let objectUrl: undefined | string; try { const fullSizeImage = await messaging.fetchLinkPreviewImage( imageHref, abortSignal ); if (abortSignal.aborted) { return null; } if (!fullSizeImage) { throw new Error('Failed to fetch link preview image'); } // Ensure that this file is either small enough or is resized to meet our // requirements for attachments const withBlob = await autoScale({ contentType: fullSizeImage.contentType, file: new Blob([fullSizeImage.data], { type: fullSizeImage.contentType, }), fileName: title, highQuality: true, }); const data = await fileToBytes(withBlob.file); objectUrl = URL.createObjectURL(withBlob.file); const blurHash = await imageToBlurHash(withBlob.file); const dimensions = await VisualAttachment.getImageDimensions({ objectUrl, logger: log, }); image = { data, size: data.byteLength, ...dimensions, plaintextHash: Bytes.toHex(sha256(data)), contentType: stringToMIMEType(withBlob.file.type), blurHash, }; } catch (error) { // We still want to show the preview if we failed to get an image log.error( 'getPreview failed to get image for link preview:', error.message ); } finally { if (objectUrl) { URL.revokeObjectURL(objectUrl); } } } if (abortSignal.aborted) { return null; } return { date: date || null, description: description || null, image, title, url, }; } async function getStickerPackPreview( url: string, abortSignal: Readonly ): Promise { const isPackDownloaded = ( pack?: StickerPackDBType ): pack is StickerPackDBType => { if (!pack) { return false; } return pack.status === 'downloaded' || pack.status === 'installed'; }; const isPackValid = (pack?: StickerPackDBType): pack is StickerPackDBType => { if (!pack) { return false; } return ( pack.status === 'ephemeral' || pack.status === 'downloaded' || pack.status === 'installed' ); }; const dataFromLink = Stickers.getDataFromLink(url); if (!dataFromLink) { return null; } const { id, key } = dataFromLink; try { const keyBytes = Bytes.fromHex(key); const keyBase64 = Bytes.toBase64(keyBytes); const existing = Stickers.getStickerPack(id); if (!isPackDownloaded(existing)) { await Stickers.downloadEphemeralPack(id, keyBase64); } if (abortSignal.aborted) { return null; } const pack = Stickers.getStickerPack(id); if (!isPackValid(pack)) { return null; } if (pack.key !== keyBase64) { return null; } const { title, coverStickerId } = pack; const sticker = pack.stickers[coverStickerId]; const data = pack.status === 'ephemeral' ? await window.Signal.Migrations.readTempData(sticker) : await window.Signal.Migrations.readStickerData(sticker); if (abortSignal.aborted) { return null; } let contentType: MIMEType; const sniffedMimeType = sniffImageMimeType(data); if (sniffedMimeType) { contentType = sniffedMimeType; } else { log.warn( 'getStickerPackPreview: Unable to sniff sticker MIME type; falling back to WebP' ); contentType = IMAGE_WEBP; } return { date: null, description: null, image: { ...sticker, data, size: data.byteLength, contentType, }, title, url, }; } catch (error) { log.error('getStickerPackPreview error:', Errors.toLogFormat(error)); return null; } finally { if (id) { await Stickers.removeEphemeralPack(id); } } } async function getGroupPreview( url: string, abortSignal: Readonly ): Promise { const urlObject = maybeParseUrl(url); if (!urlObject) { return null; } const { hash } = urlObject; if (!hash) { return null; } const groupData = hash.slice(1); const { inviteLinkPassword, masterKey } = window.Signal.Groups.parseGroupLink(groupData); const fields = window.Signal.Groups.deriveGroupFields( Bytes.fromBase64(masterKey) ); const id = Bytes.toBase64(fields.id); const logId = `groupv2(${id})`; const secretParams = Bytes.toBase64(fields.secretParams); log.info(`getGroupPreview/${logId}: Fetching pre-join state`); const result = await window.Signal.Groups.getPreJoinGroupInfo( inviteLinkPassword, masterKey ); if (abortSignal.aborted) { return null; } const title = window.Signal.Groups.decryptGroupTitle( dropNull(result.title), secretParams ) || window.i18n('icu:unknownGroup'); const description = window.i18n('icu:GroupV2--join--group-metadata--full', { memberCount: result?.memberCount ?? 0, }); let image: undefined | LinkPreviewImage; if (result.avatar) { try { const data = await window.Signal.Groups.decryptGroupAvatar( result.avatar, secretParams ); image = { data, size: data.byteLength, contentType: IMAGE_JPEG, blurHash: await imageToBlurHash( new Blob([data], { type: IMAGE_JPEG, }) ), }; } catch (error) { const errorString = Errors.toLogFormat(error); log.error( `getGroupPreview/${logId}: Failed to fetch avatar ${errorString}` ); } } if (abortSignal.aborted) { return null; } return { date: null, description, image, title, url, }; } async function getCallLinkPreview( url: string, _abortSignal: Readonly ): Promise { const parsedUrl = linkCallRoute.fromUrl(url); if (parsedUrl == null) { throw new Error('Failed to parse call link URL'); } const callLinkRootKey = CallLinkRootKey.parse(parsedUrl.args.key); const callLinkState = await calling.readCallLink(callLinkRootKey); if (callLinkState == null || callLinkState.revoked) { return null; } return { url, title: callLinkState.name === '' ? window.i18n('icu:calling__call-link-default-title') : callLinkState.name, description: window.i18n('icu:message--call-link-description'), image: undefined, date: null, }; }