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.
		
		
		
		
		
			
		
			
				
	
	
		
			257 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			TypeScript
		
	
			
		
		
	
	
			257 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			TypeScript
		
	
| /* eslint-disable no-case-declarations */
 | |
| import { CommunityInfo, LegacyGroupInfo, UserGroupsType } from 'libsession_util_nodejs';
 | |
| import { Data } from '../../../data/data';
 | |
| import { OpenGroupData } from '../../../data/opengroups';
 | |
| import { ConversationModel } from '../../../models/conversation';
 | |
| import {
 | |
|   CommunityInfoFromDBValues,
 | |
|   assertUnreachable,
 | |
|   getCommunityInfoFromDBValues,
 | |
|   getLegacyGroupInfoFromDBValues,
 | |
| } from '../../../types/sqlSharedTypes';
 | |
| import { UserGroupsWrapperActions } from '../../../webworker/workers/browser/libsession_worker_interface';
 | |
| import { ConvoHub } from '../../conversations';
 | |
| import { PubKey } from '../../types';
 | |
| import { CONVERSATION_PRIORITIES } from '../../../models/conversationAttributes';
 | |
| 
 | |
| /**
 | |
|  * Returns true if that conversation is an active group
 | |
|  */
 | |
| function isUserGroupToStoreInWrapper(convo: ConversationModel): boolean {
 | |
|   return (
 | |
|     isCommunityToStoreInWrapper(convo) ||
 | |
|     isLegacyGroupToStoreInWrapper(convo) ||
 | |
|     isGroupToStoreInWrapper(convo)
 | |
|   );
 | |
| }
 | |
| 
 | |
| function isCommunityToStoreInWrapper(convo: ConversationModel): boolean {
 | |
|   return convo.isGroup() && convo.isPublic() && convo.isActive();
 | |
| }
 | |
| 
 | |
| function isLegacyGroupToStoreInWrapper(convo: ConversationModel): boolean {
 | |
|   return (
 | |
|     convo.isGroup() &&
 | |
|     PubKey.is05Pubkey(convo.id) && // we only check legacy group here
 | |
|     convo.isActive() &&
 | |
|     !convo.isKickedFromGroup() // we cannot have a left group anymore. We remove it when we leave it.
 | |
|   );
 | |
| }
 | |
| 
 | |
| function isGroupToStoreInWrapper(convo: ConversationModel): boolean {
 | |
|   return convo.isGroup() && PubKey.is03Pubkey(convo.id) && convo.isActive();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * We do not want to include groups left in the wrapper, but when receiving a list
 | |
|  * of wrappers from the network we need to check against the one present locally
 | |
|  * but already left, to know we need to remove them.
 | |
|  *
 | |
|  * This is to take care of this case:
 | |
|  * - deviceA creates group
 | |
|  * - deviceB joins group
 | |
|  * - deviceA leaves the group
 | |
|  * - deviceB leaves the group
 | |
|  * - deviceA removes the group entirely from the wrapper
 | |
|  * - deviceB receives the wrapper update and needs to remove the group from the DB
 | |
|  *
 | |
|  * But, as the group was already left, it would not be accounted for by `isLegacyGroupToStoreInWrapper`
 | |
|  *
 | |
|  */
 | |
| function isLegacyGroupToRemoveFromDBIfNotInWrapper(convo: ConversationModel): boolean {
 | |
|   // this filter is based on `isLegacyGroupToStoreInWrapper`
 | |
|   return (
 | |
|     convo.isGroup() && !convo.isPublic() && convo.id.startsWith('05') // new closed groups won't start with 05
 | |
|   );
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Fetches the specified convo and updates the required field in the wrapper.
 | |
|  * If that community does not exist in the wrapper, it is created before being updated.
 | |
|  * Same applies for a legacy group.
 | |
|  */
 | |
| async function insertGroupsFromDBIntoWrapperAndRefresh(
 | |
|   convoId: string
 | |
| ): Promise<CommunityInfoFromDBValues | LegacyGroupInfo | null> {
 | |
|   const foundConvo = ConvoHub.use().get(convoId);
 | |
|   if (!foundConvo) {
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   if (!SessionUtilUserGroups.isUserGroupToStoreInWrapper(foundConvo)) {
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   const convoType: UserGroupsType = SessionUtilUserGroups.isCommunityToStoreInWrapper(foundConvo)
 | |
|     ? 'Community'
 | |
|     : PubKey.is03Pubkey(convoId)
 | |
|       ? 'Group'
 | |
|       : 'LegacyGroup';
 | |
| 
 | |
|   switch (convoType) {
 | |
|     case 'Community':
 | |
|       const asOpengroup = foundConvo.toOpenGroupV2();
 | |
| 
 | |
|       const roomDetails = OpenGroupData.getV2OpenGroupRoomByRoomId(asOpengroup);
 | |
|       if (!roomDetails) {
 | |
|         return null;
 | |
|       }
 | |
| 
 | |
|       // 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 fullUrl = await UserGroupsWrapperActions.buildFullUrlFromDetails(
 | |
|         roomDetails.serverUrl,
 | |
|         roomDetails.roomId,
 | |
|         roomDetails.serverPublicKey
 | |
|       );
 | |
| 
 | |
|       const wrapperComm = getCommunityInfoFromDBValues({
 | |
|         priority: foundConvo.get('priority') || CONVERSATION_PRIORITIES.default, // this has to be a direct call to .get
 | |
|         fullUrl,
 | |
|       });
 | |
| 
 | |
|       try {
 | |
|         window.log.debug(`inserting into usergroup wrapper "${JSON.stringify(wrapperComm)}"...`);
 | |
|         // this does the create or the update of the matching existing community
 | |
|         await UserGroupsWrapperActions.setCommunityByFullUrl(
 | |
|           wrapperComm.fullUrl,
 | |
|           wrapperComm.priority
 | |
|         );
 | |
| 
 | |
|         // returned for testing purposes only
 | |
|         return {
 | |
|           fullUrl: wrapperComm.fullUrl,
 | |
|           priority: wrapperComm.priority,
 | |
|         };
 | |
|       } catch (e) {
 | |
|         window.log.warn(`UserGroupsWrapperActions.set of ${convoId} failed with ${e.message}`);
 | |
|         // we still let this go through
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case 'LegacyGroup':
 | |
|       const encryptionKeyPair = await Data.getLatestClosedGroupEncryptionKeyPair(convoId);
 | |
|       // Note: For any fields stored in both the DB and libsession,
 | |
|       // we have to make direct calls to.get() and NOT the wrapped getPriority(), etc...
 | |
|       const wrapperLegacyGroup = getLegacyGroupInfoFromDBValues({
 | |
|         id: foundConvo.id,
 | |
|         priority: foundConvo.get('priority') || CONVERSATION_PRIORITIES.default,
 | |
|         members: foundConvo.get('members') || [],
 | |
|         groupAdmins: foundConvo.getGroupAdmins(), // cannot be changed for legacy groups, so we don't care
 | |
|         expirationMode: foundConvo.get('expirationMode') || 'off',
 | |
|         expireTimer: foundConvo.get('expireTimer') || 0,
 | |
|         displayNameInProfile: foundConvo.getRealSessionUsername(),
 | |
|         encPubkeyHex: encryptionKeyPair?.publicHex || '',
 | |
|         encSeckeyHex: encryptionKeyPair?.privateHex || '',
 | |
|         lastJoinedTimestamp: foundConvo.getLastJoinedTimestamp(),
 | |
|       });
 | |
| 
 | |
|       try {
 | |
|         window.log.debug(
 | |
|           `inserting into usergroup wrapper "${foundConvo.id}"... }`,
 | |
|           JSON.stringify(wrapperLegacyGroup)
 | |
|         );
 | |
|         // this does the create or the update of the matching existing legacy group
 | |
|         await UserGroupsWrapperActions.setLegacyGroup(wrapperLegacyGroup);
 | |
|         // returned for testing purposes only
 | |
|         return wrapperLegacyGroup;
 | |
|       } catch (e) {
 | |
|         window.log.warn(`UserGroupsWrapperActions.set of ${convoId} failed with ${e.message}`);
 | |
|         // we still let this go through
 | |
|       }
 | |
|       break;
 | |
|     case 'Group':
 | |
|       // The 03-group is a bit different that the others as most fields are not to be updated.
 | |
|       // Indeed, they are more up to date on the group's swarm than ours and we don't want to keep both in sync.
 | |
|       if (!PubKey.is03Pubkey(convoId)) {
 | |
|         throw new Error('not a 03 group');
 | |
|       }
 | |
|       const groupInfo = {
 | |
|         pubkeyHex: convoId,
 | |
|         authData: null, // only updated when we process a new invite
 | |
|         invitePending: null, // only updated when we accept an invite
 | |
|         disappearingTimerSeconds: null, // not updated except when we process an invite/create a group
 | |
|         joinedAtSeconds: null, // no need to update this one except when we process an invite, maybe
 | |
|         name: null, // not updated except when we process an invite/create a group
 | |
|         secretKey: null, // not updated except when we process an promote/create a group
 | |
|         kicked: foundConvo.isKickedFromGroup() ?? null,
 | |
|         priority: foundConvo.getPriority() ?? null, // for 03 group, the priority is only tracked with libsession, so this is fine
 | |
|       };
 | |
|       try {
 | |
|         window.log.debug(
 | |
|           `inserting into usergroup wrapper "${foundConvo.id}"... }`,
 | |
|           JSON.stringify(groupInfo)
 | |
|         );
 | |
|         // this does the create or the update of the matching existing group
 | |
|         await UserGroupsWrapperActions.setGroup(groupInfo);
 | |
| 
 | |
|         // returned for testing purposes only
 | |
|         return null;
 | |
|       } catch (e) {
 | |
|         window.log.warn(`UserGroupsWrapperActions.set of ${convoId} failed with ${e.message}`);
 | |
|         // we still let this go through
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       assertUnreachable(
 | |
|         convoType,
 | |
|         `insertGroupsFromDBIntoWrapperAndRefresh case not handeld "${convoType}"`
 | |
|       );
 | |
|   }
 | |
|   return null;
 | |
| }
 | |
| 
 | |
| async function getCommunityByConvoIdNotCached(convoId: string) {
 | |
|   return UserGroupsWrapperActions.getCommunityByFullUrl(convoId);
 | |
| }
 | |
| 
 | |
| async function getAllCommunitiesNotCached(): Promise<Array<CommunityInfo>> {
 | |
|   return UserGroupsWrapperActions.getAllCommunities();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Removes the matching community from the wrapper and from the cached list of communities
 | |
|  */
 | |
| async function removeCommunityFromWrapper(_convoId: string, fullUrlWithOrWithoutPubkey: string) {
 | |
|   const fromWrapper = await UserGroupsWrapperActions.getCommunityByFullUrl(
 | |
|     fullUrlWithOrWithoutPubkey
 | |
|   );
 | |
| 
 | |
|   if (fromWrapper) {
 | |
|     await UserGroupsWrapperActions.eraseCommunityByFullUrl(fromWrapper.fullUrlWithPubkey);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * 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 getUserGroupTypes(): Array<UserGroupsType> {
 | |
|   return ['Community', 'LegacyGroup', 'Group'];
 | |
| }
 | |
| 
 | |
| export const SessionUtilUserGroups = {
 | |
|   // shared
 | |
|   isUserGroupToStoreInWrapper,
 | |
|   insertGroupsFromDBIntoWrapperAndRefresh,
 | |
|   getUserGroupTypes,
 | |
| 
 | |
|   // communities
 | |
|   isCommunityToStoreInWrapper,
 | |
|   getAllCommunitiesNotCached,
 | |
|   getCommunityByConvoIdNotCached,
 | |
|   removeCommunityFromWrapper,
 | |
| 
 | |
|   // legacy group
 | |
|   isLegacyGroupToStoreInWrapper,
 | |
|   isLegacyGroupToRemoveFromDBIfNotInWrapper,
 | |
| 
 | |
|   // group 03
 | |
|   isGroupToStoreInWrapper,
 | |
| };
 |