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.
session-desktop/ts/session/apis/snode_api/swarm_polling_config/SwarmPollingGroupConfig.ts

199 lines
7.7 KiB
TypeScript

import { GroupPubkeyType } from 'libsession_util_nodejs';
import { isEmpty, isFinite, isNumber } from 'lodash';
import { to_hex } from 'libsodium-wrappers-sumo';
import { Data } from '../../../../data/data';
import { messagesExpired } from '../../../../state/ducks/conversations';
import { groupInfoActions } from '../../../../state/ducks/metaGroups';
import {
MetaGroupWrapperActions,
UserGroupsWrapperActions,
} from '../../../../webworker/workers/browser/libsession_worker_interface';
import { ed25519Str, fromBase64ToArray } from '../../../utils/String';
import { GroupPendingRemovals } from '../../../utils/job_runners/jobs/GroupPendingRemovalsJob';
import { LibSessionUtil } from '../../../utils/libsession/libsession_utils';
import { SnodeNamespaces } from '../namespaces';
import { RetrieveMessageItemWithNamespace } from '../types';
import { ConvoHub } from '../../../conversations';
/**
* This is a basic optimization to avoid running the logic when the `deleteBeforeSeconds`
* and the `deleteAttachBeforeSeconds` does not change between each polls.
* Essentially, when the `deleteBeforeSeconds` is set in the group info config,
* - on start that map will be empty so we will run the logic to delete any messages sent before that.
* - after each poll, we will only rerun the logic if the new `deleteBeforeSeconds` is higher than the current setting.
*
*/
const lastAppliedRemoveMsgSentBeforeSeconds = new Map<GroupPubkeyType, number>();
const lastAppliedRemoveAttachmentSentBeforeSeconds = new Map<GroupPubkeyType, number>();
async function handleMetaMergeResults(groupPk: GroupPubkeyType) {
const infos = await MetaGroupWrapperActions.infoGet(groupPk);
if (window.sessionFeatureFlags.debug.debugLibsessionDumps) {
const dumps = await MetaGroupWrapperActions.metaMakeDump(groupPk);
window.log.info(
`pushChangesToGroupSwarmIfNeeded: current metadump: ${ed25519Str(groupPk)}:`,
to_hex(dumps)
);
}
if (infos) {
if (infos.isDestroyed) {
window.log.info(`${ed25519Str(groupPk)} is marked as destroyed after merge. Removing it.`);
await ConvoHub.use().deleteGroup(groupPk, {
sendLeaveMessage: false,
fromSyncMessage: false,
emptyGroupButKeepAsKicked: true, // we just got something from the group's swarm, so it is not pendingInvite
deleteAllMessagesOnSwarm: false,
forceDestroyForAllMembers: false,
});
} else {
if (
isNumber(infos.deleteBeforeSeconds) &&
isFinite(infos.deleteBeforeSeconds) &&
infos.deleteBeforeSeconds > 0 &&
(lastAppliedRemoveMsgSentBeforeSeconds.get(groupPk) || Number.MAX_SAFE_INTEGER) >
infos.deleteBeforeSeconds
) {
// delete any messages in this conversation sent before that timestamp (in seconds)
const deletedMsgIds = await Data.removeAllMessagesInConversationSentBefore({
deleteBeforeSeconds: infos.deleteBeforeSeconds,
conversationId: groupPk,
});
window.log.info(
`removeAllMessagesInConversationSentBefore of ${ed25519Str(groupPk)} before ${infos.deleteBeforeSeconds}: `,
deletedMsgIds
);
window.inboxStore.dispatch(
messagesExpired(deletedMsgIds.map(messageId => ({ conversationKey: groupPk, messageId })))
);
lastAppliedRemoveMsgSentBeforeSeconds.set(groupPk, infos.deleteBeforeSeconds);
}
if (
isNumber(infos.deleteAttachBeforeSeconds) &&
isFinite(infos.deleteAttachBeforeSeconds) &&
infos.deleteAttachBeforeSeconds > 0 &&
(lastAppliedRemoveAttachmentSentBeforeSeconds.get(groupPk) || Number.MAX_SAFE_INTEGER) >
infos.deleteAttachBeforeSeconds
) {
// delete any attachments in this conversation sent before that timestamp (in seconds)
const impactedMsgModels = await Data.getAllMessagesWithAttachmentsInConversationSentBefore({
deleteAttachBeforeSeconds: infos.deleteAttachBeforeSeconds,
conversationId: groupPk,
});
window.log.info(
`getAllMessagesWithAttachmentsInConversationSentBefore of ${ed25519Str(groupPk)} before ${infos.deleteAttachBeforeSeconds}: impactedMsgModelsIds `,
impactedMsgModels.map(m => m.id)
);
for (let index = 0; index < impactedMsgModels.length; index++) {
const msg = impactedMsgModels[index];
// eslint-disable-next-line no-await-in-loop
await msg?.cleanup();
}
lastAppliedRemoveAttachmentSentBeforeSeconds.set(groupPk, infos.deleteAttachBeforeSeconds);
}
}
const membersWithPendingRemovals =
await MetaGroupWrapperActions.memberGetAllPendingRemovals(groupPk);
if (membersWithPendingRemovals.length) {
const group = await UserGroupsWrapperActions.getGroup(groupPk);
if (group && group.secretKey && !isEmpty(group.secretKey)) {
await GroupPendingRemovals.addJob({ groupPk });
}
}
}
const convo = ConvoHub.use().get(groupPk);
const refreshedInfos = await MetaGroupWrapperActions.infoGet(groupPk);
if (convo) {
let changes = false;
if (refreshedInfos.name !== convo.get('displayNameInProfile')) {
convo.set({ displayNameInProfile: refreshedInfos.name || undefined });
changes = true;
}
const expectedMode = refreshedInfos.expirySeconds ? 'deleteAfterSend' : 'off';
if (
refreshedInfos.expirySeconds !== convo.get('expireTimer') ||
expectedMode !== convo.get('expirationMode')
) {
convo.set({
expireTimer: refreshedInfos.expirySeconds || undefined,
expirationMode: expectedMode,
});
changes = true;
}
if (changes) {
await convo.commit();
}
}
}
async function handleGroupSharedConfigMessages(
groupConfigMessages: Array<RetrieveMessageItemWithNamespace>,
groupPk: GroupPubkeyType
) {
try {
window.log.info(
`received groupConfigMessages count: ${groupConfigMessages.length} for groupPk:${ed25519Str(
groupPk
)}`
);
if (groupConfigMessages.find(m => !m.storedAt)) {
throw new Error('all incoming group config message should have a timestamp');
}
const infos = groupConfigMessages
.filter(m => m.namespace === SnodeNamespaces.ClosedGroupInfo)
.map(info => {
return { data: fromBase64ToArray(info.data), hash: info.hash };
});
const members = groupConfigMessages
.filter(m => m.namespace === SnodeNamespaces.ClosedGroupMembers)
.map(info => {
return { data: fromBase64ToArray(info.data), hash: info.hash };
});
const keys = groupConfigMessages
.filter(m => m.namespace === SnodeNamespaces.ClosedGroupKeys)
.map(info => {
return {
data: fromBase64ToArray(info.data),
hash: info.hash,
timestampMs: info.storedAt,
};
});
const toMerge = {
groupInfo: infos,
groupKeys: keys,
groupMember: members,
};
window.log.info(
`received keys:${toMerge.groupKeys.length}, infos:${toMerge.groupInfo.length}, members:${
toMerge.groupMember.length
} for groupPk:${ed25519Str(groupPk)}`
);
// do the merge with our current state
await MetaGroupWrapperActions.metaMerge(groupPk, toMerge);
await handleMetaMergeResults(groupPk);
// save updated dumps to the DB right away
await LibSessionUtil.saveDumpsToDb(groupPk);
// refresh the redux slice with the merged result
window.inboxStore.dispatch(
groupInfoActions.refreshGroupDetailsFromWrapper({
groupPk,
})
);
} catch (e) {
window.log.warn(
`handleGroupSharedConfigMessages of ${groupConfigMessages.length} failed with ${e.message}`
);
// not rethrowing
}
}
export const SwarmPollingGroupConfig = { handleGroupSharedConfigMessages };