229 lines
6.3 KiB
TypeScript
229 lines
6.3 KiB
TypeScript
// Copyright 2024 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
import type { CallLinkState as RingRTCCallLinkState } from '@signalapp/ringrtc';
|
|
import {
|
|
CallLinkRootKey,
|
|
CallLinkRestrictions as RingRTCCallLinkRestrictions,
|
|
} from '@signalapp/ringrtc';
|
|
import { Aci } from '@signalapp/libsignal-client';
|
|
import { z } from 'zod';
|
|
import { v4 as generateUuid } from 'uuid';
|
|
import * as RemoteConfig from '../RemoteConfig';
|
|
import type { CallLinkAuthCredentialPresentation } from './zkgroup';
|
|
import {
|
|
CallLinkAuthCredential,
|
|
CallLinkSecretParams,
|
|
GenericServerPublicParams,
|
|
} from './zkgroup';
|
|
import { getCheckedCallLinkAuthCredentialsForToday } from '../services/groupCredentialFetcher';
|
|
import * as durations from './durations';
|
|
import * as Bytes from '../Bytes';
|
|
import type {
|
|
CallLinkConversationType,
|
|
CallLinkType,
|
|
CallLinkRecord,
|
|
CallLinkStateType,
|
|
} from '../types/CallLink';
|
|
import {
|
|
callLinkRecordSchema,
|
|
CallLinkRestrictions,
|
|
toCallLinkRestrictions,
|
|
} from '../types/CallLink';
|
|
import type { LocalizerType } from '../types/Util';
|
|
import { isTestOrMockEnvironment } from '../environment';
|
|
import { getColorForCallLink } from './getColorForCallLink';
|
|
import {
|
|
AdhocCallStatus,
|
|
CallDirection,
|
|
CallType,
|
|
type CallHistoryDetails,
|
|
} from '../types/CallDisposition';
|
|
import { CallMode } from '../types/Calling';
|
|
|
|
export const CALL_LINK_DEFAULT_STATE = {
|
|
name: '',
|
|
restrictions: CallLinkRestrictions.Unknown,
|
|
revoked: false,
|
|
expiration: null,
|
|
};
|
|
|
|
export function isCallLinksCreateEnabled(): boolean {
|
|
if (isTestOrMockEnvironment()) {
|
|
return true;
|
|
}
|
|
return RemoteConfig.getValue('desktop.calling.adhoc.create') === 'TRUE';
|
|
}
|
|
|
|
export function getRoomIdFromRootKey(rootKey: CallLinkRootKey): string {
|
|
return rootKey.deriveRoomId().toString('hex');
|
|
}
|
|
|
|
export function getCallLinkRootKeyFromUrlKey(key: string): Uint8Array {
|
|
// Returns `Buffer` which inherits from `Uint8Array`
|
|
return CallLinkRootKey.parse(key).bytes;
|
|
}
|
|
|
|
export async function getCallLinkAuthCredentialPresentation(
|
|
callLinkRootKey: CallLinkRootKey
|
|
): Promise<CallLinkAuthCredentialPresentation> {
|
|
const credentials = getCheckedCallLinkAuthCredentialsForToday(
|
|
'getCallLinkAuthCredentialPresentation'
|
|
);
|
|
const todaysCredentials = credentials.today.credential;
|
|
const credential = new CallLinkAuthCredential(
|
|
Buffer.from(todaysCredentials, 'base64')
|
|
);
|
|
|
|
const genericServerPublicParamsBase64 = window.getGenericServerPublicParams();
|
|
const genericServerPublicParams = new GenericServerPublicParams(
|
|
Buffer.from(genericServerPublicParamsBase64, 'base64')
|
|
);
|
|
|
|
const ourAci = window.textsecure.storage.user.getAci();
|
|
if (ourAci == null) {
|
|
throw new Error('Failed to get our ACI');
|
|
}
|
|
const userId = Aci.fromUuid(ourAci);
|
|
|
|
const callLinkSecretParams = CallLinkSecretParams.deriveFromRootKey(
|
|
callLinkRootKey.bytes
|
|
);
|
|
const presentation = credential.present(
|
|
userId,
|
|
credentials.today.redemptionTime / durations.SECOND,
|
|
genericServerPublicParams,
|
|
callLinkSecretParams
|
|
);
|
|
return presentation;
|
|
}
|
|
|
|
export function callLinkToConversation(
|
|
callLink: CallLinkType,
|
|
i18n: LocalizerType
|
|
): CallLinkConversationType {
|
|
const { roomId, name, rootKey } = callLink;
|
|
return {
|
|
id: roomId,
|
|
type: 'callLink',
|
|
color: getColorForCallLink(rootKey),
|
|
isMe: false,
|
|
title: name || i18n('icu:calling__call-link-default-title'),
|
|
sharedGroupNames: [],
|
|
acceptedMessageRequest: true,
|
|
badges: [],
|
|
};
|
|
}
|
|
|
|
export function getPlaceholderCallLinkConversation(
|
|
roomId: string,
|
|
i18n: LocalizerType
|
|
): CallLinkConversationType {
|
|
return {
|
|
id: roomId,
|
|
type: 'callLink',
|
|
isMe: false,
|
|
title: i18n('icu:calling__call-link-default-title'),
|
|
sharedGroupNames: [],
|
|
acceptedMessageRequest: true,
|
|
badges: [],
|
|
};
|
|
}
|
|
|
|
export function toRootKeyBytes(rootKey: string): Uint8Array {
|
|
return CallLinkRootKey.parse(rootKey).bytes;
|
|
}
|
|
|
|
export function fromRootKeyBytes(rootKey: Uint8Array): string {
|
|
return CallLinkRootKey.fromBytes(rootKey as Buffer).toString();
|
|
}
|
|
|
|
export function toAdminKeyBytes(adminKey: string): Buffer {
|
|
return Buffer.from(adminKey, 'base64');
|
|
}
|
|
|
|
export function fromAdminKeyBytes(adminKey: Uint8Array): string {
|
|
return Bytes.toBase64(adminKey);
|
|
}
|
|
|
|
/**
|
|
* RingRTC conversions
|
|
*/
|
|
|
|
export function callLinkStateFromRingRTC(
|
|
state: RingRTCCallLinkState
|
|
): CallLinkStateType {
|
|
return {
|
|
name: state.name,
|
|
restrictions: toCallLinkRestrictions(state.restrictions),
|
|
revoked: state.revoked,
|
|
expiration: state.expiration.getTime(),
|
|
};
|
|
}
|
|
|
|
const RingRTCCallLinkRestrictionsSchema = z.nativeEnum(
|
|
RingRTCCallLinkRestrictions
|
|
);
|
|
|
|
export function callLinkRestrictionsToRingRTC(
|
|
restrictions: CallLinkRestrictions
|
|
): RingRTCCallLinkRestrictions {
|
|
return RingRTCCallLinkRestrictionsSchema.parse(restrictions);
|
|
}
|
|
|
|
/**
|
|
* DB record conversions
|
|
*/
|
|
|
|
export function callLinkToRecord(callLink: CallLinkType): CallLinkRecord {
|
|
if (callLink.rootKey == null) {
|
|
throw new Error('CallLink.callLinkToRecord: rootKey is null');
|
|
}
|
|
|
|
const rootKey = toRootKeyBytes(callLink.rootKey);
|
|
const adminKey = callLink.adminKey
|
|
? toAdminKeyBytes(callLink.adminKey)
|
|
: null;
|
|
return callLinkRecordSchema.parse({
|
|
roomId: callLink.roomId,
|
|
rootKey,
|
|
adminKey,
|
|
name: callLink.name,
|
|
restrictions: callLink.restrictions,
|
|
revoked: callLink.revoked ? 1 : 0,
|
|
expiration: callLink.expiration,
|
|
});
|
|
}
|
|
|
|
export function callLinkFromRecord(record: CallLinkRecord): CallLinkType {
|
|
if (record.rootKey == null) {
|
|
throw new Error('CallLink.callLinkFromRecord: rootKey is null');
|
|
}
|
|
|
|
// root keys in memory are strings for simplicity
|
|
const rootKey = fromRootKeyBytes(record.rootKey);
|
|
const adminKey = record.adminKey ? fromAdminKeyBytes(record.adminKey) : null;
|
|
return {
|
|
roomId: record.roomId,
|
|
rootKey,
|
|
adminKey,
|
|
name: record.name,
|
|
restrictions: toCallLinkRestrictions(record.restrictions),
|
|
revoked: record.revoked === 1,
|
|
expiration: record.expiration,
|
|
};
|
|
}
|
|
|
|
export function toCallHistoryFromUnusedCallLink(
|
|
callLink: CallLinkType
|
|
): CallHistoryDetails {
|
|
return {
|
|
callId: generateUuid(),
|
|
peerId: callLink.roomId,
|
|
ringerId: null,
|
|
mode: CallMode.Adhoc,
|
|
type: CallType.Adhoc,
|
|
direction: CallDirection.Incoming,
|
|
timestamp: Date.now(),
|
|
status: AdhocCallStatus.Pending,
|
|
};
|
|
}
|