signal-desktop/ts/sql/server/callLinks.ts

463 lines
12 KiB
TypeScript
Raw Normal View History

2024-04-01 19:19:35 +00:00
// Copyright 2024 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import { CallLinkRootKey } from '@signalapp/ringrtc';
2024-09-04 18:06:06 +00:00
import type {
CallLinkRecord,
CallLinkStateType,
CallLinkType,
DefunctCallLinkType,
2024-09-04 18:06:06 +00:00
} from '../../types/CallLink';
2024-04-01 19:19:35 +00:00
import {
callLinkRestrictionsSchema,
callLinkRecordSchema,
defunctCallLinkRecordSchema,
2024-04-01 19:19:35 +00:00
} from '../../types/CallLink';
import { toAdminKeyBytes } from '../../util/callLinks';
2024-04-25 17:09:05 +00:00
import {
callLinkToRecord,
callLinkFromRecord,
defunctCallLinkToRecord,
defunctCallLinkFromRecord,
} from '../../util/callLinksRingrtc';
2024-07-22 18:16:33 +00:00
import type { ReadableDB, WritableDB } from '../Interface';
import { prepare } from '../Server';
2024-04-01 19:19:35 +00:00
import { sql } from '../util';
import { strictAssert } from '../../util/assert';
2024-10-09 16:35:24 +00:00
import { CallStatusValue, DirectCallStatus } from '../../types/CallDisposition';
2024-10-02 19:03:10 +00:00
import { parseStrict, parseUnknown } from '../../util/schemas';
2024-04-01 19:19:35 +00:00
2024-07-22 18:16:33 +00:00
export function callLinkExists(db: ReadableDB, roomId: string): boolean {
2024-04-01 19:19:35 +00:00
const [query, params] = sql`
SELECT 1
FROM callLinks
WHERE roomId = ${roomId};
`;
return db.prepare(query).pluck(true).get(params) === 1;
}
2024-07-22 18:16:33 +00:00
export function getCallLinkByRoomId(
db: ReadableDB,
2024-04-25 17:09:05 +00:00
roomId: string
2024-07-22 18:16:33 +00:00
): CallLinkType | undefined {
2024-09-04 18:06:06 +00:00
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 {
2024-04-25 17:09:05 +00:00
const row = prepare(db, 'SELECT * FROM callLinks WHERE roomId = $roomId').get(
{
roomId,
}
);
if (!row) {
return undefined;
}
2024-10-02 19:03:10 +00:00
return parseUnknown(callLinkRecordSchema, row as unknown);
2024-04-25 17:09:05 +00:00
}
2024-07-22 18:16:33 +00:00
export function getAllCallLinks(db: ReadableDB): ReadonlyArray<CallLinkType> {
2024-04-01 19:19:35 +00:00
const [query] = sql`
SELECT * FROM callLinks;
`;
return db
.prepare(query)
.all()
2024-10-02 19:03:10 +00:00
.map((item: unknown) =>
callLinkFromRecord(parseUnknown(callLinkRecordSchema, item))
);
2024-04-01 19:19:35 +00:00
}
2024-07-22 18:16:33 +00:00
function _insertCallLink(db: WritableDB, callLink: CallLinkType): void {
2024-04-01 19:19:35 +00:00
const { roomId, rootKey } = callLink;
assertRoomIdMatchesRootKey(roomId, rootKey);
const data = callLinkToRecord(callLink);
prepare(
db,
`
INSERT INTO callLinks (
roomId,
rootKey,
adminKey,
name,
restrictions,
revoked,
2024-09-04 18:06:06 +00:00
expiration,
storageID,
storageVersion,
storageUnknownFields,
storageNeedsSync
2024-04-01 19:19:35 +00:00
) VALUES (
$roomId,
$rootKey,
$adminKey,
$name,
$restrictions,
$revoked,
2024-09-04 18:06:06 +00:00
$expiration,
$storageID,
$storageVersion,
$storageUnknownFields,
$storageNeedsSync
2024-04-01 19:19:35 +00:00
)
`
).run(data);
}
2024-07-22 18:16:33 +00:00
export function insertCallLink(db: WritableDB, callLink: CallLinkType): void {
2024-04-01 19:19:35 +00:00
_insertCallLink(db, callLink);
}
2024-09-04 18:06:06 +00:00
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);
}
2024-07-22 18:16:33 +00:00
export function updateCallLinkState(
db: WritableDB,
2024-04-01 19:19:35 +00:00
roomId: string,
2024-04-25 17:09:05 +00:00
callLinkState: CallLinkStateType
2024-07-22 18:16:33 +00:00
): CallLinkType {
2024-04-25 17:09:05 +00:00
const { name, restrictions, expiration, revoked } = callLinkState;
2024-10-02 19:03:10 +00:00
const restrictionsValue = parseStrict(
callLinkRestrictionsSchema,
restrictions
);
2024-04-01 19:19:35 +00:00
const [query, params] = sql`
UPDATE callLinks
SET
name = ${name},
restrictions = ${restrictionsValue},
expiration = ${expiration},
revoked = ${revoked ? 1 : 0}
2024-06-10 15:23:43 +00:00
WHERE roomId = ${roomId}
RETURNING *;
2024-04-01 19:19:35 +00:00
`;
2024-10-02 19:03:10 +00:00
const row: unknown = db.prepare(query).get(params);
2024-06-10 15:23:43 +00:00
strictAssert(row, 'Expected row to be returned');
2024-10-02 19:03:10 +00:00
return callLinkFromRecord(parseUnknown(callLinkRecordSchema, row));
2024-04-01 19:19:35 +00:00
}
2024-07-22 18:16:33 +00:00
export function updateCallLinkAdminKeyByRoomId(
db: WritableDB,
2024-04-25 17:09:05 +00:00
roomId: string,
adminKey: string
2024-07-22 18:16:33 +00:00
): void {
2024-04-25 17:09:05 +00:00
const adminKeyBytes = toAdminKeyBytes(adminKey);
prepare(
db,
`
UPDATE callLinks
SET adminKey = $adminKeyBytes
WHERE roomId = $roomId;
`
).run({ roomId, adminKeyBytes });
}
2024-04-01 19:19:35 +00:00
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'
);
}
2024-10-09 16:35:24 +00:00
export function deleteCallHistoryByRoomId(
db: WritableDB,
roomId: string
): void {
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);
})();
}
2024-10-09 16:35:24 +00:00
/**
* Deletes a non-admin call link from the local database, or if it's an admin call link,
* then marks it for deletion and storage sync.
*
* @returns boolean: True if storage sync is needed; False if not
*/
export function beginDeleteCallLink(db: WritableDB, roomId: string): boolean {
return 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);
2024-10-09 16:35:24 +00:00
// If we successfully deleted the call link, then it was a non-admin call link
// and we're done
if (result.changes !== 0) {
return false;
}
2024-10-09 16:35:24 +00:00
const 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 = 1
WHERE adminKey IS NOT NULL
AND deleted IS NOT 1
AND roomId = ${roomId};
`;
const deleteAdminLinkResult = db
.prepare(markAdminCallLinksDeletedQuery)
.run(markAdminCallLinksDeletedParams);
return deleteAdminLinkResult.changes > 0;
})();
}
2024-10-09 16:35:24 +00:00
export function deleteCallLinkAndHistory(db: WritableDB, roomId: string): void {
db.transaction(() => {
2024-10-09 16:35:24 +00:00
const [deleteCallLinkQuery, deleteCallLinkParams] = sql`
DELETE FROM callLinks
WHERE roomId = ${roomId};
`;
db.prepare(deleteCallLinkQuery).run(deleteCallLinkParams);
const [deleteCallHistoryQuery, clearCallHistoryParams] = sql`
UPDATE callsHistory
SET
status = ${DirectCallStatus.Deleted},
timestamp = ${Date.now()}
WHERE peerId = ${roomId};
`;
db.prepare(deleteCallHistoryQuery).run(clearCallHistoryParams);
})();
}
/**
* Deletes all non-admin call link from the local database, and marks all admin call links
* for deletion and storage sync.
*
* @returns boolean: True if storage sync is needed; False if not
*/
export function beginDeleteAllCallLinks(db: WritableDB): boolean {
const deletedAt = new Date().getTime();
return db.transaction(() => {
2024-09-04 18:06:06 +00:00
const [markAdminCallLinksDeletedQuery, markAdminCallLinksDeletedParams] =
sql`
UPDATE callLinks
2024-09-04 18:06:06 +00:00
SET
deleted = 1,
deletedAt = ${deletedAt},
storageNeedsSync = 1
2024-10-09 16:35:24 +00:00
WHERE adminKey IS NOT NULL
AND deleted IS NOT 1;
`;
2024-10-09 16:35:24 +00:00
const markAdminCallLinksDeletedResult = db
.prepare(markAdminCallLinksDeletedQuery)
.run(markAdminCallLinksDeletedParams);
const [deleteNonAdminCallLinksQuery] = sql`
DELETE FROM callLinks
WHERE adminKey IS NULL;
`;
db.prepare(deleteNonAdminCallLinksQuery).run();
2024-10-09 16:35:24 +00:00
// If admin call links were marked deleted, then storage will need sync
return markAdminCallLinksDeletedResult.changes > 0;
})();
}
2024-09-04 18:06:06 +00:00
// When you need to access the deleted field
export function getAllCallLinkRecordsWithAdminKey(
db: ReadableDB
): ReadonlyArray<CallLinkRecord> {
const [query] = sql`
SELECT * FROM callLinks
WHERE adminKey IS NOT NULL
AND rootKey IS NOT NULL;
`;
return db
.prepare(query)
.all()
2024-10-02 19:03:10 +00:00
.map((item: unknown) => parseUnknown(callLinkRecordSchema, item));
2024-09-04 18:06:06 +00:00
}
2024-10-09 16:35:24 +00:00
export function getAllAdminCallLinks(
db: ReadableDB
): ReadonlyArray<CallLinkType> {
return getAllCallLinkRecordsWithAdminKey(db).map((record: CallLinkRecord) =>
callLinkFromRecord(record)
);
}
export function getAllMarkedDeletedCallLinkRoomIds(
db: ReadableDB
): ReadonlyArray<string> {
const [query] = sql`
SELECT roomId FROM callLinks WHERE deleted = 1;
`;
return db.prepare(query).pluck().all();
}
2024-09-04 18:06:06 +00:00
// TODO: Run this after uploading storage records, maybe periodically on startup
export function finalizeDeleteCallLink(db: WritableDB, roomId: string): void {
const [query, params] = sql`
2024-09-04 18:06:06 +00:00
DELETE FROM callLinks
WHERE roomId = ${roomId}
AND deleted = 1
AND storageNeedsSync = 0;
`;
db.prepare(query).run(params);
}
2024-08-13 18:39:04 +00:00
export function _removeAllCallLinks(db: WritableDB): void {
const [query, params] = sql`
DELETE FROM callLinks;
`;
db.prepare(query).run(params);
}
export function defunctCallLinkExists(db: ReadableDB, roomId: string): boolean {
const [query, params] = sql`
SELECT 1
FROM defunctCallLinks
WHERE roomId = ${roomId};
`;
return db.prepare(query).pluck(true).get(params) === 1;
}
export function getAllDefunctCallLinksWithAdminKey(
db: ReadableDB
): ReadonlyArray<DefunctCallLinkType> {
const [query] = sql`
SELECT *
FROM defunctCallLinks
WHERE adminKey IS NOT NULL;
`;
return db
.prepare(query)
.all()
.map((item: unknown) =>
defunctCallLinkFromRecord(parseUnknown(defunctCallLinkRecordSchema, item))
);
}
export function insertDefunctCallLink(
db: WritableDB,
defunctCallLink: DefunctCallLinkType
): void {
const { roomId, rootKey } = defunctCallLink;
assertRoomIdMatchesRootKey(roomId, rootKey);
const data = defunctCallLinkToRecord(defunctCallLink);
prepare(
db,
`
INSERT INTO defunctCallLinks (
roomId,
rootKey,
adminKey,
storageID,
storageVersion,
storageUnknownFields,
storageNeedsSync
) VALUES (
$roomId,
$rootKey,
$adminKey,
$storageID,
$storageVersion,
$storageUnknownFields,
$storageNeedsSync
)
ON CONFLICT (roomId) DO NOTHING;
`
).run(data);
}
export function updateDefunctCallLink(
db: WritableDB,
defunctCallLink: DefunctCallLinkType
): void {
const { roomId, rootKey } = defunctCallLink;
assertRoomIdMatchesRootKey(roomId, rootKey);
const data = defunctCallLinkToRecord(defunctCallLink);
// Do not write roomId or rootKey since they should never change
db.prepare(
`
UPDATE callLinks
SET
storageID = $storageID,
storageVersion = $storageVersion,
storageUnknownFields = $storageUnknownFields,
storageNeedsSync = $storageNeedsSync
WHERE roomId = $roomId
`
).run(data);
}