import { from_hex } from 'libsodium-wrappers-sumo'; import { isArray, isEmpty, isEqual } from 'lodash'; import { ContactInfoSet, LegacyGroupInfo, LegacyGroupMemberInfo } from 'session_util_wrapper'; import { OpenGroupV2Room } from '../data/opengroups'; import { ConversationAttributes } from '../models/conversationAttributes'; import { OpenGroupRequestCommonType } from '../session/apis/open_group_api/opengroupV2/ApiUtil'; import { fromHexToArray } from '../session/utils/String'; import { ConfigWrapperObjectTypes } from '../webworker/workers/browser/libsession_worker_functions'; /** * This wrapper can be used to make a function type not async, asynced. * We use it in the typing of the database communication, because the data calls (renderer side) have essentially the same signature of the sql calls (node side), with an added `await` */ export type AsyncWrapper any> = ( ...args: Parameters ) => Promise>; /** * This type is used to build from an objectType filled with functions, a new object type where all the functions their async equivalent */ export type AsyncObjectWrapper any>> = { [Property in keyof Type]: AsyncWrapper; }; export type MsgDuplicateSearchOpenGroup = Array<{ sender: string; serverTimestamp: number; // senderBlinded?: string; // for a message we sent, we need a blinded id and an unblinded one }>; export type UpdateLastHashType = { convoId: string; snode: string; hash: string; expiresAt: number; namespace: number; }; export type ConfigDumpRow = { variant: ConfigWrapperObjectTypes; // the variant this entry is about. (user pr, contacts, ...) publicKey: string; // either our pubkey if a dump for our own swarm or the closed group pubkey data: Uint8Array; // the blob returned by libsession.dump() call }; export type ConfigDumpRowWithoutData = Pick; export const CONFIG_DUMP_TABLE = 'configDump'; // ========== configdump export type ConfigDumpDataNode = { getByVariantAndPubkey: ( variant: ConfigWrapperObjectTypes, publicKey: string ) => Array; saveConfigDump: (dump: ConfigDumpRow) => void; getAllDumpsWithData: () => Array; getAllDumpsWithoutData: () => Array; }; // ========== unprocessed export type UnprocessedParameter = { id: string; version: number; envelope: string; timestamp: number; // serverTimestamp: number; attempts: number; messageHash: string; senderIdentity?: string; decrypted?: string; // added once the envelopes's content is decrypted with updateCacheWithDecryptedContent source?: string; // added once the envelopes's content is decrypted with updateCacheWithDecryptedContent }; export type UnprocessedDataNode = { saveUnprocessed: (data: UnprocessedParameter) => void; updateUnprocessedAttempts: (id: string, attempts: number) => void; updateUnprocessedWithData: (id: string, data: UnprocessedParameter) => void; getUnprocessedById: (id: string) => UnprocessedParameter | undefined; getUnprocessedCount: () => number; getAllUnprocessed: () => Array; removeUnprocessed: (id: string) => void; removeAllUnprocessed: () => void; }; // ======== attachment downloads export type AttachmentDownloadMessageDetails = { messageId: string; type: 'preview' | 'quote' | 'attachment'; index: number; isOpenGroupV2: boolean; openGroupV2Details: OpenGroupRequestCommonType | undefined; }; export type SaveConversationReturn = { unreadCount: number; mentionedUs: boolean; lastReadTimestampMessage: number | null; } | null; /** * This function returns a contactInfo for the wrapper to understand from the DB values. * Created in this file so we can reuse it during the migration (node side), and from the renderer side */ export function getContactInfoFromDBValues({ id, dbApproved, dbApprovedMe, dbBlocked, dbName, dbNickname, priority, dbProfileUrl, dbProfileKey, expirationTimerSeconds, }: { id: string; dbApproved: boolean; dbApprovedMe: boolean; dbBlocked: boolean; dbNickname: string | undefined; dbName: string | undefined; priority: number; dbProfileUrl: string | undefined; dbProfileKey: string | undefined; expirationTimerSeconds: number | undefined; }): ContactInfoSet { const wrapperContact: ContactInfoSet = { id, approved: !!dbApproved, approvedMe: !!dbApprovedMe, blocked: !!dbBlocked, priority, nickname: dbNickname, name: dbName, expirationTimerSeconds: !!expirationTimerSeconds && isFinite(expirationTimerSeconds) && expirationTimerSeconds > 0 ? expirationTimerSeconds : 0, // TODOLATER add the expiration mode handling expirationMode: !!expirationTimerSeconds && isFinite(expirationTimerSeconds) && expirationTimerSeconds > 0 ? 'disappearAfterSend' : 'off', }; if ( wrapperContact.profilePicture?.url !== dbProfileUrl || !isEqual(wrapperContact.profilePicture?.key, dbProfileKey) ) { wrapperContact.profilePicture = { url: dbProfileUrl || null, key: dbProfileKey && !isEmpty(dbProfileKey) ? fromHexToArray(dbProfileKey) : null, }; } return wrapperContact; } /** * This function returns a CommunityInfo for the wrapper to understand from the DB values. * It is created in this file so we can reuse it during the migration (node side), and from the renderer side */ export function getCommunityInfoFromDBValues({ priority, fullUrl, }: { priority: number; fullUrl: string; }) { const community = { fullUrl, priority, }; return community; } function maybeArrayJSONtoArray(arr: string | Array): Array { try { if (isArray(arr)) { return arr; } const parsed = JSON.parse(arr); if (isArray(parsed)) { return parsed; } return []; } catch (e) { return []; } } export function getLegacyGroupInfoFromDBValues({ id, priority, members: maybeMembers, displayNameInProfile, expireTimer, encPubkeyHex, encSeckeyHex, groupAdmins: maybeAdmins, lastJoinedTimestamp, }: Pick< ConversationAttributes, 'id' | 'priority' | 'displayNameInProfile' | 'expireTimer' | 'lastJoinedTimestamp' > & { encPubkeyHex: string; encSeckeyHex: string; members: string | Array; groupAdmins: string | Array; }) { const admins: Array = maybeArrayJSONtoArray(maybeAdmins); const members: Array = maybeArrayJSONtoArray(maybeMembers); const wrappedMembers: Array = (members || []).map(m => { return { isAdmin: admins.includes(m), pubkeyHex: m, }; }); const legacyGroup: LegacyGroupInfo = { pubkeyHex: id, disappearingTimerSeconds: !expireTimer ? 0 : expireTimer, name: displayNameInProfile || '', priority: priority || 0, members: wrappedMembers, encPubkey: !isEmpty(encPubkeyHex) ? from_hex(encPubkeyHex) : new Uint8Array(), encSeckey: !isEmpty(encSeckeyHex) ? from_hex(encSeckeyHex) : new Uint8Array(), joinedAtSeconds: Math.floor(lastJoinedTimestamp / 1000), }; return legacyGroup; } /** * This function can be used to make sure all the possible values as input of a switch as taken care off, without having a default case. * */ export function assertUnreachable(_x: never, message: string): never { const msg = `assertUnreachable: Didn't expect to get here with "${message}"`; console.info(msg); throw new Error(msg); } export function roomHasBlindEnabled(openGroup?: OpenGroupV2Room) { return capabilitiesListHasBlindEnabled(openGroup?.capabilities); } export function capabilitiesListHasBlindEnabled(caps?: Array | null) { return Boolean(caps?.includes('blind')); } export function roomHasReactionsEnabled(openGroup?: OpenGroupV2Room) { return Boolean(openGroup?.capabilities?.includes('reactions')); }