You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
285 lines
10 KiB
TypeScript
285 lines
10 KiB
TypeScript
import { BaseConvoInfoVolatile, ConvoVolatileType } from 'libsession_util_nodejs';
|
|
import { isEmpty } from 'lodash';
|
|
import { Data } from '../../../data/data';
|
|
import { OpenGroupData } from '../../../data/opengroups';
|
|
import { ConversationModel } from '../../../models/conversation';
|
|
import { assertUnreachable } from '../../../types/sqlSharedTypes';
|
|
import {
|
|
ConvoInfoVolatileWrapperActions,
|
|
UserGroupsWrapperActions,
|
|
} from '../../../webworker/workers/browser/libsession_worker_interface';
|
|
import { OpenGroupUtils } from '../../apis/open_group_api/utils';
|
|
import { getConversationController } from '../../conversations';
|
|
import { SessionUtilContact } from './libsession_utils_contacts';
|
|
import { SessionUtilUserGroups } from './libsession_utils_user_groups';
|
|
import { SessionUtilUserProfile } from './libsession_utils_user_profile';
|
|
|
|
/**
|
|
* The key of this map is the convoId as stored in the database.
|
|
*/
|
|
const mapped1o1WrapperValues = new Map<string, BaseConvoInfoVolatile>();
|
|
|
|
/**
|
|
* The key of this map is the convoId as stored in the database. So the legacy group 05 sessionID
|
|
*/
|
|
const mappedLegacyGroupWrapperValues = new Map<string, BaseConvoInfoVolatile>();
|
|
|
|
/**
|
|
* The key of this map is the convoId as stored in the database, so withoutpubkey
|
|
*/
|
|
const mappedCommunityWrapperValues = new Map<string, BaseConvoInfoVolatile>();
|
|
|
|
/**
|
|
* Returns true if that conversation should be stored in the conversation volatile info wrapper.
|
|
* It actually relies on the two other wrappers to know what to store:
|
|
* - Usergroups to know which communities and legacy group to store
|
|
* - Contacts to know which contacts to store
|
|
* - UserProfile to keep track of the `unread` state of the Note To Self conversation
|
|
*/
|
|
function isConvoToStoreInWrapper(convo: ConversationModel): boolean {
|
|
return (
|
|
SessionUtilUserGroups.isUserGroupToStoreInWrapper(convo) || // this checks for community & legacy group
|
|
SessionUtilContact.isContactToStoreInWrapper(convo) || // this checks for contacts
|
|
SessionUtilUserProfile.isUserProfileToStoreInWrapper(convo.id) // this checks for our own pubkey, as we want to keep track of the read state for the Note To Self
|
|
);
|
|
}
|
|
|
|
function getConvoType(convo: ConversationModel): ConvoVolatileType {
|
|
const convoType: ConvoVolatileType =
|
|
SessionUtilContact.isContactToStoreInWrapper(convo) ||
|
|
SessionUtilUserProfile.isUserProfileToStoreInWrapper(convo.id)
|
|
? '1o1'
|
|
: SessionUtilUserGroups.isCommunityToStoreInWrapper(convo)
|
|
? 'Community'
|
|
: 'LegacyGroup';
|
|
|
|
return convoType;
|
|
}
|
|
|
|
/**
|
|
* Updates the required field in the wrapper from the data from the `ConversationController`
|
|
* If that community does not exist in the wrapper, it is created before being updated.
|
|
* Same applies for a legacy group.
|
|
*/
|
|
async function insertConvoFromDBIntoWrapperAndRefresh(convoId: string): Promise<void> {
|
|
// this is too slow to fetch from the database the up to date data here. Let's hope that what we have in memory is up to date enough
|
|
const foundConvo = getConversationController().get(convoId);
|
|
if (!foundConvo || !isConvoToStoreInWrapper(foundConvo)) {
|
|
return;
|
|
}
|
|
const isForcedUnread = foundConvo.isMarkedUnread();
|
|
const timestampFromDbMs = (await Data.fetchConvoMemoryDetails(convoId))?.lastReadTimestampMessage;
|
|
|
|
// Note: not having a last read timestamp fallsback to 0, which keeps the existing value in the wrapper if it is already set (as done in src/convo_info_volatile_config.cpp)
|
|
// we actually do the max() of whatever is inside the wrapper and the value from the DB
|
|
const lastReadMessageTimestamp =
|
|
!!timestampFromDbMs && isFinite(timestampFromDbMs) && timestampFromDbMs > 0
|
|
? timestampFromDbMs
|
|
: 0;
|
|
|
|
window.log.debug(
|
|
`inserting into convoVolatile wrapper: ${convoId} lastMessageReadTimestamp:${lastReadMessageTimestamp} forcedUnread:${isForcedUnread}...`
|
|
);
|
|
|
|
const convoType = getConvoType(foundConvo);
|
|
switch (convoType) {
|
|
case '1o1':
|
|
try {
|
|
// this saves the details for contacts and `Note To Self`
|
|
await ConvoInfoVolatileWrapperActions.set1o1(
|
|
convoId,
|
|
lastReadMessageTimestamp,
|
|
isForcedUnread
|
|
);
|
|
await refreshConvoVolatileCached(convoId, false, false);
|
|
} catch (e) {
|
|
window.log.warn(
|
|
`ConvoInfoVolatileWrapperActions.set1o1 of ${convoId} failed with ${e.message}`
|
|
);
|
|
}
|
|
break;
|
|
case 'LegacyGroup':
|
|
try {
|
|
await ConvoInfoVolatileWrapperActions.setLegacyGroup(
|
|
convoId,
|
|
lastReadMessageTimestamp,
|
|
isForcedUnread
|
|
);
|
|
await refreshConvoVolatileCached(convoId, true, false);
|
|
} catch (e) {
|
|
window.log.warn(
|
|
`ConvoInfoVolatileWrapperActions.setLegacyGroup of ${convoId} failed with ${e.message}`
|
|
);
|
|
}
|
|
break;
|
|
case 'Community':
|
|
try {
|
|
const asOpengroup = foundConvo.toOpenGroupV2();
|
|
const roomDetails = OpenGroupData.getV2OpenGroupRoomByRoomId(asOpengroup);
|
|
if (!roomDetails || isEmpty(roomDetails.serverPublicKey)) {
|
|
return;
|
|
}
|
|
|
|
// we need to build the full URL with the pubkey so we can add it to the wrapper. Let's reuse the exposed method from the wrapper for that
|
|
const fullUrlWithPubkey = await UserGroupsWrapperActions.buildFullUrlFromDetails(
|
|
roomDetails.serverUrl,
|
|
roomDetails.roomId,
|
|
roomDetails.serverPublicKey
|
|
);
|
|
|
|
// this does the create or the update of the matching existing community
|
|
await ConvoInfoVolatileWrapperActions.setCommunityByFullUrl(
|
|
fullUrlWithPubkey,
|
|
lastReadMessageTimestamp,
|
|
isForcedUnread
|
|
);
|
|
|
|
await refreshConvoVolatileCached(convoId, false, false);
|
|
} catch (e) {
|
|
window.log.warn(
|
|
`ConvoInfoVolatileWrapperActions.setCommunityByFullUrl of ${convoId} failed with ${e.message}`
|
|
);
|
|
}
|
|
break;
|
|
default:
|
|
assertUnreachable(
|
|
convoType,
|
|
`insertConvoFromDBIntoWrapperAndRefresh unhandled case "${convoType}"`
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param isLegacyGroup we need this to know if the corresponding 05 starting pubkey is associated with a legacy group or not
|
|
* @param duringAppStart set this to true if we should just fetch the cached value but not trigger a UI refresh of the corresponding conversation
|
|
*/
|
|
async function refreshConvoVolatileCached(
|
|
convoId: string,
|
|
isLegacyGroup: boolean,
|
|
duringAppStart: boolean
|
|
) {
|
|
try {
|
|
let convoType: ConvoVolatileType = '1o1';
|
|
let refreshed = false;
|
|
|
|
if (OpenGroupUtils.isOpenGroupV2(convoId)) {
|
|
convoType = 'Community';
|
|
} else if (convoId.startsWith('05') && isLegacyGroup) {
|
|
convoType = 'LegacyGroup';
|
|
} else if (convoId.startsWith('05')) {
|
|
convoType = '1o1';
|
|
}
|
|
|
|
switch (convoType) {
|
|
case '1o1':
|
|
const fromWrapper1o1 = await ConvoInfoVolatileWrapperActions.get1o1(convoId);
|
|
if (fromWrapper1o1) {
|
|
mapped1o1WrapperValues.set(convoId, fromWrapper1o1);
|
|
}
|
|
refreshed = true;
|
|
break;
|
|
case 'LegacyGroup':
|
|
const fromWrapperLegacyGroup = await ConvoInfoVolatileWrapperActions.getLegacyGroup(
|
|
convoId
|
|
);
|
|
if (fromWrapperLegacyGroup) {
|
|
mappedLegacyGroupWrapperValues.set(convoId, fromWrapperLegacyGroup);
|
|
}
|
|
refreshed = true;
|
|
break;
|
|
case 'Community':
|
|
const fromWrapperCommunity = await ConvoInfoVolatileWrapperActions.getCommunity(convoId);
|
|
if (fromWrapperCommunity && fromWrapperCommunity.fullUrlWithPubkey) {
|
|
mappedCommunityWrapperValues.set(convoId, fromWrapperCommunity);
|
|
}
|
|
refreshed = true;
|
|
break;
|
|
|
|
default:
|
|
assertUnreachable(convoType, `refreshConvoVolatileCached unhandled case "${convoType}"`);
|
|
}
|
|
|
|
if (refreshed && !duringAppStart) {
|
|
getConversationController()
|
|
.get(convoId)
|
|
?.triggerUIRefresh();
|
|
}
|
|
} catch (e) {
|
|
window.log.info(`refreshMappedValue for volatile convoID: ${convoId}`, e.message);
|
|
}
|
|
}
|
|
|
|
function getVolatileInfoCached(convoId: string): BaseConvoInfoVolatile | undefined {
|
|
return (
|
|
mapped1o1WrapperValues.get(convoId) ||
|
|
mappedLegacyGroupWrapperValues.get(convoId) ||
|
|
mappedCommunityWrapperValues.get(convoId)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Removes the matching community from the wrapper and from the cached list of communities
|
|
*/
|
|
async function removeCommunityFromWrapper(convoId: string, fullUrlWithPubkey: string) {
|
|
try {
|
|
await ConvoInfoVolatileWrapperActions.eraseCommunityByFullUrl(fullUrlWithPubkey);
|
|
} catch (e) {
|
|
window.log.warn('removeCommunityFromWrapper failed with ', e.message);
|
|
}
|
|
|
|
mappedCommunityWrapperValues.delete(convoId);
|
|
}
|
|
|
|
/**
|
|
* Removes the matching legacy group from the wrapper and from the cached list of legacy groups
|
|
*/
|
|
async function removeLegacyGroupFromWrapper(convoId: string) {
|
|
try {
|
|
await ConvoInfoVolatileWrapperActions.eraseLegacyGroup(convoId);
|
|
} catch (e) {
|
|
window.log.warn('removeLegacyGroupFromWrapper failed with ', e.message);
|
|
}
|
|
mappedLegacyGroupWrapperValues.delete(convoId);
|
|
}
|
|
|
|
/**
|
|
* Removes the matching legacy group from the wrapper and from the cached list of legacy groups
|
|
*/
|
|
async function removeContactFromWrapper(convoId: string) {
|
|
try {
|
|
await ConvoInfoVolatileWrapperActions.erase1o1(convoId);
|
|
} catch (e) {
|
|
window.log.warn('removeContactFromWrapper failed with ', e.message);
|
|
}
|
|
mapped1o1WrapperValues.delete(convoId);
|
|
}
|
|
|
|
/**
|
|
* This function can be used where there are things to do for all the types handled by this wrapper.
|
|
* You can do a loop on all the types handled by this wrapper and have a switch using assertUnreachable to get errors when not every case is handled.
|
|
*
|
|
*
|
|
* Note: Ideally, we'd like to have this type in the wrapper index.d.ts, but it would require it to be a index.ts instead, which causes a whole other bunch of issues because it is a native node module.
|
|
*/
|
|
function getConvoInfoVolatileTypes(): Array<ConvoVolatileType> {
|
|
return ['1o1', 'LegacyGroup', 'Community'];
|
|
}
|
|
|
|
export const SessionUtilConvoInfoVolatile = {
|
|
// shared
|
|
isConvoToStoreInWrapper,
|
|
insertConvoFromDBIntoWrapperAndRefresh,
|
|
refreshConvoVolatileCached,
|
|
getConvoInfoVolatileTypes,
|
|
getVolatileInfoCached,
|
|
|
|
// 1o1
|
|
removeContactFromWrapper,
|
|
|
|
// legacy group
|
|
removeLegacyGroupFromWrapper, // a group can be removed but also just marked hidden, so only call this function when the group is completely removed // TODOLATER
|
|
|
|
// communities
|
|
removeCommunityFromWrapper,
|
|
};
|