// Copyright 2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { isEqual, isNumber } from 'lodash'; import Long from 'long'; import { CallLinkRootKey } from '@signalapp/ringrtc'; import { uuidToBytes, bytesToUuid } from '../util/uuidToBytes'; import { deriveMasterKeyFromGroupV1 } from '../Crypto'; import * as Bytes from '../Bytes'; import { deriveGroupFields, waitThenMaybeUpdateGroup, waitThenRespondToGroupV2Migration, } from '../groups'; import { assertDev, strictAssert } from '../util/assert'; import { dropNull } from '../util/dropNull'; import { missingCaseError } from '../util/missingCaseError'; import { isNotNil } from '../util/isNotNil'; import { PhoneNumberSharingMode, parsePhoneNumberSharingMode, } from '../util/phoneNumberSharingMode'; import { PhoneNumberDiscoverability, parsePhoneNumberDiscoverability, } from '../util/phoneNumberDiscoverability'; import { arePinnedConversationsEqual } from '../util/arePinnedConversationsEqual'; import type { ConversationModel } from '../models/conversations'; import { getSafeLongFromTimestamp, getTimestampFromLong, } from '../util/timestampLongUtils'; import { canHaveUsername } from '../util/getTitle'; import { get as getUniversalExpireTimer, set as setUniversalExpireTimer, } from '../util/universalExpireTimer'; import { ourProfileKeyService } from './ourProfileKey'; import { isGroupV1, isGroupV2 } from '../util/whatTypeOfConversation'; import { DurationInSeconds } from '../util/durations'; import * as preferredReactionEmoji from '../reactions/preferredReactionEmoji'; import { SignalService as Proto } from '../protobuf'; import * as log from '../logging/log'; import { normalizeStoryDistributionId } from '../types/StoryDistributionId'; import type { StoryDistributionIdString } from '../types/StoryDistributionId'; import type { ServiceIdString } from '../types/ServiceId'; import { normalizeServiceId, normalizePni, ServiceIdKind, isUntaggedPniString, toUntaggedPni, toTaggedPni, } from '../types/ServiceId'; import { normalizeAci } from '../util/normalizeAci'; import { isAciString } from '../util/isAciString'; import * as Stickers from '../types/Stickers'; import type { StoryDistributionWithMembersType, StickerPackInfoType, } from '../sql/Interface'; import { DataReader, DataWriter } from '../sql/Client'; import { MY_STORY_ID, StorySendMode } from '../types/Stories'; import { findAndDeleteOnboardingStoryIfExists } from '../util/findAndDeleteOnboardingStoryIfExists'; import { downloadOnboardingStory } from '../util/downloadOnboardingStory'; import { drop } from '../util/drop'; import { redactExtendedStorageID } from '../util/privacy'; import type { CallLinkRecord } from '../types/CallLink'; import { callLinkFromRecord, fromRootKeyBytes, getRoomIdFromRootKey, } from '../util/callLinksRingrtc'; import { CALL_LINK_DELETED_STORAGE_RECORD_TTL, fromAdminKeyBytes, toCallHistoryFromUnusedCallLink, } from '../util/callLinks'; import { isOlderThan } from '../util/timestamp'; import { callLinkRefreshJobQueue } from '../jobs/callLinkRefreshJobQueue'; const MY_STORY_BYTES = uuidToBytes(MY_STORY_ID); type RecordClass = | Proto.IAccountRecord | Proto.IContactRecord | Proto.IGroupV1Record | Proto.IGroupV2Record; export type MergeResultType = Readonly<{ hasConflict: boolean; shouldDrop?: boolean; conversation?: ConversationModel; needsProfileFetch?: boolean; updatedConversations?: ReadonlyArray; oldStorageID?: string; oldStorageVersion?: number; details: ReadonlyArray; }>; type HasConflictResultType = Readonly<{ hasConflict: boolean; details: ReadonlyArray; }>; function toRecordVerified(verified: number): Proto.ContactRecord.IdentityState { const VERIFIED_ENUM = window.textsecure.storage.protocol.VerifiedStatus; const STATE_ENUM = Proto.ContactRecord.IdentityState; switch (verified) { case VERIFIED_ENUM.VERIFIED: return STATE_ENUM.VERIFIED; case VERIFIED_ENUM.UNVERIFIED: return STATE_ENUM.UNVERIFIED; default: return STATE_ENUM.DEFAULT; } } function fromRecordVerified( verified: Proto.ContactRecord.IdentityState ): number { const VERIFIED_ENUM = window.textsecure.storage.protocol.VerifiedStatus; const STATE_ENUM = Proto.ContactRecord.IdentityState; switch (verified) { case STATE_ENUM.VERIFIED: return VERIFIED_ENUM.VERIFIED; case STATE_ENUM.UNVERIFIED: return VERIFIED_ENUM.UNVERIFIED; default: return VERIFIED_ENUM.DEFAULT; } } function addUnknownFields( record: RecordClass, conversation: ConversationModel, details: Array ): void { if (record.$unknownFields) { details.push('adding unknown fields'); conversation.set({ storageUnknownFields: Bytes.toBase64( Bytes.concatenate(record.$unknownFields) ), }); } else if (conversation.get('storageUnknownFields')) { // If the record doesn't have unknown fields attached but we have them // saved locally then we need to clear it out details.push('clearing unknown fields'); conversation.unset('storageUnknownFields'); } } function applyUnknownFields( record: RecordClass, conversation: ConversationModel ): void { const storageUnknownFields = conversation.get('storageUnknownFields'); if (storageUnknownFields) { log.info( 'storageService.applyUnknownFields: Applying unknown fields for', conversation.idForLogging() ); // eslint-disable-next-line no-param-reassign record.$unknownFields = [Bytes.fromBase64(storageUnknownFields)]; } } export async function toContactRecord( conversation: ConversationModel ): Promise { const contactRecord = new Proto.ContactRecord(); const aci = conversation.getAci(); if (aci) { contactRecord.aci = aci; } const e164 = conversation.get('e164'); if (e164) { contactRecord.serviceE164 = e164; } const username = conversation.get('username'); const ourID = window.ConversationController.getOurConversationId(); if (username && canHaveUsername(conversation.attributes, ourID)) { contactRecord.username = username; } const pni = conversation.getPni(); if (pni) { contactRecord.pni = toUntaggedPni(pni); } contactRecord.pniSignatureVerified = conversation.get('pniSignatureVerified') ?? false; const profileKey = conversation.get('profileKey'); if (profileKey) { contactRecord.profileKey = Bytes.fromBase64(String(profileKey)); } const serviceId = aci ?? pni; const identityKey = serviceId ? await window.textsecure.storage.protocol.loadIdentityKey(serviceId) : undefined; if (identityKey) { contactRecord.identityKey = identityKey; } const verified = conversation.get('verified'); if (verified) { contactRecord.identityState = toRecordVerified(Number(verified)); } const profileName = conversation.get('profileName'); if (profileName) { contactRecord.givenName = profileName; } const profileFamilyName = conversation.get('profileFamilyName'); if (profileFamilyName) { contactRecord.familyName = profileFamilyName; } const nicknameGivenName = conversation.get('nicknameGivenName'); const nicknameFamilyName = conversation.get('nicknameFamilyName'); if (nicknameGivenName || nicknameFamilyName) { contactRecord.nickname = { given: nicknameGivenName, family: nicknameFamilyName, }; } const note = conversation.get('note'); if (note) { contactRecord.note = note; } const systemGivenName = conversation.get('systemGivenName'); if (systemGivenName) { contactRecord.systemGivenName = systemGivenName; } const systemFamilyName = conversation.get('systemFamilyName'); if (systemFamilyName) { contactRecord.systemFamilyName = systemFamilyName; } const systemNickname = conversation.get('systemNickname'); if (systemNickname) { contactRecord.systemNickname = systemNickname; } contactRecord.blocked = conversation.isBlocked(); contactRecord.hidden = conversation.get('removalStage') !== undefined; contactRecord.whitelisted = Boolean(conversation.get('profileSharing')); contactRecord.archived = Boolean(conversation.get('isArchived')); contactRecord.markedUnread = Boolean(conversation.get('markedUnread')); contactRecord.mutedUntilTimestamp = getSafeLongFromTimestamp( conversation.get('muteExpiresAt') ); if (conversation.get('hideStory') !== undefined) { contactRecord.hideStory = Boolean(conversation.get('hideStory')); } contactRecord.unregisteredAtTimestamp = getSafeLongFromTimestamp( conversation.get('firstUnregisteredAt') ); applyUnknownFields(contactRecord, conversation); return contactRecord; } export function toAccountRecord( conversation: ConversationModel ): Proto.AccountRecord { const accountRecord = new Proto.AccountRecord(); if (conversation.get('profileKey')) { accountRecord.profileKey = Bytes.fromBase64( String(conversation.get('profileKey')) ); } if (conversation.get('profileName')) { accountRecord.givenName = conversation.get('profileName') || ''; } if (conversation.get('profileFamilyName')) { accountRecord.familyName = conversation.get('profileFamilyName') || ''; } const avatarUrl = window.storage.get('avatarUrl'); if (avatarUrl !== undefined) { accountRecord.avatarUrl = avatarUrl; } const username = conversation.get('username'); if (username !== undefined) { accountRecord.username = username; } accountRecord.noteToSelfArchived = Boolean(conversation.get('isArchived')); accountRecord.noteToSelfMarkedUnread = Boolean( conversation.get('markedUnread') ); accountRecord.readReceipts = Boolean(window.Events.getReadReceiptSetting()); accountRecord.sealedSenderIndicators = Boolean( window.storage.get('sealedSenderIndicators') ); accountRecord.typingIndicators = Boolean( window.Events.getTypingIndicatorSetting() ); accountRecord.linkPreviews = Boolean(window.Events.getLinkPreviewSetting()); const preferContactAvatars = window.storage.get('preferContactAvatars'); if (preferContactAvatars !== undefined) { accountRecord.preferContactAvatars = Boolean(preferContactAvatars); } const primarySendsSms = window.storage.get('primarySendsSms'); if (primarySendsSms !== undefined) { accountRecord.primarySendsSms = Boolean(primarySendsSms); } const rawPreferredReactionEmoji = window.storage.get( 'preferredReactionEmoji' ); if (preferredReactionEmoji.canBeSynced(rawPreferredReactionEmoji)) { accountRecord.preferredReactionEmoji = rawPreferredReactionEmoji; } const universalExpireTimer = getUniversalExpireTimer(); if (universalExpireTimer) { accountRecord.universalExpireTimer = Number(universalExpireTimer); } const PHONE_NUMBER_SHARING_MODE_ENUM = Proto.AccountRecord.PhoneNumberSharingMode; const phoneNumberSharingMode = parsePhoneNumberSharingMode( window.storage.get('phoneNumberSharingMode') ); switch (phoneNumberSharingMode) { case PhoneNumberSharingMode.Everybody: accountRecord.phoneNumberSharingMode = PHONE_NUMBER_SHARING_MODE_ENUM.EVERYBODY; break; case PhoneNumberSharingMode.ContactsOnly: case PhoneNumberSharingMode.Nobody: accountRecord.phoneNumberSharingMode = PHONE_NUMBER_SHARING_MODE_ENUM.NOBODY; break; default: throw missingCaseError(phoneNumberSharingMode); } const phoneNumberDiscoverability = parsePhoneNumberDiscoverability( window.storage.get('phoneNumberDiscoverability') ); switch (phoneNumberDiscoverability) { case PhoneNumberDiscoverability.Discoverable: accountRecord.notDiscoverableByPhoneNumber = false; break; case PhoneNumberDiscoverability.NotDiscoverable: accountRecord.notDiscoverableByPhoneNumber = true; break; default: throw missingCaseError(phoneNumberDiscoverability); } const pinnedConversations = window.storage .get('pinnedConversationIds', new Array()) .map(id => { const pinnedConversation = window.ConversationController.get(id); if (pinnedConversation) { const pinnedConversationRecord = new Proto.AccountRecord.PinnedConversation(); if (pinnedConversation.get('type') === 'private') { pinnedConversationRecord.identifier = 'contact'; pinnedConversationRecord.contact = { serviceId: pinnedConversation.getServiceId(), e164: pinnedConversation.get('e164'), }; } else if (isGroupV1(pinnedConversation.attributes)) { pinnedConversationRecord.identifier = 'legacyGroupId'; const groupId = pinnedConversation.get('groupId'); if (!groupId) { throw new Error( 'toAccountRecord: trying to pin a v1 Group without groupId' ); } pinnedConversationRecord.legacyGroupId = Bytes.fromBinary(groupId); } else if (isGroupV2(pinnedConversation.attributes)) { pinnedConversationRecord.identifier = 'groupMasterKey'; const masterKey = pinnedConversation.get('masterKey'); if (!masterKey) { throw new Error( 'toAccountRecord: trying to pin a v2 Group without masterKey' ); } pinnedConversationRecord.groupMasterKey = Bytes.fromBase64(masterKey); } return pinnedConversationRecord; } return undefined; }) .filter( ( pinnedConversationClass ): pinnedConversationClass is Proto.AccountRecord.PinnedConversation => pinnedConversationClass !== undefined ); accountRecord.pinnedConversations = pinnedConversations; const subscriberId = window.storage.get('subscriberId'); if (Bytes.isNotEmpty(subscriberId)) { accountRecord.subscriberId = subscriberId; } const subscriberCurrencyCode = window.storage.get( 'backupsSubscriberCurrencyCode' ); if (typeof subscriberCurrencyCode === 'string') { accountRecord.subscriberCurrencyCode = subscriberCurrencyCode; } const donorSubscriptionManuallyCancelled = window.storage.get( 'donorSubscriptionManuallyCancelled' ); if (typeof donorSubscriptionManuallyCancelled === 'boolean') { accountRecord.donorSubscriptionManuallyCancelled = donorSubscriptionManuallyCancelled; } const backupsSubscriberId = window.storage.get('backupsSubscriberId'); if (Bytes.isNotEmpty(backupsSubscriberId)) { accountRecord.backupsSubscriberId = backupsSubscriberId; } const backupsSubscriberCurrencyCode = window.storage.get( 'backupsSubscriberCurrencyCode' ); if (typeof backupsSubscriberCurrencyCode === 'string') { accountRecord.backupsSubscriberCurrencyCode = backupsSubscriberCurrencyCode; } const backupsSubscriptionManuallyCancelled = window.storage.get( 'backupsSubscriptionManuallyCancelled' ); if (typeof backupsSubscriptionManuallyCancelled === 'boolean') { accountRecord.backupsSubscriptionManuallyCancelled = backupsSubscriptionManuallyCancelled; } const displayBadgesOnProfile = window.storage.get('displayBadgesOnProfile'); if (displayBadgesOnProfile !== undefined) { accountRecord.displayBadgesOnProfile = displayBadgesOnProfile; } const keepMutedChatsArchived = window.storage.get('keepMutedChatsArchived'); if (keepMutedChatsArchived !== undefined) { accountRecord.keepMutedChatsArchived = keepMutedChatsArchived; } const hasSetMyStoriesPrivacy = window.storage.get('hasSetMyStoriesPrivacy'); if (hasSetMyStoriesPrivacy !== undefined) { accountRecord.hasSetMyStoriesPrivacy = hasSetMyStoriesPrivacy; } const hasViewedOnboardingStory = window.storage.get( 'hasViewedOnboardingStory' ); if (hasViewedOnboardingStory !== undefined) { accountRecord.hasViewedOnboardingStory = hasViewedOnboardingStory; } const hasCompletedUsernameOnboarding = window.storage.get( 'hasCompletedUsernameOnboarding' ); if (hasCompletedUsernameOnboarding !== undefined) { accountRecord.hasCompletedUsernameOnboarding = hasCompletedUsernameOnboarding; } const hasSeenGroupStoryEducationSheet = window.storage.get( 'hasSeenGroupStoryEducationSheet' ); if (hasSeenGroupStoryEducationSheet !== undefined) { accountRecord.hasSeenGroupStoryEducationSheet = hasSeenGroupStoryEducationSheet; } const hasStoriesDisabled = window.storage.get('hasStoriesDisabled'); accountRecord.storiesDisabled = hasStoriesDisabled === true; const storyViewReceiptsEnabled = window.storage.get( 'storyViewReceiptsEnabled' ); if (storyViewReceiptsEnabled !== undefined) { accountRecord.storyViewReceiptsEnabled = storyViewReceiptsEnabled ? Proto.OptionalBool.ENABLED : Proto.OptionalBool.DISABLED; } else { accountRecord.storyViewReceiptsEnabled = Proto.OptionalBool.UNSET; } // Username link { const color = window.storage.get('usernameLinkColor'); const linkData = window.storage.get('usernameLink'); if (linkData?.entropy.length && linkData?.serverId.length) { accountRecord.usernameLink = { color, entropy: linkData.entropy, serverId: linkData.serverId, }; } } applyUnknownFields(accountRecord, conversation); return accountRecord; } export function toGroupV1Record( conversation: ConversationModel ): Proto.GroupV1Record { const groupV1Record = new Proto.GroupV1Record(); groupV1Record.id = Bytes.fromBinary(String(conversation.get('groupId'))); groupV1Record.blocked = conversation.isBlocked(); groupV1Record.whitelisted = Boolean(conversation.get('profileSharing')); groupV1Record.archived = Boolean(conversation.get('isArchived')); groupV1Record.markedUnread = Boolean(conversation.get('markedUnread')); groupV1Record.mutedUntilTimestamp = getSafeLongFromTimestamp( conversation.get('muteExpiresAt') ); applyUnknownFields(groupV1Record, conversation); return groupV1Record; } export function toGroupV2Record( conversation: ConversationModel ): Proto.GroupV2Record { const groupV2Record = new Proto.GroupV2Record(); const masterKey = conversation.get('masterKey'); if (masterKey !== undefined) { groupV2Record.masterKey = Bytes.fromBase64(masterKey); } groupV2Record.blocked = conversation.isBlocked(); groupV2Record.whitelisted = Boolean(conversation.get('profileSharing')); groupV2Record.archived = Boolean(conversation.get('isArchived')); groupV2Record.markedUnread = Boolean(conversation.get('markedUnread')); groupV2Record.mutedUntilTimestamp = getSafeLongFromTimestamp( conversation.get('muteExpiresAt') ); groupV2Record.dontNotifyForMentionsIfMuted = Boolean( conversation.get('dontNotifyForMentionsIfMuted') ); groupV2Record.hideStory = Boolean(conversation.get('hideStory')); const storySendMode = conversation.get('storySendMode'); if (storySendMode !== undefined) { if (storySendMode === StorySendMode.IfActive) { groupV2Record.storySendMode = Proto.GroupV2Record.StorySendMode.DEFAULT; } else if (storySendMode === StorySendMode.Never) { groupV2Record.storySendMode = Proto.GroupV2Record.StorySendMode.DISABLED; } else if (storySendMode === StorySendMode.Always) { groupV2Record.storySendMode = Proto.GroupV2Record.StorySendMode.ENABLED; } else { throw missingCaseError(storySendMode); } } applyUnknownFields(groupV2Record, conversation); return groupV2Record; } export function toStoryDistributionListRecord( storyDistributionList: StoryDistributionWithMembersType ): Proto.StoryDistributionListRecord { const storyDistributionListRecord = new Proto.StoryDistributionListRecord(); storyDistributionListRecord.identifier = uuidToBytes( storyDistributionList.id ); storyDistributionListRecord.name = storyDistributionList.name; storyDistributionListRecord.deletedAtTimestamp = getSafeLongFromTimestamp( storyDistributionList.deletedAtTimestamp ); storyDistributionListRecord.allowsReplies = Boolean( storyDistributionList.allowsReplies ); storyDistributionListRecord.isBlockList = Boolean( storyDistributionList.isBlockList ); storyDistributionListRecord.recipientServiceIds = storyDistributionList.members; if (storyDistributionList.storageUnknownFields) { storyDistributionListRecord.$unknownFields = [ storyDistributionList.storageUnknownFields, ]; } return storyDistributionListRecord; } export function toStickerPackRecord( stickerPack: StickerPackInfoType ): Proto.StickerPackRecord { const stickerPackRecord = new Proto.StickerPackRecord(); stickerPackRecord.packId = Bytes.fromHex(stickerPack.id); if (stickerPack.uninstalledAt !== undefined) { stickerPackRecord.deletedAtTimestamp = Long.fromNumber( stickerPack.uninstalledAt ); } else { stickerPackRecord.packKey = Bytes.fromBase64(stickerPack.key); if (stickerPack.position) { stickerPackRecord.position = stickerPack.position; } } if (stickerPack.storageUnknownFields) { stickerPackRecord.$unknownFields = [stickerPack.storageUnknownFields]; } return stickerPackRecord; } // callLinkDbRecord exposes additional fields not available on CallLinkType export function toCallLinkRecord( callLinkDbRecord: CallLinkRecord ): Proto.CallLinkRecord { strictAssert(callLinkDbRecord.rootKey, 'toCallLinkRecord: no rootKey'); const callLinkRecord = new Proto.CallLinkRecord(); callLinkRecord.rootKey = callLinkDbRecord.rootKey; if (callLinkDbRecord.deleted === 1) { // adminKey is intentionally omitted for deleted call links. callLinkRecord.deletedAtTimestampMs = Long.fromNumber( callLinkDbRecord.deletedAt || new Date().getTime() ); } else { strictAssert( callLinkDbRecord.adminKey, 'toCallLinkRecord: no adminPasskey' ); callLinkRecord.adminPasskey = callLinkDbRecord.adminKey; } if (callLinkDbRecord.storageUnknownFields) { callLinkRecord.$unknownFields = [callLinkDbRecord.storageUnknownFields]; } return callLinkRecord; } type MessageRequestCapableRecord = Proto.IContactRecord | Proto.IGroupV1Record; function applyMessageRequestState( record: MessageRequestCapableRecord, conversation: ConversationModel ): void { const messageRequestEnum = Proto.SyncMessage.MessageRequestResponse.Type; if (record.blocked) { void conversation.applyMessageRequestResponse(messageRequestEnum.BLOCK, { fromSync: true, viaStorageServiceSync: true, }); } else if (record.whitelisted) { // unblocking is also handled by this function which is why the next // condition is part of the else-if and not separate void conversation.applyMessageRequestResponse(messageRequestEnum.ACCEPT, { fromSync: true, viaStorageServiceSync: true, }); } else if (!record.blocked) { // if the condition above failed the state could still be blocked=false // in which case we should unblock the conversation conversation.unblock({ viaStorageServiceSync: true }); } if (record.whitelisted === false) { conversation.disableProfileSharing({ viaStorageServiceSync: true }); } } type RecordClassObject = { // eslint-disable-next-line @typescript-eslint/no-explicit-any [key: string]: any; }; function doRecordsConflict( localRecord: RecordClassObject, remoteRecord: RecordClassObject ): HasConflictResultType { const details = new Array(); for (const key of Object.keys(remoteRecord)) { const localValue = localRecord[key]; const remoteValue = remoteRecord[key]; // Sometimes we have a ByteBuffer and an Uint8Array, this ensures that we // are comparing them both equally by converting them into base64 string. if (localValue instanceof Uint8Array) { const areEqual = Bytes.areEqual(localValue, remoteValue); if (!areEqual) { details.push(`key=${key}: different bytes`); } continue; } // If both types are Long we can use Long's equals to compare them if (Long.isLong(localValue) || typeof localValue === 'number') { if (!Long.isLong(remoteValue) && typeof remoteValue !== 'number') { details.push(`key=${key}: type mismatch`); continue; } const areEqual = Long.fromValue(localValue).equals( Long.fromValue(remoteValue) ); if (!areEqual) { details.push(`key=${key}: different integers`); } continue; } if (key === 'pinnedConversations') { const areEqual = arePinnedConversationsEqual(localValue, remoteValue); if (!areEqual) { details.push('pinnedConversations'); } continue; } if (localValue === remoteValue) { continue; } // Sometimes we get `null` values from Protobuf and they should default to // false, empty string, or 0 for these records we do not count them as // conflicting. if ( (!remoteValue || (Long.isLong(remoteValue) && remoteValue.isZero())) && (!localValue || (Long.isLong(localValue) && localValue.isZero())) ) { continue; } const areEqual = isEqual(localValue, remoteValue); if (!areEqual) { details.push(`key=${key}: different values`); } } return { hasConflict: details.length > 0, details, }; } function doesRecordHavePendingChanges( mergedRecord: RecordClass, serviceRecord: RecordClass, conversation: ConversationModel ): HasConflictResultType { const shouldSync = Boolean(conversation.get('needsStorageServiceSync')); if (!shouldSync) { return { hasConflict: false, details: [] }; } const { hasConflict, details } = doRecordsConflict( mergedRecord, serviceRecord ); if (!hasConflict) { conversation.set({ needsStorageServiceSync: false }); } return { hasConflict, details, }; } export async function mergeGroupV1Record( storageID: string, storageVersion: number, groupV1Record: Proto.IGroupV1Record ): Promise { const redactedStorageID = redactExtendedStorageID({ storageID, storageVersion, }); if (!groupV1Record.id) { throw new Error(`No ID for ${redactedStorageID}`); } const groupId = Bytes.toBinary(groupV1Record.id); let details = new Array(); // Attempt to fetch an existing group pertaining to the `groupId` or create // a new group and populate it with the attributes from the record. let conversation = window.ConversationController.get(groupId); // Because ConversationController.get retrieves all types of records we // may sometimes have a situation where we get a record of groupv1 type // where the binary representation of its ID matches a v2 record in memory. // Here we ensure that the record we're about to process is GV1 otherwise // we drop the update. if (conversation && !isGroupV1(conversation.attributes)) { throw new Error( `Record has group type mismatch ${conversation.idForLogging()}` ); } if (!conversation) { // It's possible this group was migrated to a GV2 if so we attempt to // retrieve the master key and find the conversation locally. If we // are successful then we continue setting and applying state. const masterKeyBuffer = deriveMasterKeyFromGroupV1(groupV1Record.id); const fields = deriveGroupFields(masterKeyBuffer); const derivedGroupV2Id = Bytes.toBase64(fields.id); details.push( 'failed to find group by v1 id ' + `attempting lookup by v2 groupv2(${derivedGroupV2Id})` ); conversation = window.ConversationController.get(derivedGroupV2Id); } if (!conversation) { if (groupV1Record.id.byteLength !== 16) { throw new Error('Not a valid gv1'); } conversation = await window.ConversationController.getOrCreateAndWait( groupId, 'group' ); details.push('created a new group locally'); } const oldStorageID = conversation.get('storageID'); const oldStorageVersion = conversation.get('storageVersion'); if (!isGroupV1(conversation.attributes)) { details.push('GV1 record for GV2 group, dropping'); return { // Note: conflicts cause immediate uploads, but we should upload // only in response to user's action. hasConflict: false, shouldDrop: true, conversation, oldStorageID, oldStorageVersion, details, }; } conversation.set({ isArchived: Boolean(groupV1Record.archived), markedUnread: Boolean(groupV1Record.markedUnread), storageID, storageVersion, }); conversation.setMuteExpiration( getTimestampFromLong(groupV1Record.mutedUntilTimestamp), { viaStorageServiceSync: true, } ); applyMessageRequestState(groupV1Record, conversation); let hasPendingChanges: boolean; if (isGroupV1(conversation.attributes)) { addUnknownFields(groupV1Record, conversation, details); const { hasConflict, details: extraDetails } = doesRecordHavePendingChanges( toGroupV1Record(conversation), groupV1Record, conversation ); details = details.concat(extraDetails); hasPendingChanges = hasConflict; } else { // We cannot preserve unknown fields if local group is V2 and the remote is // still V1, because the storageItem that we'll put into manifest will have // a different record type. // We want to upgrade group in the storage after merging it. hasPendingChanges = true; details.push('marking v1 group for an update to v2'); } return { hasConflict: hasPendingChanges, conversation, oldStorageID, oldStorageVersion, details, updatedConversations: [conversation], }; } function getGroupV2Conversation( masterKeyBuffer: Uint8Array ): ConversationModel { const groupFields = deriveGroupFields(masterKeyBuffer); const groupId = Bytes.toBase64(groupFields.id); const masterKey = Bytes.toBase64(masterKeyBuffer); const secretParams = Bytes.toBase64(groupFields.secretParams); const publicParams = Bytes.toBase64(groupFields.publicParams); // First we check for an existing GroupV2 group const groupV2 = window.ConversationController.get(groupId); if (groupV2) { groupV2.maybeRepairGroupV2({ masterKey, secretParams, publicParams, }); return groupV2; } // Then check for V1 group with matching derived GV2 id const groupV1 = window.ConversationController.getByDerivedGroupV2Id(groupId); if (groupV1) { return groupV1; } const conversationId = window.ConversationController.ensureGroup(groupId, { // Note: We don't set active_at, because we don't want the group to show until // we have information about it beyond these initial details. // see maybeUpdateGroup(). groupVersion: 2, masterKey, secretParams, publicParams, }); const conversation = window.ConversationController.get(conversationId); if (!conversation) { throw new Error( `getGroupV2Conversation: Failed to create conversation for groupv2(${groupId})` ); } return conversation; } export async function mergeGroupV2Record( storageID: string, storageVersion: number, groupV2Record: Proto.IGroupV2Record ): Promise { const redactedStorageID = redactExtendedStorageID({ storageID, storageVersion, }); if (!groupV2Record.masterKey) { throw new Error(`No master key for ${redactedStorageID}`); } const masterKeyBuffer = groupV2Record.masterKey; const conversation = getGroupV2Conversation(masterKeyBuffer); const oldStorageID = conversation.get('storageID'); const oldStorageVersion = conversation.get('storageVersion'); const recordStorySendMode = groupV2Record.storySendMode ?? Proto.GroupV2Record.StorySendMode.DEFAULT; let storySendMode: StorySendMode; if (recordStorySendMode === Proto.GroupV2Record.StorySendMode.DEFAULT) { storySendMode = StorySendMode.IfActive; } else if ( recordStorySendMode === Proto.GroupV2Record.StorySendMode.DISABLED ) { storySendMode = StorySendMode.Never; } else if ( recordStorySendMode === Proto.GroupV2Record.StorySendMode.ENABLED ) { storySendMode = StorySendMode.Always; } else { throw missingCaseError(recordStorySendMode); } conversation.set({ hideStory: Boolean(groupV2Record.hideStory), isArchived: Boolean(groupV2Record.archived), markedUnread: Boolean(groupV2Record.markedUnread), dontNotifyForMentionsIfMuted: Boolean( groupV2Record.dontNotifyForMentionsIfMuted ), storageID, storageVersion, storySendMode, }); conversation.setMuteExpiration( getTimestampFromLong(groupV2Record.mutedUntilTimestamp), { viaStorageServiceSync: true, } ); applyMessageRequestState(groupV2Record, conversation); let details = new Array(); addUnknownFields(groupV2Record, conversation, details); const { hasConflict, details: extraDetails } = doesRecordHavePendingChanges( toGroupV2Record(conversation), groupV2Record, conversation ); details = details.concat(extraDetails); const isGroupNewToUs = !isNumber(conversation.get('revision')); const isFirstSync = !window.storage.get('storageFetchComplete'); const dropInitialJoinMessage = isFirstSync; if (isGroupV1(conversation.attributes)) { // If we found a GroupV1 conversation from this incoming GroupV2 record, we need to // migrate it! // We don't await this because this could take a very long time, waiting for queues to // empty, etc. void waitThenRespondToGroupV2Migration({ conversation, }); } else if (isGroupNewToUs) { // We don't need to update GroupV2 groups all the time. We fetch group state the first // time we hear about these groups, from then on we rely on incoming messages or // the user opening that conversation. // We don't await this because this could take a very long time, waiting for queues to // empty, etc. void waitThenMaybeUpdateGroup( { conversation, dropInitialJoinMessage, }, { viaFirstStorageSync: isFirstSync } ); } return { hasConflict, conversation, updatedConversations: [conversation], oldStorageID, oldStorageVersion, details, }; } export async function mergeContactRecord( storageID: string, storageVersion: number, originalContactRecord: Proto.IContactRecord ): Promise { const contactRecord = { ...originalContactRecord, aci: originalContactRecord.aci ? normalizeAci(originalContactRecord.aci, 'ContactRecord.aci') : undefined, pni: originalContactRecord.pni && isUntaggedPniString(originalContactRecord.pni) ? normalizePni( toTaggedPni(originalContactRecord.pni), 'ContactRecord.pni' ) : undefined, }; const e164 = dropNull(contactRecord.serviceE164); const { aci } = contactRecord; const pni = dropNull(contactRecord.pni); const pniSignatureVerified = contactRecord.pniSignatureVerified || false; const serviceId = aci || pni; // All contacts must have UUID if (!serviceId) { return { hasConflict: false, shouldDrop: true, details: ['no uuid'] }; } // Contacts should not have PNI as ACI if (aci && !isAciString(aci)) { return { hasConflict: false, shouldDrop: true, details: ['invalid aci'] }; } if ( window.storage.user.getOurServiceIdKind(serviceId) !== ServiceIdKind.Unknown ) { return { hasConflict: false, shouldDrop: true, details: ['our own uuid'] }; } const { conversation } = window.ConversationController.maybeMergeContacts({ aci, e164, pni, fromPniSignature: pniSignatureVerified, reason: 'mergeContactRecord', }); // We're going to ignore this; it's likely a PNI-only contact we've already merged if (conversation.getServiceId() !== serviceId) { const previousStorageID = conversation.get('storageID'); const redactedpreviousStorageID = previousStorageID ? redactExtendedStorageID({ storageID: previousStorageID, storageVersion: conversation.get('storageVersion'), }) : ''; log.warn( `mergeContactRecord: ${conversation.idForLogging()} ` + `with storageId ${redactedpreviousStorageID} ` + `had serviceId that didn't match provided serviceId ${serviceId}` ); return { hasConflict: false, shouldDrop: true, details: [], }; } await conversation.updateUsername(dropNull(contactRecord.username), { shouldSave: false, }); let needsProfileFetch = false; if (contactRecord.profileKey && contactRecord.profileKey.length > 0) { needsProfileFetch = await conversation.setProfileKey( Bytes.toBase64(contactRecord.profileKey), { viaStorageServiceSync: true, reason: 'mergeContactRecord' } ); } let details = new Array(); const remoteName = dropNull(contactRecord.givenName); const remoteFamilyName = dropNull(contactRecord.familyName); const localName = conversation.get('profileName'); const localFamilyName = conversation.get('profileFamilyName'); if ( remoteName && (localName !== remoteName || localFamilyName !== remoteFamilyName) ) { // Local name doesn't match remote name, fetch profile if (localName) { drop( conversation.getProfiles().catch(() => { /* nothing to do here; logging already happened */ }) ); details.push('refreshing profile'); } else { conversation.set({ profileName: remoteName, profileFamilyName: remoteFamilyName, }); details.push('updated profile name'); } } conversation.set({ systemGivenName: dropNull(contactRecord.systemGivenName), systemFamilyName: dropNull(contactRecord.systemFamilyName), systemNickname: dropNull(contactRecord.systemNickname), nicknameGivenName: dropNull(contactRecord.nickname?.given), nicknameFamilyName: dropNull(contactRecord.nickname?.family), note: dropNull(contactRecord.note), }); // https://github.com/signalapp/Signal-Android/blob/fc3db538bcaa38dc149712a483d3032c9c1f3998/app/src/main/java/org/thoughtcrime/securesms/database/RecipientDatabase.kt#L921-L936 if (contactRecord.identityKey) { const verified = await conversation.safeGetVerified(); let { identityState } = contactRecord; if (identityState == null) { details.push('identity state was null, reverting to default state'); identityState = Proto.ContactRecord.IdentityState.DEFAULT; } const newVerified = fromRecordVerified(identityState); const needsNotification = await window.textsecure.storage.protocol.updateIdentityAfterSync( serviceId, newVerified, contactRecord.identityKey ); if (verified !== newVerified) { details.push( `updating verified state from=${verified} to=${newVerified}` ); conversation.set({ verified: newVerified }); } const VERIFIED_ENUM = window.textsecure.storage.protocol.VerifiedStatus; if (needsNotification) { details.push('adding a verified notification'); await conversation.addVerifiedChange( conversation.id, newVerified === VERIFIED_ENUM.VERIFIED, { local: false } ); } } applyMessageRequestState(contactRecord, conversation); addUnknownFields(contactRecord, conversation, details); const oldStorageID = conversation.get('storageID'); const oldStorageVersion = conversation.get('storageVersion'); conversation.set({ hideStory: Boolean(contactRecord.hideStory), isArchived: Boolean(contactRecord.archived), markedUnread: Boolean(contactRecord.markedUnread), storageID, storageVersion, }); if (contactRecord.hidden) { await conversation.removeContact({ viaStorageServiceSync: true, shouldSave: false, }); } else { await conversation.restoreContact({ viaStorageServiceSync: true, shouldSave: false, }); } conversation.setMuteExpiration( getTimestampFromLong(contactRecord.mutedUntilTimestamp), { viaStorageServiceSync: true, } ); if ( !contactRecord.unregisteredAtTimestamp || contactRecord.unregisteredAtTimestamp.equals(0) ) { conversation.setRegistered({ fromStorageService: true, shouldSave: false }); } else { conversation.setUnregistered({ timestamp: getTimestampFromLong(contactRecord.unregisteredAtTimestamp), fromStorageService: true, shouldSave: false, }); } const { hasConflict, details: extraDetails } = doesRecordHavePendingChanges( await toContactRecord(conversation), contactRecord, conversation ); details = details.concat(extraDetails); return { hasConflict, conversation, updatedConversations: [conversation], needsProfileFetch, oldStorageID, oldStorageVersion, details, }; } export async function mergeAccountRecord( storageID: string, storageVersion: number, accountRecord: Proto.IAccountRecord ): Promise { let details = new Array(); const { linkPreviews, notDiscoverableByPhoneNumber, noteToSelfArchived, noteToSelfMarkedUnread, phoneNumberSharingMode, pinnedConversations, profileKey, readReceipts, sealedSenderIndicators, typingIndicators, preferContactAvatars, primarySendsSms, universalExpireTimer, preferredReactionEmoji: rawPreferredReactionEmoji, subscriberId, subscriberCurrencyCode, donorSubscriptionManuallyCancelled, backupsSubscriberId, backupsSubscriberCurrencyCode, backupsSubscriptionManuallyCancelled, displayBadgesOnProfile, keepMutedChatsArchived, hasCompletedUsernameOnboarding, hasSeenGroupStoryEducationSheet, hasSetMyStoriesPrivacy, hasViewedOnboardingStory, storiesDisabled, storyViewReceiptsEnabled, username, usernameLink, } = accountRecord; const updatedConversations = new Array(); await window.storage.put('read-receipt-setting', Boolean(readReceipts)); if (typeof sealedSenderIndicators === 'boolean') { await window.storage.put('sealedSenderIndicators', sealedSenderIndicators); } if (typeof typingIndicators === 'boolean') { await window.storage.put('typingIndicators', typingIndicators); } if (typeof linkPreviews === 'boolean') { await window.storage.put('linkPreviews', linkPreviews); } if (typeof preferContactAvatars === 'boolean') { const previous = window.storage.get('preferContactAvatars'); await window.storage.put('preferContactAvatars', preferContactAvatars); if (Boolean(previous) !== Boolean(preferContactAvatars)) { await window.ConversationController.forceRerender(); } } if (typeof primarySendsSms === 'boolean') { await window.storage.put('primarySendsSms', primarySendsSms); } if (preferredReactionEmoji.canBeSynced(rawPreferredReactionEmoji)) { const localPreferredReactionEmoji = window.storage.get('preferredReactionEmoji') || []; if (!isEqual(localPreferredReactionEmoji, rawPreferredReactionEmoji)) { log.warn( 'storageService: remote and local preferredReactionEmoji do not match', localPreferredReactionEmoji.length, rawPreferredReactionEmoji.length ); } await window.storage.put( 'preferredReactionEmoji', rawPreferredReactionEmoji ); } void setUniversalExpireTimer( DurationInSeconds.fromSeconds(universalExpireTimer || 0) ); const PHONE_NUMBER_SHARING_MODE_ENUM = Proto.AccountRecord.PhoneNumberSharingMode; let phoneNumberSharingModeToStore: PhoneNumberSharingMode; switch (phoneNumberSharingMode) { case undefined: case null: case PHONE_NUMBER_SHARING_MODE_ENUM.EVERYBODY: phoneNumberSharingModeToStore = PhoneNumberSharingMode.Everybody; break; case PHONE_NUMBER_SHARING_MODE_ENUM.UNKNOWN: case PHONE_NUMBER_SHARING_MODE_ENUM.NOBODY: phoneNumberSharingModeToStore = PhoneNumberSharingMode.Nobody; break; default: assertDev( false, `storageService.mergeAccountRecord: Got an unexpected phone number sharing mode: ${phoneNumberSharingMode}. Falling back to default` ); phoneNumberSharingModeToStore = PhoneNumberSharingMode.Everybody; break; } await window.storage.put( 'phoneNumberSharingMode', phoneNumberSharingModeToStore ); const discoverability = notDiscoverableByPhoneNumber ? PhoneNumberDiscoverability.NotDiscoverable : PhoneNumberDiscoverability.Discoverable; await window.storage.put('phoneNumberDiscoverability', discoverability); if (profileKey) { void ourProfileKeyService.set(profileKey); } if (pinnedConversations) { const modelPinnedConversations = window .getConversations() .filter(conversation => Boolean(conversation.get('isPinned'))); const modelPinnedConversationIds = modelPinnedConversations.map( conversation => conversation.get('id') ); const missingStoragePinnedConversationIds = window.storage .get('pinnedConversationIds', new Array()) .filter(id => !modelPinnedConversationIds.includes(id)); if (missingStoragePinnedConversationIds.length !== 0) { log.warn( 'mergeAccountRecord: pinnedConversationIds in storage does not match pinned Conversation models' ); } const locallyPinnedConversations = modelPinnedConversations.concat( missingStoragePinnedConversationIds .map(conversationId => window.ConversationController.get(conversationId) ) .filter( (conversation): conversation is ConversationModel => conversation !== undefined ) ); details.push( `local pinned=${locallyPinnedConversations.length}`, `remote pinned=${pinnedConversations.length}` ); const remotelyPinnedConversations = pinnedConversations .map(({ contact, legacyGroupId, groupMasterKey }) => { let conversation: ConversationModel | undefined; if (contact) { if (!contact.serviceId && !contact.e164) { log.error( 'storageService.mergeAccountRecord: No serviceId or e164 on contact' ); return undefined; } conversation = window.ConversationController.lookupOrCreate({ serviceId: contact.serviceId ? normalizeServiceId( contact.serviceId, 'AccountRecord.pin.serviceId' ) : undefined, e164: contact.e164, reason: 'storageService.mergeAccountRecord', }); } else if (legacyGroupId && legacyGroupId.length) { const groupId = Bytes.toBinary(legacyGroupId); conversation = window.ConversationController.get(groupId); } else if (groupMasterKey && groupMasterKey.length) { const groupFields = deriveGroupFields(groupMasterKey); const groupId = Bytes.toBase64(groupFields.id); conversation = window.ConversationController.get(groupId); } else { log.error( 'storageService.mergeAccountRecord: Invalid identifier received' ); } if (!conversation) { log.error( 'storageService.mergeAccountRecord: missing conversation id.' ); return undefined; } return conversation; }) .filter(isNotNil); const remotelyPinnedConversationIds = remotelyPinnedConversations.map( ({ id }) => id ); const conversationsToUnpin = locallyPinnedConversations.filter( ({ id }) => !remotelyPinnedConversationIds.includes(id) ); details.push( `unpinning=${conversationsToUnpin.length}`, `pinning=${remotelyPinnedConversations.length}` ); conversationsToUnpin.forEach(conversation => { conversation.set({ isPinned: false }); updatedConversations.push(conversation); }); remotelyPinnedConversations.forEach(conversation => { conversation.set({ isPinned: true, isArchived: false }); updatedConversations.push(conversation); }); await window.storage.put( 'pinnedConversationIds', remotelyPinnedConversationIds ); } if (Bytes.isNotEmpty(subscriberId)) { await window.storage.put('subscriberId', subscriberId); } if (typeof subscriberCurrencyCode === 'string') { await window.storage.put('subscriberCurrencyCode', subscriberCurrencyCode); } if (donorSubscriptionManuallyCancelled != null) { await window.storage.put( 'donorSubscriptionManuallyCancelled', donorSubscriptionManuallyCancelled ); } if (Bytes.isNotEmpty(backupsSubscriberId)) { await window.storage.put('backupsSubscriberId', backupsSubscriberId); } if (typeof backupsSubscriberCurrencyCode === 'string') { await window.storage.put( 'backupsSubscriberCurrencyCode', backupsSubscriberCurrencyCode ); } if (backupsSubscriptionManuallyCancelled != null) { await window.storage.put( 'backupsSubscriptionManuallyCancelled', backupsSubscriptionManuallyCancelled ); } await window.storage.put( 'displayBadgesOnProfile', Boolean(displayBadgesOnProfile) ); await window.storage.put( 'keepMutedChatsArchived', Boolean(keepMutedChatsArchived) ); await window.storage.put( 'hasSetMyStoriesPrivacy', Boolean(hasSetMyStoriesPrivacy) ); { const hasViewedOnboardingStoryBool = Boolean(hasViewedOnboardingStory); await window.storage.put( 'hasViewedOnboardingStory', hasViewedOnboardingStoryBool ); if (hasViewedOnboardingStoryBool) { drop(findAndDeleteOnboardingStoryIfExists()); } else { drop(downloadOnboardingStory()); } } { const hasCompletedUsernameOnboardingBool = Boolean( hasCompletedUsernameOnboarding ); await window.storage.put( 'hasCompletedUsernameOnboarding', hasCompletedUsernameOnboardingBool ); } { const hasCompletedUsernameOnboardingBool = Boolean( hasSeenGroupStoryEducationSheet ); await window.storage.put( 'hasSeenGroupStoryEducationSheet', hasCompletedUsernameOnboardingBool ); } { const hasStoriesDisabled = Boolean(storiesDisabled); await window.storage.put('hasStoriesDisabled', hasStoriesDisabled); window.textsecure.server?.onHasStoriesDisabledChange(hasStoriesDisabled); } switch (storyViewReceiptsEnabled) { case Proto.OptionalBool.ENABLED: await window.storage.put('storyViewReceiptsEnabled', true); break; case Proto.OptionalBool.DISABLED: await window.storage.put('storyViewReceiptsEnabled', false); break; case Proto.OptionalBool.UNSET: default: // Do nothing break; } if (usernameLink?.entropy?.length && usernameLink?.serverId?.length) { const oldLink = window.storage.get('usernameLink'); if ( window.storage.get('usernameLinkCorrupted') && (!oldLink || !Bytes.areEqual(usernameLink.entropy, oldLink.entropy) || !Bytes.areEqual(usernameLink.serverId, oldLink.serverId)) ) { details.push('clearing username link corruption'); await window.storage.remove('usernameLinkCorrupted'); } await Promise.all([ usernameLink.color && window.storage.put('usernameLinkColor', usernameLink.color), window.storage.put('usernameLink', { entropy: usernameLink.entropy, serverId: usernameLink.serverId, }), ]); } else { await Promise.all([ window.storage.remove('usernameLinkColor'), window.storage.remove('usernameLink'), ]); } const ourID = window.ConversationController.getOurConversationId(); if (!ourID) { throw new Error('Could not find ourID'); } const conversation = await window.ConversationController.getOrCreateAndWait( ourID, 'private' ); addUnknownFields(accountRecord, conversation, details); const oldStorageID = conversation.get('storageID'); const oldStorageVersion = conversation.get('storageVersion'); if ( window.storage.get('usernameCorrupted') && username !== conversation.get('username') ) { details.push('clearing username corruption'); await window.storage.remove('usernameCorrupted'); } conversation.set({ isArchived: Boolean(noteToSelfArchived), markedUnread: Boolean(noteToSelfMarkedUnread), username: dropNull(username), storageID, storageVersion, }); let needsProfileFetch = false; if (profileKey && profileKey.length > 0) { needsProfileFetch = await conversation.setProfileKey( Bytes.toBase64(profileKey), { viaStorageServiceSync: true, reason: 'mergeAccountRecord' } ); const avatarUrl = dropNull(accountRecord.avatarUrl); await conversation.setProfileAvatar(avatarUrl, profileKey); await window.storage.put('avatarUrl', avatarUrl); } const { hasConflict, details: extraDetails } = doesRecordHavePendingChanges( toAccountRecord(conversation), accountRecord, conversation ); updatedConversations.push(conversation); details = details.concat(extraDetails); return { hasConflict, conversation, updatedConversations, needsProfileFetch, oldStorageID, oldStorageVersion, details, }; } export async function mergeStoryDistributionListRecord( storageID: string, storageVersion: number, storyDistributionListRecord: Proto.IStoryDistributionListRecord ): Promise { const redactedStorageID = redactExtendedStorageID({ storageID, storageVersion, }); if (!storyDistributionListRecord.identifier) { throw new Error( `No storyDistributionList identifier for ${redactedStorageID}` ); } const details: Array = []; const isMyStory = Bytes.areEqual( MY_STORY_BYTES, storyDistributionListRecord.identifier ); let listId: StoryDistributionIdString; if (isMyStory) { listId = MY_STORY_ID; } else { const uuid = bytesToUuid(storyDistributionListRecord.identifier); strictAssert(uuid, 'mergeStoryDistributionListRecord: no distribution id'); listId = normalizeStoryDistributionId( uuid, 'mergeStoryDistributionListRecord' ); } const localStoryDistributionList = await DataReader.getStoryDistributionWithMembers(listId); const remoteListMembers: Array = ( storyDistributionListRecord.recipientServiceIds || [] ).map(id => normalizeServiceId(id, 'mergeStoryDistributionListRecord')); if (storyDistributionListRecord.$unknownFields) { details.push('adding unknown fields'); } const deletedAtTimestamp = getTimestampFromLong( storyDistributionListRecord.deletedAtTimestamp ); const storyDistribution: StoryDistributionWithMembersType = { id: listId, name: String(storyDistributionListRecord.name), deletedAtTimestamp: isMyStory ? undefined : deletedAtTimestamp, allowsReplies: Boolean(storyDistributionListRecord.allowsReplies), isBlockList: Boolean(storyDistributionListRecord.isBlockList), members: remoteListMembers, senderKeyInfo: localStoryDistributionList?.senderKeyInfo, storageID, storageVersion, storageUnknownFields: storyDistributionListRecord.$unknownFields ? Bytes.concatenate(storyDistributionListRecord.$unknownFields) : null, storageNeedsSync: Boolean(localStoryDistributionList?.storageNeedsSync), }; if (!localStoryDistributionList) { await DataWriter.createNewStoryDistribution(storyDistribution); const shouldSave = false; window.reduxActions.storyDistributionLists.createDistributionList( storyDistribution.name, remoteListMembers, storyDistribution, shouldSave ); return { details, hasConflict: false, }; } const oldStorageID = localStoryDistributionList.storageID; const oldStorageVersion = localStoryDistributionList.storageVersion; const needsToClearUnknownFields = !storyDistributionListRecord.$unknownFields && localStoryDistributionList.storageUnknownFields; if (needsToClearUnknownFields) { details.push('clearing unknown fields'); } const isBadRemoteData = !deletedAtTimestamp && !storyDistribution.name; if (isBadRemoteData) { Object.assign(storyDistribution, { name: localStoryDistributionList.name, members: localStoryDistributionList.members, }); } const { hasConflict, details: conflictDetails } = doRecordsConflict( toStoryDistributionListRecord(storyDistribution), storyDistributionListRecord ); const localMembersListSet = new Set(localStoryDistributionList.members); const toAdd: Array = remoteListMembers.filter( serviceId => !localMembersListSet.has(serviceId) ); const remoteMemberListSet = new Set(remoteListMembers); const toRemove: Array = localStoryDistributionList.members.filter( serviceId => !remoteMemberListSet.has(serviceId) ); details.push('updated'); await DataWriter.modifyStoryDistributionWithMembers(storyDistribution, { toAdd, toRemove, }); window.reduxActions.storyDistributionLists.modifyDistributionList({ allowsReplies: Boolean(storyDistribution.allowsReplies), deletedAtTimestamp: storyDistribution.deletedAtTimestamp, id: storyDistribution.id, isBlockList: Boolean(storyDistribution.isBlockList), membersToAdd: toAdd, membersToRemove: toRemove, name: storyDistribution.name, }); return { details: [...details, ...conflictDetails], hasConflict, oldStorageID, oldStorageVersion, }; } export async function mergeStickerPackRecord( storageID: string, storageVersion: number, stickerPackRecord: Proto.IStickerPackRecord ): Promise { const redactedStorageID = redactExtendedStorageID({ storageID, storageVersion, }); if (!stickerPackRecord.packId || Bytes.isEmpty(stickerPackRecord.packId)) { throw new Error(`No stickerPackRecord identifier for ${redactedStorageID}`); } const details: Array = []; const id = Bytes.toHex(stickerPackRecord.packId); const localStickerPack = await DataReader.getStickerPackInfo(id); if (stickerPackRecord.$unknownFields) { details.push('adding unknown fields'); } const storageUnknownFields = stickerPackRecord.$unknownFields ? Bytes.concatenate(stickerPackRecord.$unknownFields) : null; let stickerPack: StickerPackInfoType; if (stickerPackRecord.deletedAtTimestamp?.toNumber()) { stickerPack = { id, uninstalledAt: stickerPackRecord.deletedAtTimestamp.toNumber(), storageID, storageVersion, storageUnknownFields, storageNeedsSync: false, }; } else { if ( !stickerPackRecord.packKey || Bytes.isEmpty(stickerPackRecord.packKey) ) { throw new Error(`No stickerPackRecord key for ${redactedStorageID}`); } stickerPack = { id, key: Bytes.toBase64(stickerPackRecord.packKey), position: 'position' in stickerPackRecord ? stickerPackRecord.position : (localStickerPack?.position ?? undefined), storageID, storageVersion, storageUnknownFields, storageNeedsSync: false, }; } const oldStorageID = localStickerPack?.storageID; const oldStorageVersion = localStickerPack?.storageVersion; const needsToClearUnknownFields = !stickerPack.storageUnknownFields && localStickerPack?.storageUnknownFields; if (needsToClearUnknownFields) { details.push('clearing unknown fields'); } const { hasConflict, details: conflictDetails } = doRecordsConflict( toStickerPackRecord(stickerPack), stickerPackRecord ); const wasUninstalled = Boolean(localStickerPack?.uninstalledAt); const isUninstalled = Boolean(stickerPack.uninstalledAt); details.push( `wasUninstalled=${wasUninstalled}`, `isUninstalled=${isUninstalled}`, `oldPosition=${localStickerPack?.position ?? '?'}`, `newPosition=${stickerPack.position ?? '?'}` ); if (localStickerPack && !wasUninstalled && isUninstalled) { assertDev(localStickerPack.key, 'Installed sticker pack has no key'); window.reduxActions.stickers.uninstallStickerPack( localStickerPack.id, localStickerPack.key, { fromStorageService: true } ); } else if ((!localStickerPack || wasUninstalled) && !isUninstalled) { assertDev(stickerPack.key, 'Sticker pack does not have key'); const status = Stickers.getStickerPackStatus(stickerPack.id); if (status === 'downloaded') { window.reduxActions.stickers.installStickerPack( stickerPack.id, stickerPack.key, { fromStorageService: true, } ); } else { void Stickers.downloadStickerPack(stickerPack.id, stickerPack.key, { finalStatus: 'installed', fromStorageService: true, }); } } await DataWriter.updateStickerPackInfo(stickerPack); return { details: [...details, ...conflictDetails], hasConflict, oldStorageID, oldStorageVersion, }; } export async function mergeCallLinkRecord( storageID: string, storageVersion: number, callLinkRecord: Proto.ICallLinkRecord ): Promise { const redactedStorageID = redactExtendedStorageID({ storageID, storageVersion, }); // callLinkRecords must have rootKey if (!callLinkRecord.rootKey) { return { hasConflict: false, shouldDrop: true, details: ['no rootKey'] }; } const details: Array = []; const rootKeyString = fromRootKeyBytes(callLinkRecord.rootKey); const adminKeyString = callLinkRecord.adminPasskey ? fromAdminKeyBytes(callLinkRecord.adminPasskey) : null; const callLinkRootKey = CallLinkRootKey.parse(rootKeyString); const roomId = getRoomIdFromRootKey(callLinkRootKey); const logId = `mergeCallLinkRecord(${redactedStorageID}, ${roomId})`; const localCallLinkDbRecord = await DataReader.getCallLinkRecordByRoomId(roomId); const deletedAt: number | null = callLinkRecord.deletedAtTimestampMs != null ? getTimestampFromLong(callLinkRecord.deletedAtTimestampMs) : null; const shouldDrop = deletedAt != null && isOlderThan(deletedAt, CALL_LINK_DELETED_STORAGE_RECORD_TTL); if (shouldDrop) { details.push('expired deleted call link; scheduling for removal'); } const callLinkDbRecord: CallLinkRecord = { roomId, rootKey: callLinkRecord.rootKey, adminKey: callLinkRecord.adminPasskey ?? null, name: localCallLinkDbRecord?.name ?? '', restrictions: localCallLinkDbRecord?.restrictions ?? 0, expiration: localCallLinkDbRecord?.expiration ?? null, revoked: localCallLinkDbRecord?.revoked === 1 ? 1 : 0, deleted: deletedAt ? 1 : 0, deletedAt, storageID, storageVersion, storageUnknownFields: callLinkRecord.$unknownFields ? Bytes.concatenate(callLinkRecord.$unknownFields) : null, storageNeedsSync: localCallLinkDbRecord?.storageNeedsSync === 1 ? 1 : 0, }; if (!localCallLinkDbRecord) { if (deletedAt) { log.info( `${logId}: Found deleted call link with no matching local record, skipping` ); } else { log.info(`${logId}: Discovered new call link, creating locally`); details.push('creating call link'); // Create CallLink and call history item const callLink = callLinkFromRecord(callLinkDbRecord); const callHistory = toCallHistoryFromUnusedCallLink(callLink); await Promise.all([ DataWriter.insertCallLink(callLink), DataWriter.saveCallHistory(callHistory), ]); drop(callLinkRefreshJobQueue.add({ roomId: callLink.roomId })); window.reduxActions.callHistory.addCallHistory(callHistory); } return { details, hasConflict: false, shouldDrop, }; } const oldStorageID = localCallLinkDbRecord.storageID || undefined; const oldStorageVersion = localCallLinkDbRecord.storageVersion || undefined; const needsToClearUnknownFields = !callLinkRecord.$unknownFields && localCallLinkDbRecord.storageUnknownFields; if (needsToClearUnknownFields) { details.push('clearing unknown fields'); } const isBadRemoteData = Boolean(deletedAt && adminKeyString); if (isBadRemoteData) { log.warn( `${logId}: Found bad remote data: deletedAtTimestampMs and adminPasskey were both present. Assuming deleted.` ); } const { hasConflict, details: conflictDetails } = doRecordsConflict( toCallLinkRecord(callLinkDbRecord), callLinkRecord ); // First update local record details.push('updated'); const callLink = callLinkFromRecord(callLinkDbRecord); await DataWriter.updateCallLink(callLink); // Deleted in storage but we have it locally: Delete locally too and update redux if (deletedAt && localCallLinkDbRecord.deleted !== 1) { // Another device deleted the link and uploaded to storage, and we learned about it log.info(`${logId}: Discovered deleted call link, deleting locally`); details.push('deleting locally'); await DataWriter.beginDeleteCallLink(roomId, { storageNeedsSync: false, deletedAt, }); // No need to delete via RingRTC as we assume the originating device did that already await DataWriter.finalizeDeleteCallLink(roomId); window.reduxActions.calling.handleCallLinkDelete({ roomId }); } else if (!deletedAt && localCallLinkDbRecord.deleted === 1) { // Not deleted in storage, but we've marked it as deleted locally. // Skip doing anything, we will update things locally after sync. log.warn(`${logId}: Found call link, but it was marked deleted locally.`); } else { window.reduxActions.calling.handleCallLinkUpdate({ rootKey: rootKeyString, adminKey: adminKeyString, }); } return { details: [...details, ...conflictDetails], hasConflict, shouldDrop, oldStorageID, oldStorageVersion, }; }