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

136 lines
3.2 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-04-25 17:09:05 +00:00
import type { CallLinkStateType, CallLinkType } from '../../types/CallLink';
2024-04-01 19:19:35 +00:00
import {
callLinkRestrictionsSchema,
callLinkRecordSchema,
} from '../../types/CallLink';
2024-04-25 17:09:05 +00:00
import {
callLinkToRecord,
callLinkFromRecord,
toAdminKeyBytes,
} from '../../util/callLinks';
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-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-04-25 17:09:05 +00:00
const row = prepare(db, 'SELECT * FROM callLinks WHERE roomId = $roomId').get(
{
roomId,
}
);
if (!row) {
return undefined;
}
return callLinkFromRecord(callLinkRecordSchema.parse(row));
}
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()
.map(item => callLinkFromRecord(callLinkRecordSchema.parse(item)));
}
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,
expiration
) VALUES (
$roomId,
$rootKey,
$adminKey,
$name,
$restrictions,
$revoked,
$expiration
)
`
).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-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-04-01 19:19:35 +00:00
const restrictionsValue = callLinkRestrictionsSchema.parse(restrictions);
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-06-10 15:23:43 +00:00
const row = db.prepare(query).get(params);
strictAssert(row, 'Expected row to be returned');
return callLinkFromRecord(callLinkRecordSchema.parse(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'
);
}