// Copyright 2023 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import Long from 'long'; import type { Call, PeekInfo, LocalDeviceState } from '@signalapp/ringrtc'; import { CallState, ConnectionState, JoinState, callIdFromEra, callIdFromRingId, RingUpdate, } from '@signalapp/ringrtc'; import { v4 as generateGuid } from 'uuid'; import { isEqual } from 'lodash'; import { strictAssert } from './assert'; import { DataReader, DataWriter } from '../sql/Client'; import { SignalService as Proto } from '../protobuf'; import { bytesToUuid, uuidToBytes } from './uuidToBytes'; import { missingCaseError } from './missingCaseError'; import { CallEndedReason, CallMode, GroupCallJoinState, } from '../types/Calling'; import type { AciString } from '../types/ServiceId'; import { isAciString } from './isAciString'; import { isMe } from './whatTypeOfConversation'; import * as log from '../logging/log'; import * as Errors from '../types/errors'; import { incrementMessageCounter } from './incrementMessageCounter'; import { ReadStatus } from '../messages/MessageReadStatus'; import { SeenStatus, maxSeenStatus } from '../MessageSeenStatus'; import { canConversationBeUnarchived } from './canConversationBeUnarchived'; import type { ConversationAttributesType, MessageAttributesType, } from '../model-types'; import { singleProtoJobQueue } from '../jobs/singleProtoJobQueue'; import MessageSender from '../textsecure/SendMessage'; import * as Bytes from '../Bytes'; import type { CallDetails, CallEvent, CallEventDetails, CallHistoryDetails, CallHistoryGroup, CallLogEventDetails, CallStatus, GroupCallMeta, } from '../types/CallDisposition'; import { DirectCallStatus, GroupCallStatus, callEventNormalizeSchema, CallType, CallDirection, callEventDetailsSchema, LocalCallEvent, RemoteCallEvent, callHistoryDetailsSchema, callDetailsSchema, AdhocCallStatus, CallStatusValue, callLogEventNormalizeSchema, CallLogEvent, } from '../types/CallDisposition'; import type { ConversationType } from '../state/ducks/conversations'; import type { ConversationModel } from '../models/conversations'; import { drop } from './drop'; import { sendCallLinkUpdateSync } from './sendCallLinkUpdateSync'; // utils // ----- // call link roomIds are automatically redacted via redactCallLinkRoomIds() export function peerIdToLog(peerId: string, mode: CallMode): string { return mode === CallMode.Group ? `groupv2(${peerId})` : peerId; } export function formatCallEvent(callEvent: CallEventDetails): string { const { callId, peerId, direction, event, eventSource, type, mode, ringerId, timestamp, } = callEvent; const peerIdLog = peerIdToLog(peerId, mode); return `CallEvent (${callId}, ${peerIdLog}, ${mode}, ${event}, ${direction}, ${type}, ${mode}, ${timestamp}, ${ringerId}, ${eventSource})`; } export function formatCallHistory(callHistory: CallHistoryDetails): string { const { callId, peerId, direction, status, type, mode, timestamp, ringerId } = callHistory; const peerIdLog = peerIdToLog(peerId, mode); return `CallHistory (${callId}, ${peerIdLog}, ${mode}, ${status}, ${direction}, ${type}, ${mode}, ${timestamp}, ${ringerId})`; } export function formatCallHistoryGroup( callHistoryGroup: CallHistoryGroup ): string { const { peerId, direction, status, type, mode, timestamp } = callHistoryGroup; return `CallHistoryGroup (${peerId}, ${mode}, ${status}, ${direction}, ${type}, ${mode}, ${timestamp})`; } export function formatPeekInfo(peekInfo: PeekInfo): string { const { eraId, deviceCount, creator } = peekInfo; const callId = eraId != null ? getCallIdFromEra(eraId) : null; const creatorAci = creator != null ? getCreatorAci(creator) : null; return `PeekInfo (${eraId}, ${callId}, ${creatorAci}, ${deviceCount})`; } export function formatLocalDeviceState( localDeviceState: LocalDeviceState ): string { const connectionState = ConnectionState[localDeviceState.connectionState]; const joinState = JoinState[localDeviceState.joinState]; return `LocalDeviceState (${connectionState}, ${joinState})`; } export function getCallIdFromRing(ringId: bigint): string { return Long.fromValue(callIdFromRingId(ringId)).toString(); } export function getCallIdFromEra(eraId: string): string { return Long.fromValue(callIdFromEra(eraId)).toString(); } export function getCreatorAci(creator: Buffer): AciString { const aci = bytesToUuid(creator); strictAssert(aci != null, 'creator uuid buffer was not a valid uuid'); strictAssert(isAciString(aci), 'creator uuid buffer was not a valid aci'); return aci; } export function getGroupCallMeta( peekInfo: PeekInfo | null ): GroupCallMeta | null { if (peekInfo?.eraId == null || peekInfo?.creator == null) { return null; } const callId = getCallIdFromEra(peekInfo.eraId); const ringerId = bytesToUuid(peekInfo.creator); strictAssert(ringerId != null, 'peekInfo.creator was invalid uuid'); strictAssert(isAciString(ringerId), 'peekInfo.creator was invalid aci'); return { callId, ringerId }; } export function getPeerIdFromConversation( conversation: ConversationAttributesType | ConversationType ): AciString | string { if (conversation.type === 'direct' || conversation.type === 'private') { strictAssert( isAciString(conversation.serviceId), 'ACI must exist for direct chat' ); return conversation.serviceId; } strictAssert( conversation.groupId != null, 'groupId must exist for group chat' ); return conversation.groupId; } export function convertJoinState(joinState: JoinState): GroupCallJoinState { if (joinState === JoinState.Joined) { return GroupCallJoinState.Joined; } if (joinState === JoinState.Joining) { return GroupCallJoinState.Joining; } if (joinState === JoinState.NotJoined) { return GroupCallJoinState.NotJoined; } if (joinState === JoinState.Pending) { return GroupCallJoinState.Pending; } throw missingCaseError(joinState); } // Call Events <-> Protos // ---------------------- export function getCallEventForProto( callEventProto: Proto.SyncMessage.ICallEvent, eventSource: string ): CallEventDetails { const callEvent = callEventNormalizeSchema.parse(callEventProto); const { callId, peerId, timestamp } = callEvent; let type: CallType; if (callEvent.type === Proto.SyncMessage.CallEvent.Type.GROUP_CALL) { type = CallType.Group; } else if (callEvent.type === Proto.SyncMessage.CallEvent.Type.AUDIO_CALL) { type = CallType.Audio; } else if (callEvent.type === Proto.SyncMessage.CallEvent.Type.VIDEO_CALL) { type = CallType.Video; } else if (callEvent.type === Proto.SyncMessage.CallEvent.Type.AD_HOC_CALL) { type = CallType.Adhoc; } else { throw new TypeError(`Unknown call type ${callEvent.type}`); } let mode: CallMode; if (type === CallType.Group) { mode = CallMode.Group; } else if (type === CallType.Adhoc) { mode = CallMode.Adhoc; } else { mode = CallMode.Direct; } let direction: CallDirection; if (callEvent.direction === Proto.SyncMessage.CallEvent.Direction.INCOMING) { direction = CallDirection.Incoming; } else if ( callEvent.direction === Proto.SyncMessage.CallEvent.Direction.OUTGOING ) { direction = CallDirection.Outgoing; } else { throw new TypeError(`Unknown call direction ${callEvent.direction}`); } let event: RemoteCallEvent; if (callEvent.event === Proto.SyncMessage.CallEvent.Event.ACCEPTED) { event = RemoteCallEvent.Accepted; } else if ( callEvent.event === Proto.SyncMessage.CallEvent.Event.NOT_ACCEPTED ) { event = RemoteCallEvent.NotAccepted; } else if (callEvent.event === Proto.SyncMessage.CallEvent.Event.DELETE) { event = RemoteCallEvent.Delete; } else if (callEvent.event === Proto.SyncMessage.CallEvent.Event.OBSERVED) { event = RemoteCallEvent.Observed; } else { throw new TypeError(`Unknown call event ${callEvent.event}`); } return callEventDetailsSchema.parse({ callId, peerId, ringerId: null, mode, type, direction, timestamp, event, eventSource, }); } const callLogEventFromProto: Partial< Record > = { [Proto.SyncMessage.CallLogEvent.Type.CLEAR]: CallLogEvent.Clear, [Proto.SyncMessage.CallLogEvent.Type.MARKED_AS_READ]: CallLogEvent.MarkedAsRead, [Proto.SyncMessage.CallLogEvent.Type.MARKED_AS_READ_IN_CONVERSATION]: CallLogEvent.MarkedAsReadInConversation, }; export function getCallLogEventForProto( callLogEventProto: Proto.SyncMessage.ICallLogEvent ): CallLogEventDetails { const callLogEvent = callLogEventNormalizeSchema.parse(callLogEventProto); const type = callLogEventFromProto[callLogEvent.type]; if (type == null) { throw new TypeError(`Unknown call log event ${callLogEvent.type}`); } return { type, timestamp: callLogEvent.timestamp, peerId: callLogEvent.peerId ?? null, callId: callLogEvent.callId ?? null, }; } const directionToProto = { [CallDirection.Incoming]: Proto.SyncMessage.CallEvent.Direction.INCOMING, [CallDirection.Outgoing]: Proto.SyncMessage.CallEvent.Direction.OUTGOING, }; const typeToProto = { [CallType.Audio]: Proto.SyncMessage.CallEvent.Type.AUDIO_CALL, [CallType.Video]: Proto.SyncMessage.CallEvent.Type.VIDEO_CALL, [CallType.Group]: Proto.SyncMessage.CallEvent.Type.GROUP_CALL, [CallType.Adhoc]: Proto.SyncMessage.CallEvent.Type.AD_HOC_CALL, }; const statusToProto: Record< CallStatus, Proto.SyncMessage.CallEvent.Event | null > = { [CallStatusValue.Accepted]: Proto.SyncMessage.CallEvent.Event.ACCEPTED, [CallStatusValue.Declined]: Proto.SyncMessage.CallEvent.Event.NOT_ACCEPTED, [CallStatusValue.Deleted]: Proto.SyncMessage.CallEvent.Event.DELETE, [CallStatusValue.Missed]: null, [CallStatusValue.Pending]: null, [CallStatusValue.GenericGroupCall]: null, [CallStatusValue.GenericAdhocCall]: Proto.SyncMessage.CallEvent.Event.OBSERVED, [CallStatusValue.OutgoingRing]: null, [CallStatusValue.Ringing]: null, [CallStatusValue.Joined]: null, [CallStatusValue.JoinedAdhoc]: Proto.SyncMessage.CallEvent.Event.ACCEPTED, }; function shouldSyncStatus(callStatus: CallStatus) { return statusToProto[callStatus] != null; } export function getBytesForPeerId(callHistory: CallHistoryDetails): Uint8Array { let peerId = callHistory.mode === CallMode.Adhoc ? Bytes.fromBase64(callHistory.peerId) : uuidToBytes(callHistory.peerId); if (peerId.length === 0) { peerId = Bytes.fromBase64(callHistory.peerId); } return peerId; } export function getProtoForCallHistory( callHistory: CallHistoryDetails ): Proto.SyncMessage.ICallEvent | null { const event = statusToProto[callHistory.status]; strictAssert( event != null, `getProtoForCallHistory: Cannot create proto for status ${formatCallHistory( callHistory )}` ); return new Proto.SyncMessage.CallEvent({ peerId: getBytesForPeerId(callHistory), callId: Long.fromString(callHistory.callId), type: typeToProto[callHistory.type], direction: directionToProto[callHistory.direction], event, timestamp: Long.fromNumber(callHistory.timestamp), }); } // Local Events // ------------ const endedReasonToEvent: Record = { // Accepted [CallEndedReason.AcceptedOnAnotherDevice]: LocalCallEvent.Accepted, // Hangup (Incoming = Declined, Outgoing = Missed) [CallEndedReason.RemoteHangup]: LocalCallEvent.RemoteHangup, [CallEndedReason.LocalHangup]: LocalCallEvent.Hangup, // Missed [CallEndedReason.Busy]: LocalCallEvent.Missed, [CallEndedReason.BusyOnAnotherDevice]: LocalCallEvent.Missed, [CallEndedReason.ConnectionFailure]: LocalCallEvent.Missed, [CallEndedReason.Glare]: LocalCallEvent.Missed, [CallEndedReason.GlareFailure]: LocalCallEvent.Missed, [CallEndedReason.InternalFailure]: LocalCallEvent.Missed, [CallEndedReason.ReCall]: LocalCallEvent.Missed, [CallEndedReason.ReceivedOfferExpired]: LocalCallEvent.Missed, [CallEndedReason.ReceivedOfferWhileActive]: LocalCallEvent.Missed, [CallEndedReason.ReceivedOfferWithGlare]: LocalCallEvent.Missed, [CallEndedReason.RemoteHangupNeedPermission]: LocalCallEvent.Missed, [CallEndedReason.SignalingFailure]: LocalCallEvent.Missed, [CallEndedReason.Timeout]: LocalCallEvent.Missed, [CallEndedReason.Declined]: LocalCallEvent.Missed, [CallEndedReason.DeclinedOnAnotherDevice]: LocalCallEvent.Missed, }; export function getLocalCallEventFromCallEndedReason( callEndedReason: CallEndedReason ): LocalCallEvent { log.info('getLocalCallEventFromCallEndedReason', callEndedReason); return endedReasonToEvent[callEndedReason]; } export function getLocalCallEventFromDirectCall( call: Call ): LocalCallEvent | null { log.info('getLocalCallEventFromDirectCall', call.state); if (call.state === CallState.Accepted) { return LocalCallEvent.Accepted; } if (call.state === CallState.Ended) { strictAssert(call.endedReason != null, 'Call ended without reason'); return getLocalCallEventFromCallEndedReason(call.endedReason); } if (call.state === CallState.Ringing) { return LocalCallEvent.Ringing; } if (call.state === CallState.Prering) { return null; } if (call.state === CallState.Reconnecting) { return null; } throw missingCaseError(call.state); } const ringUpdateToEvent: Record = { [RingUpdate.AcceptedOnAnotherDevice]: LocalCallEvent.Accepted, [RingUpdate.BusyLocally]: LocalCallEvent.Missed, [RingUpdate.BusyOnAnotherDevice]: LocalCallEvent.Missed, [RingUpdate.CancelledByRinger]: LocalCallEvent.Missed, [RingUpdate.DeclinedOnAnotherDevice]: LocalCallEvent.Missed, [RingUpdate.ExpiredRequest]: LocalCallEvent.Missed, [RingUpdate.Requested]: LocalCallEvent.Ringing, }; export function getLocalCallEventFromRingUpdate( update: RingUpdate ): LocalCallEvent | null { log.info('getLocalCallEventFromRingUpdate', RingUpdate[update]); return ringUpdateToEvent[update]; } export function getLocalCallEventFromJoinState( joinState: GroupCallJoinState | null, groupCallMeta: GroupCallMeta ): LocalCallEvent | null { const direction = getCallDirectionFromRingerId(groupCallMeta.ringerId); log.info( 'getLocalCallEventFromGroupCall', direction, joinState != null ? GroupCallJoinState[joinState] : null ); if (direction === CallDirection.Incoming) { if (joinState === GroupCallJoinState.Joined) { return LocalCallEvent.Accepted; } if (joinState === GroupCallJoinState.NotJoined || joinState == null) { return LocalCallEvent.Started; } if ( joinState === GroupCallJoinState.Joining || joinState === GroupCallJoinState.Pending ) { return LocalCallEvent.Accepted; } throw missingCaseError(joinState); } else { if (joinState === GroupCallJoinState.NotJoined || joinState == null) { return LocalCallEvent.Started; } return LocalCallEvent.Ringing; } } // Call Direction // -------------- function getCallDirectionFromRingerId( ringerId: AciString | string ): CallDirection { const ringerConversation = window.ConversationController.get(ringerId); strictAssert( ringerConversation != null, `getCallDirectionFromRingerId: Missing ringer conversation (${ringerId})` ); const direction = isMe(ringerConversation.attributes) ? CallDirection.Outgoing : CallDirection.Incoming; return direction; } // Call Details // ------------ export function getCallDetailsFromDirectCall( peerId: AciString | string, call: Call ): CallDetails { return callDetailsSchema.parse({ callId: Long.fromValue(call.callId).toString(), peerId, ringerId: call.isIncoming ? call.remoteUserId : null, mode: CallMode.Direct, type: call.isVideoCall ? CallType.Video : CallType.Audio, direction: call.isIncoming ? CallDirection.Incoming : CallDirection.Outgoing, timestamp: Date.now(), }); } export function getCallDetailsFromEndedDirectCall( callId: string, peerId: AciString | string, ringerId: AciString | string, wasVideoCall: boolean, timestamp: number ): CallDetails { return callDetailsSchema.parse({ callId, peerId, ringerId, mode: CallMode.Direct, type: wasVideoCall ? CallType.Video : CallType.Audio, direction: getCallDirectionFromRingerId(ringerId), timestamp, }); } export function getCallDetailsFromGroupCallMeta( peerId: AciString | string, groupCallMeta: GroupCallMeta ): CallDetails { return callDetailsSchema.parse({ callId: groupCallMeta.callId, peerId, ringerId: groupCallMeta.ringerId, mode: CallMode.Group, type: CallType.Group, direction: getCallDirectionFromRingerId(groupCallMeta.ringerId), timestamp: Date.now(), }); } export function getCallDetailsForAdhocCall( peerId: AciString | string, callId: string ): CallDetails { return callDetailsSchema.parse({ callId, peerId, ringerId: null, mode: CallMode.Adhoc, type: CallType.Adhoc, // Direction is only outgoing when your action causes ringing for others. // As Adhoc calls do not support ringing, this is always incoming for now direction: CallDirection.Incoming, timestamp: Date.now(), }); } // Call Event Details // ------------------ export function getCallEventDetails( callDetails: CallDetails, event: LocalCallEvent, eventSource: string ): CallEventDetails { return callEventDetailsSchema.parse({ ...callDetails, event, eventSource }); } // transitions // ----------- export function transitionCallHistory( callHistory: CallHistoryDetails | null, callEvent: CallEventDetails ): CallHistoryDetails { const { callId, peerId, ringerId, mode, type, direction, event } = callEvent; if (callHistory != null) { strictAssert(callHistory.callId === callId, 'callId must be same'); strictAssert(callHistory.peerId === peerId, 'peerId must be same'); strictAssert( ringerId == null || callHistory.ringerId === ringerId, 'ringerId must be same if it exists' ); strictAssert(callHistory.direction === direction, 'direction must be same'); strictAssert(callHistory.type === type, 'type must be same'); strictAssert(callHistory.mode === mode, 'mode must be same'); } const prevStatus = callHistory?.status ?? null; let status: CallStatus; if (mode === CallMode.Direct) { status = transitionDirectCallStatus( prevStatus as DirectCallStatus | null, event, direction ); } else if (mode === CallMode.Group) { status = transitionGroupCallStatus( prevStatus as GroupCallStatus | null, event, direction ); } else if (mode === CallMode.Adhoc) { status = transitionAdhocCallStatus( prevStatus as AdhocCallStatus | null, event, direction ); } else { throw missingCaseError(mode); } const timestamp = transitionTimestamp(callHistory, callEvent); log.info( `transitionCallHistory: Transitioned call history timestamp (before: ${callHistory?.timestamp}, after: ${timestamp})` ); return callHistoryDetailsSchema.parse({ callId, peerId, ringerId, mode, type, direction, timestamp, status, }); } function transitionTimestamp( callHistory: CallHistoryDetails | null, callEvent: CallEventDetails ): number { // Sometimes when a device is asleep the timestamp will be stuck in the past. // In some cases, we'll accept whatever the most recent timestamp is. const latestTimestamp = Math.max( callEvent.timestamp, callHistory?.timestamp ?? 0 ); // Always accept the timestamp if we have no other timestamps. if (callHistory == null) { return callEvent.timestamp; } // Deleted call history should never be changed if ( callHistory.status === DirectCallStatus.Deleted || callHistory.status === GroupCallStatus.Deleted || callHistory.status === AdhocCallStatus.Deleted ) { return callHistory.timestamp; } // Accepted call history should only be changed if we get a remote accepted // event with possibly a better timestamp. if ( callHistory.status === DirectCallStatus.Accepted || callHistory.status === GroupCallStatus.Accepted || callHistory.status === GroupCallStatus.Joined || callHistory.status === AdhocCallStatus.Joined ) { if (callEvent.event === RemoteCallEvent.Accepted) { return latestTimestamp; } return callHistory.timestamp; } // Observed call history should only be changed if we get a remote observed // event with possibly a better timestamp. if (callHistory.status === AdhocCallStatus.Generic) { if (callEvent.event === RemoteCallEvent.Observed) { return latestTimestamp; } return callHistory.timestamp; } // Declined call history should only be changed if if we transition to an // accepted state or get a remote 'not accepted' event with possibly a better // timestamp. if ( callHistory.status === DirectCallStatus.Declined || callHistory.status === GroupCallStatus.Declined ) { if (callEvent.event === RemoteCallEvent.NotAccepted) { return latestTimestamp; } return callHistory.timestamp; } // We don't care about holding onto timestamps that were from these states if ( callHistory.status === DirectCallStatus.Pending || callHistory.status === GroupCallStatus.GenericGroupCall || callHistory.status === GroupCallStatus.OutgoingRing || callHistory.status === GroupCallStatus.Ringing || callHistory.status === DirectCallStatus.Missed || callHistory.status === GroupCallStatus.Missed || callHistory.status === AdhocCallStatus.Pending ) { return latestTimestamp; } throw missingCaseError(callHistory.status); } function transitionDirectCallStatus( status: DirectCallStatus | null, callEvent: CallEvent, direction: CallDirection ): DirectCallStatus { log.info('transitionDirectCallStatus', status, callEvent, direction); // In all cases if we get a delete event, we need to delete the call, and never // transition from deleted. if ( callEvent === RemoteCallEvent.Delete || callEvent === LocalCallEvent.Delete || status === DirectCallStatus.Deleted ) { return DirectCallStatus.Deleted; } if ( callEvent === RemoteCallEvent.Accepted || callEvent === LocalCallEvent.Accepted ) { return DirectCallStatus.Accepted; } if (status === DirectCallStatus.Accepted) { return status; } if (callEvent === RemoteCallEvent.NotAccepted) { if (status === DirectCallStatus.Declined) { return DirectCallStatus.Declined; } return DirectCallStatus.Missed; } if (callEvent === RemoteCallEvent.Observed) { throw new Error( `callHistoryDetails: Direct calls shouldn't send ${callEvent}` ); } if (callEvent === LocalCallEvent.Missed) { return DirectCallStatus.Missed; } if (callEvent === LocalCallEvent.Declined) { return DirectCallStatus.Declined; } if (callEvent === LocalCallEvent.Hangup) { if (direction === CallDirection.Incoming) { return DirectCallStatus.Declined; } return DirectCallStatus.Missed; } if (callEvent === LocalCallEvent.RemoteHangup) { if (direction === CallDirection.Incoming) { return DirectCallStatus.Missed; } return DirectCallStatus.Declined; } if ( callEvent === LocalCallEvent.Started || callEvent === LocalCallEvent.Ringing ) { return DirectCallStatus.Pending; } throw missingCaseError(callEvent); } function transitionGroupCallStatus( status: GroupCallStatus | null, event: CallEvent, direction: CallDirection ): GroupCallStatus { log.info('transitionGroupCallStatus', status, event, direction); // In all cases if we get a delete event, we need to delete the call, and never // transition from deleted. if ( event === RemoteCallEvent.Delete || event === LocalCallEvent.Delete || status === GroupCallStatus.Deleted ) { return GroupCallStatus.Deleted; } if (status === GroupCallStatus.Accepted) { return status; } if ( event === RemoteCallEvent.NotAccepted || event === RemoteCallEvent.Observed ) { throw new Error(`callHistoryDetails: Group calls shouldn't send ${event}`); } if (event === RemoteCallEvent.Accepted || event === LocalCallEvent.Accepted) { switch (status) { case null: { return GroupCallStatus.Joined; } case GroupCallStatus.Ringing: case GroupCallStatus.Missed: case GroupCallStatus.Declined: { return GroupCallStatus.Accepted; } case GroupCallStatus.GenericGroupCall: { return GroupCallStatus.Joined; } case GroupCallStatus.Joined: case GroupCallStatus.OutgoingRing: { return status; } default: { throw missingCaseError(status); } } } if (event === LocalCallEvent.Started) { return GroupCallStatus.GenericGroupCall; } if (event === LocalCallEvent.Ringing) { return GroupCallStatus.Ringing; } if (event === LocalCallEvent.Missed) { return GroupCallStatus.Missed; } if (event === LocalCallEvent.Declined) { return GroupCallStatus.Declined; } if (event === LocalCallEvent.Hangup) { if (direction === CallDirection.Incoming) { return GroupCallStatus.Declined; } return GroupCallStatus.Missed; } if (event === LocalCallEvent.RemoteHangup) { if (direction === CallDirection.Incoming) { return GroupCallStatus.Missed; } return GroupCallStatus.Declined; } throw missingCaseError(event); } function transitionAdhocCallStatus( status: AdhocCallStatus | null, callEvent: CallEvent, direction: CallDirection ): AdhocCallStatus { log.info( `transitionAdhocCallStatus: status=${status} callEvent=${callEvent} direction=${direction}` ); // In all cases if we get a delete event, we need to delete the call, and never // transition from deleted. if ( callEvent === RemoteCallEvent.Delete || callEvent === LocalCallEvent.Delete || status === AdhocCallStatus.Deleted ) { return AdhocCallStatus.Deleted; } // The Accepted event is used to indicate we have fully joined an adhoc call. // If admin approval was required for a call link, this event indicates approval // was granted. if ( callEvent === RemoteCallEvent.Accepted || callEvent === LocalCallEvent.Accepted ) { return AdhocCallStatus.Joined; } if (status === AdhocCallStatus.Joined) { return status; } if ( callEvent === RemoteCallEvent.Observed || callEvent === LocalCallEvent.Started ) { return AdhocCallStatus.Generic; } // For Adhoc calls, ringing and corresponding events are not supported currently. // However we handle those events here to be exhaustive. if ( callEvent === RemoteCallEvent.NotAccepted || callEvent === LocalCallEvent.Missed || callEvent === LocalCallEvent.Declined || callEvent === LocalCallEvent.Hangup || callEvent === LocalCallEvent.RemoteHangup || // never actually happens, but need for exhaustive check callEvent === LocalCallEvent.Ringing ) { return AdhocCallStatus.Pending; } throw missingCaseError(callEvent); } // actions // ------- async function updateLocalCallHistory( callEvent: CallEventDetails, receivedAtCounter: number | null, receivedAtMS: number | null ): Promise { const conversation = window.ConversationController.get(callEvent.peerId); strictAssert( conversation != null, `updateLocalCallHistory: Conversation not found ${formatCallEvent( callEvent )}` ); return conversation.queueJob( 'updateLocalCallHistory', async () => { log.info( 'updateLocalCallHistory: Processing call event:', formatCallEvent(callEvent) ); const prevCallHistory = (await DataReader.getCallHistory(callEvent.callId, callEvent.peerId)) ?? null; if (prevCallHistory != null) { log.info( 'updateLocalCallHistory: Found previous call history:', formatCallHistory(prevCallHistory) ); } else { log.info('updateLocalCallHistory: No previous call history'); } let callHistory: CallHistoryDetails; try { callHistory = transitionCallHistory(prevCallHistory, callEvent); } catch (error) { log.error( "updateLocalCallHistory: Couldn't transition call history:", formatCallEvent(callEvent), Errors.toLogFormat(error) ); return null; } const updatedCallHistory = await saveCallHistory({ callHistory, conversation, receivedAtCounter, receivedAtMS, }); return updatedCallHistory; } ); } export async function updateAdhocCallHistory( callEvent: CallEventDetails ): Promise { const callHistory = await updateLocalAdhocCallHistory(callEvent); if (!callHistory) { return; } drop(updateRemoteCallHistory(callHistory)); } export async function updateLocalAdhocCallHistory( callEvent: CallEventDetails ): Promise { log.info( 'updateLocalAdhocCallHistory: Processing call event:', formatCallEvent(callEvent) ); const prevCallHistory = (await DataReader.getCallHistory(callEvent.callId, callEvent.peerId)) ?? null; if (prevCallHistory != null) { log.info( 'updateAdhocCallHistory: Found previous call history:', formatCallHistory(prevCallHistory) ); } else { log.info('updateAdhocCallHistory: No previous call history'); } let callHistory: CallHistoryDetails; try { callHistory = transitionCallHistory(prevCallHistory, callEvent); } catch (error) { log.error( "updateAdhocCallHistory: Couldn't transition call history:", formatCallEvent(callEvent), Errors.toLogFormat(error) ); return null; } strictAssert( callHistory.status === AdhocCallStatus.Generic || callHistory.status === AdhocCallStatus.Pending || callHistory.status === AdhocCallStatus.Joined || callHistory.status === AdhocCallStatus.Deleted, `updateAdhocCallHistory: callHistory must have adhoc status (was ${callHistory.status})` ); const isDeleted = callHistory.status === AdhocCallStatus.Deleted; if (prevCallHistory != null && isEqual(prevCallHistory, callHistory)) { log.info( 'updateAdhocCallHistory: Next call history is identical, skipping save' ); } else { log.info( 'updateAdhocCallHistory: Saving call history:', formatCallHistory(callHistory) ); await DataWriter.saveCallHistory(callHistory); /* If we're not a call link admin and this is the first call history for this link, then it means we clicked someone else's link and discovered it was active. We sync this newly observed call link so the subsequent call event OBSERVED sync message refers to a valid call link. */ if (prevCallHistory == null) { const callLink = await DataReader.getCallLinkByRoomId(callEvent.peerId); if (callLink) { log.info( `updateAdhocCallHistory: Syncing new observed call link ${callEvent.peerId}` ); drop(sendCallLinkUpdateSync(callLink)); } else { log.error( `updateAdhocCallHistory: New observed call link missing in DB: ${callEvent.peerId}` ); } } } if (isDeleted) { window.reduxActions.callHistory.removeCallHistory(callHistory.callId); } else { window.reduxActions.callHistory.addCallHistory(callHistory); } return callHistory; } async function saveCallHistory({ callHistory, conversation, receivedAtCounter, receivedAtMS, }: { callHistory: CallHistoryDetails; conversation: ConversationModel; receivedAtCounter: number | null; receivedAtMS: number | null; }): Promise { log.info( 'saveCallHistory: Saving call history:', formatCallHistory(callHistory) ); const isDeleted = callHistory.status === DirectCallStatus.Deleted || callHistory.status === GroupCallStatus.Deleted; await DataWriter.saveCallHistory(callHistory); if (isDeleted) { window.reduxActions.callHistory.removeCallHistory(callHistory.callId); } else { window.reduxActions.callHistory.addCallHistory(callHistory); } const prevMessage = await DataReader.getCallHistoryMessageByCallId({ conversationId: conversation.id, callId: callHistory.callId, }); if (prevMessage != null) { log.info( 'saveCallHistory: Found previous call history message:', prevMessage.id ); } else { log.info( 'saveCallHistory: No previous call history message', conversation.id ); } if (isDeleted) { if (prevMessage != null) { await DataWriter.removeMessage(prevMessage.id, { fromSync: true, singleProtoJobQueue, }); } return callHistory; } let unseen = false; if (callHistory.mode === CallMode.Direct) { unseen = callHistory.direction === CallDirection.Incoming && (callHistory.status === DirectCallStatus.Missed || callHistory.status === DirectCallStatus.Pending); } else if (callHistory.mode === CallMode.Group) { unseen = callHistory.direction === CallDirection.Incoming && (callHistory.status === GroupCallStatus.Ringing || callHistory.status === GroupCallStatus.GenericGroupCall || callHistory.status === GroupCallStatus.Missed); } let seenStatus = unseen ? SeenStatus.Unseen : SeenStatus.NotApplicable; if (prevMessage?.seenStatus != null) { seenStatus = maxSeenStatus(seenStatus, prevMessage.seenStatus); } const message: MessageAttributesType = { id: prevMessage?.id ?? generateGuid(), conversationId: conversation.id, type: 'call-history', timestamp: prevMessage?.timestamp ?? callHistory.timestamp, sent_at: prevMessage?.sent_at ?? callHistory.timestamp, received_at: prevMessage?.received_at ?? receivedAtCounter ?? incrementMessageCounter(), received_at_ms: prevMessage?.received_at_ms ?? receivedAtMS ?? callHistory.timestamp, readStatus: ReadStatus.Read, seenStatus, callId: callHistory.callId, }; const id = await DataWriter.saveMessage(message, { ourAci: window.textsecure.storage.user.getCheckedAci(), // We don't want to force save if we're updating an existing message forceSave: prevMessage == null, }); log.info('saveCallHistory: Saved call history message:', id); const model = window.MessageCache.__DEPRECATED$register( id, new window.Whisper.Message({ ...message, id, }), 'callDisposition' ); if (prevMessage == null) { if (callHistory.direction === CallDirection.Outgoing) { conversation.incrementSentMessageCount(); } else { conversation.incrementMessageCount(); } conversation.trigger('newmessage', model); } await conversation.updateLastMessage().catch(error => { log.error( 'saveCallHistory: Failed to update last message:', Errors.toLogFormat(error) ); }); conversation.set( 'active_at', Math.max(conversation.get('active_at') ?? 0, callHistory.timestamp) ); if (canConversationBeUnarchived(conversation.attributes)) { conversation.setArchived(false); } else { await DataWriter.updateConversation(conversation.attributes); } window.reduxActions.callHistory.updateCallHistoryUnreadCount(); return callHistory; } async function updateRemoteCallHistory( callHistory: CallHistoryDetails ): Promise { if (!shouldSyncStatus(callHistory.status)) { log.info( 'updateRemoteCallHistory: Not syncing call history:', formatCallHistory(callHistory) ); return; } log.info( 'updateRemoteCallHistory: syncing call history:', formatCallHistory(callHistory) ); try { const ourAci = window.textsecure.storage.user.getCheckedAci(); const syncMessage = MessageSender.createSyncMessage(); syncMessage.callEvent = getProtoForCallHistory(callHistory); const contentMessage = new Proto.Content(); contentMessage.syncMessage = syncMessage; const { ContentHint } = Proto.UnidentifiedSenderMessage.Message; await singleProtoJobQueue.add({ contentHint: ContentHint.RESENDABLE, serviceId: ourAci, isSyncMessage: true, protoBase64: Bytes.toBase64( Proto.Content.encode(contentMessage).finish() ), type: 'callEventSync', urgent: false, }); } catch (error) { log.error( 'updateRemoteCallHistory: Failed to queue sync message:', Errors.toLogFormat(error) ); } } export async function updateCallHistoryFromRemoteEvent( callEvent: CallEventDetails, receivedAtCounter: number, receivedAtMS: number ): Promise { if (callEvent.mode === CallMode.Direct || callEvent.mode === CallMode.Group) { await updateLocalCallHistory(callEvent, receivedAtCounter, receivedAtMS); } else if (callEvent.mode === CallMode.Adhoc) { await updateLocalAdhocCallHistory(callEvent); } } export async function updateCallHistoryFromLocalEvent( callEvent: CallEventDetails, receivedAtCounter: number | null, receivedAtMS: number | null ): Promise { const updatedCallHistory = await updateLocalCallHistory( callEvent, receivedAtCounter, receivedAtMS ); if (updatedCallHistory == null) { return; } await updateRemoteCallHistory(updatedCallHistory); } export function updateDeletedMessages(messageIds: ReadonlyArray): void { messageIds.forEach(messageId => { const message = window.MessageCache.__DEPRECATED$getById(messageId); const conversation = message?.getConversation(); if (message == null || conversation == null) { return; } window.reduxActions.conversations.messageDeleted( messageId, message.get('conversationId') ); conversation.debouncedUpdateLastMessage(); window.MessageCache.__DEPRECATED$unregister(messageId); }); } export async function clearCallHistoryDataAndSync( latestCall: CallHistoryDetails ): Promise { try { log.info( `clearCallHistory: Clearing call history before (${latestCall.callId}, ${latestCall.timestamp})` ); const messageIds = await DataWriter.clearCallHistory(latestCall); updateDeletedMessages(messageIds); log.info('clearCallHistory: Queueing sync message'); await singleProtoJobQueue.add( MessageSender.getClearCallHistoryMessage(latestCall) ); } catch (error) { log.error('clearCallHistory: Failed to clear call history', error); } } export async function markAllCallHistoryReadAndSync( latestCall: CallHistoryDetails, inConversation: boolean ): Promise { try { log.info( `markAllCallHistoryReadAndSync: Marking call history read before (${latestCall.callId}, ${latestCall.timestamp})` ); if (inConversation) { await DataWriter.markAllCallHistoryReadInConversation(latestCall); } else { await DataWriter.markAllCallHistoryRead(latestCall); } const ourAci = window.textsecure.storage.user.getCheckedAci(); const callLogEvent = new Proto.SyncMessage.CallLogEvent({ type: inConversation ? Proto.SyncMessage.CallLogEvent.Type.MARKED_AS_READ_IN_CONVERSATION : Proto.SyncMessage.CallLogEvent.Type.MARKED_AS_READ, timestamp: Long.fromNumber(latestCall.timestamp), peerId: getBytesForPeerId(latestCall), callId: Long.fromString(latestCall.callId), }); const syncMessage = MessageSender.createSyncMessage(); syncMessage.callLogEvent = callLogEvent; const contentMessage = new Proto.Content(); contentMessage.syncMessage = syncMessage; const { ContentHint } = Proto.UnidentifiedSenderMessage.Message; log.info('markAllCallHistoryReadAndSync: Queueing sync message'); await singleProtoJobQueue.add({ contentHint: ContentHint.RESENDABLE, serviceId: ourAci, isSyncMessage: true, protoBase64: Bytes.toBase64( Proto.Content.encode(contentMessage).finish() ), type: 'callLogEventSync', urgent: false, }); } catch (error) { log.error( 'markAllCallHistoryReadAndSync: Failed to mark call history read', error ); } } export async function updateLocalGroupCallHistoryTimestamp( conversationId: string, callId: string, timestamp: number ): Promise { const conversation = window.ConversationController.get(conversationId); if (conversation == null) { return null; } const peerId = getPeerIdFromConversation(conversation.attributes); const prevCallHistory = (await DataReader.getCallHistory(callId, peerId)) ?? null; // We don't have all the details to add new call history here if (prevCallHistory != null) { log.info( 'updateLocalGroupCallHistoryTimestamp: Found previous call history:', formatCallHistory(prevCallHistory) ); } else { log.info('updateLocalGroupCallHistoryTimestamp: No previous call history'); return null; } if (timestamp >= prevCallHistory.timestamp) { log.info( 'updateLocalGroupCallHistoryTimestamp: New timestamp is later than existing call history, ignoring' ); return prevCallHistory; } const updatedCallHistory = await saveCallHistory({ callHistory: { ...prevCallHistory, timestamp, }, conversation, receivedAtCounter: null, receivedAtMS: null, }); return updatedCallHistory; }