// Copyright 2024 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { CallLinkRootKey } from '@signalapp/ringrtc'; import type { CallLinkRecord, CallLinkStateType, CallLinkType, } from '../../types/CallLink'; import { callLinkRestrictionsSchema, callLinkRecordSchema, } from '../../types/CallLink'; import { toAdminKeyBytes } from '../../util/callLinks'; import { callLinkToRecord, callLinkFromRecord, } from '../../util/callLinksRingrtc'; import type { ReadableDB, WritableDB } from '../Interface'; import { prepare } from '../Server'; import { sql } from '../util'; import { strictAssert } from '../../util/assert'; import { CallStatusValue } from '../../types/CallDisposition'; export function callLinkExists(db: ReadableDB, roomId: string): boolean { const [query, params] = sql` SELECT 1 FROM callLinks WHERE roomId = ${roomId}; `; return db.prepare(query).pluck(true).get(params) === 1; } export function getCallLinkByRoomId( db: ReadableDB, roomId: string ): CallLinkType | undefined { const callLinkRecord = getCallLinkRecordByRoomId(db, roomId); if (!callLinkRecord) { return undefined; } return callLinkFromRecord(callLinkRecord); } // When you need to access all the fields (such as deleted and storage fields) export function getCallLinkRecordByRoomId( db: ReadableDB, roomId: string ): CallLinkRecord | undefined { const row = prepare(db, 'SELECT * FROM callLinks WHERE roomId = $roomId').get( { roomId, } ); if (!row) { return undefined; } return callLinkRecordSchema.parse(row); } export function getAllCallLinks(db: ReadableDB): ReadonlyArray { const [query] = sql` SELECT * FROM callLinks; `; return db .prepare(query) .all() .map(item => callLinkFromRecord(callLinkRecordSchema.parse(item))); } function _insertCallLink(db: WritableDB, callLink: CallLinkType): void { const { roomId, rootKey } = callLink; assertRoomIdMatchesRootKey(roomId, rootKey); const data = callLinkToRecord(callLink); prepare( db, ` INSERT INTO callLinks ( roomId, rootKey, adminKey, name, restrictions, revoked, expiration, storageID, storageVersion, storageUnknownFields, storageNeedsSync ) VALUES ( $roomId, $rootKey, $adminKey, $name, $restrictions, $revoked, $expiration, $storageID, $storageVersion, $storageUnknownFields, $storageNeedsSync ) ` ).run(data); } export function insertCallLink(db: WritableDB, callLink: CallLinkType): void { _insertCallLink(db, callLink); } export function updateCallLink(db: WritableDB, callLink: CallLinkType): void { const { roomId, rootKey } = callLink; assertRoomIdMatchesRootKey(roomId, rootKey); const data = callLinkToRecord(callLink); // Do not write roomId or rootKey since they should never change db.prepare( ` UPDATE callLinks SET adminKey = $adminKey, name = $name, restrictions = $restrictions, revoked = $revoked, expiration = $expiration, storageID = $storageID, storageVersion = $storageVersion, storageUnknownFields = $storageUnknownFields, storageNeedsSync = $storageNeedsSync WHERE roomId = $roomId ` ).run(data); } export function updateCallLinkState( db: WritableDB, roomId: string, callLinkState: CallLinkStateType ): CallLinkType { const { name, restrictions, expiration, revoked } = callLinkState; const restrictionsValue = callLinkRestrictionsSchema.parse(restrictions); const [query, params] = sql` UPDATE callLinks SET name = ${name}, restrictions = ${restrictionsValue}, expiration = ${expiration}, revoked = ${revoked ? 1 : 0} WHERE roomId = ${roomId} RETURNING *; `; const row = db.prepare(query).get(params); strictAssert(row, 'Expected row to be returned'); return callLinkFromRecord(callLinkRecordSchema.parse(row)); } export function updateCallLinkAdminKeyByRoomId( db: WritableDB, roomId: string, adminKey: string ): void { const adminKeyBytes = toAdminKeyBytes(adminKey); prepare( db, ` UPDATE callLinks SET adminKey = $adminKeyBytes WHERE roomId = $roomId; ` ).run({ roomId, adminKeyBytes }); } function assertRoomIdMatchesRootKey(roomId: string, rootKey: string): void { const derivedRoomId = CallLinkRootKey.parse(rootKey) .deriveRoomId() .toString('hex'); strictAssert( roomId === derivedRoomId, 'passed roomId must match roomId derived from root key' ); } function deleteCallHistoryByRoomId(db: WritableDB, roomId: string) { const [ markCallHistoryDeleteByPeerIdQuery, markCallHistoryDeleteByPeerIdParams, ] = sql` UPDATE callsHistory SET status = ${CallStatusValue.Deleted}, timestamp = ${Date.now()} WHERE peerId = ${roomId} `; db.prepare(markCallHistoryDeleteByPeerIdQuery).run( markCallHistoryDeleteByPeerIdParams ); } // This should only be called from a sync message to avoid accidentally deleting // on the client but not the server export function deleteCallLinkFromSync(db: WritableDB, roomId: string): void { db.transaction(() => { const [query, params] = sql` DELETE FROM callLinks WHERE roomId = ${roomId}; `; db.prepare(query).run(params); deleteCallHistoryByRoomId(db, roomId); })(); } export type DeleteCallLinkOptions = { storageNeedsSync: boolean; deletedAt?: number; }; export function beginDeleteCallLink( db: WritableDB, roomId: string, options: DeleteCallLinkOptions ): void { db.transaction(() => { // If adminKey is null, then we should delete the call link const [deleteNonAdminCallLinksQuery, deleteNonAdminCallLinksParams] = sql` DELETE FROM callLinks WHERE adminKey IS NULL AND roomId = ${roomId}; `; const result = db .prepare(deleteNonAdminCallLinksQuery) .run(deleteNonAdminCallLinksParams); // Skip this query if the call is already deleted if (result.changes === 0) { const { storageNeedsSync } = options; const deletedAt = options.deletedAt ?? new Date().getTime(); // If the admin key is not null, we should mark it for deletion const [markAdminCallLinksDeletedQuery, markAdminCallLinksDeletedParams] = sql` UPDATE callLinks SET deleted = 1, deletedAt = ${deletedAt}, storageNeedsSync = ${storageNeedsSync ? 1 : 0} WHERE adminKey IS NOT NULL AND roomId = ${roomId}; `; db.prepare(markAdminCallLinksDeletedQuery).run( markAdminCallLinksDeletedParams ); } deleteCallHistoryByRoomId(db, roomId); })(); } export function beginDeleteAllCallLinks(db: WritableDB): void { const deletedAt = new Date().getTime(); db.transaction(() => { const [markAdminCallLinksDeletedQuery, markAdminCallLinksDeletedParams] = sql` UPDATE callLinks SET deleted = 1, deletedAt = ${deletedAt}, storageNeedsSync = 1 WHERE adminKey IS NOT NULL; `; db.prepare(markAdminCallLinksDeletedQuery).run( markAdminCallLinksDeletedParams ); const [deleteNonAdminCallLinksQuery] = sql` DELETE FROM callLinks WHERE adminKey IS NULL; `; db.prepare(deleteNonAdminCallLinksQuery).run(); })(); } // When you need to access the deleted field export function getAllCallLinkRecordsWithAdminKey( db: ReadableDB ): ReadonlyArray { const [query] = sql` SELECT * FROM callLinks WHERE adminKey IS NOT NULL AND rootKey IS NOT NULL; `; return db .prepare(query) .all() .map(item => callLinkRecordSchema.parse(item)); } export function getAllMarkedDeletedCallLinks( db: ReadableDB ): ReadonlyArray { const [query] = sql` SELECT * FROM callLinks WHERE deleted = 1; `; return db .prepare(query) .all() .map(item => callLinkFromRecord(callLinkRecordSchema.parse(item))); } // TODO: Run this after uploading storage records, maybe periodically on startup export function finalizeDeleteCallLink(db: WritableDB, roomId: string): void { const [query, params] = sql` DELETE FROM callLinks WHERE roomId = ${roomId} AND deleted = 1 AND storageNeedsSync = 0; `; db.prepare(query).run(params); } export function _removeAllCallLinks(db: WritableDB): void { const [query, params] = sql` DELETE FROM callLinks; `; db.prepare(query).run(params); }