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.
		
		
		
		
		
			
		
			
				
	
	
		
			155 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			TypeScript
		
	
			
		
		
	
	
			155 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			TypeScript
		
	
| import { isEmpty } from 'lodash';
 | |
| import { CONVERSATION_PRIORITIES, ConversationTypeEnum } from '../../models/conversationAttributes';
 | |
| import { setLastProfileUpdateTimestamp } from '../../util/storage';
 | |
| import { UserConfigWrapperActions } from '../../webworker/workers/browser/libsession_worker_interface';
 | |
| import { getConversationController } from '../conversations';
 | |
| import { SyncUtils, UserUtils } from '../utils';
 | |
| import { fromHexToArray, sanitizeSessionUsername, toHex } from '../utils/String';
 | |
| import { AvatarDownload } from '../utils/job_runners/jobs/AvatarDownloadJob';
 | |
| import { generateFakeECKeyPair } from '../../test/test-utils/utils';
 | |
| 
 | |
| export type Profile = {
 | |
|   displayName: string | undefined;
 | |
|   profileUrl: string | null;
 | |
|   profileKey: Uint8Array | null;
 | |
|   priority: number | null; // passing null means to not update the priority at all (used for legacy config message for now)
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * This can be used to update our conversation display name with the given name right away, and plan an AvatarDownloadJob to retrieve the new avatar if needed to download it
 | |
|  */
 | |
| async function updateOurProfileSync({ displayName, profileUrl, profileKey, priority }: Profile) {
 | |
|   const us = UserUtils.getOurPubKeyStrFromCache();
 | |
|   const ourConvo = getConversationController().get(us);
 | |
|   if (!ourConvo?.id) {
 | |
|     window?.log?.warn('[profileupdate] Cannot update our profile without convo associated');
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   await updateProfileOfContact(us, displayName, profileUrl, profileKey);
 | |
|   if (priority !== null && ourConvo.get('priority') !== priority) {
 | |
|     ourConvo.set('priority', priority);
 | |
|     await ourConvo.commit();
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * This can be used to update the display name of the given pubkey right away, and plan an AvatarDownloadJob to retrieve the new avatar if needed to download it.
 | |
|  */
 | |
| async function updateProfileOfContact(
 | |
|   pubkey: string,
 | |
|   displayName: string | null | undefined,
 | |
|   profileUrl: string | null | undefined,
 | |
|   profileKey: Uint8Array | null | undefined
 | |
| ) {
 | |
|   const conversation = getConversationController().get(pubkey);
 | |
|   // TODO we should make sure that this function does not get call directly when `updateOurProfileSync` should be called instead. I.e. for avatars received in messages from ourself
 | |
|   if (!conversation || !conversation.isPrivate()) {
 | |
|     window.log.warn('updateProfileOfContact can only be used for existing and private convos');
 | |
|     return;
 | |
|   }
 | |
|   let changes = false;
 | |
|   const existingDisplayName = conversation.get('displayNameInProfile');
 | |
| 
 | |
|   // avoid setting the display name to an invalid value
 | |
|   if (existingDisplayName !== displayName && !isEmpty(displayName)) {
 | |
|     conversation.set('displayNameInProfile', displayName || undefined);
 | |
|     changes = true;
 | |
|   }
 | |
| 
 | |
|   const profileKeyHex = !profileKey || isEmpty(profileKey) ? null : toHex(profileKey);
 | |
| 
 | |
|   let avatarChanged = false;
 | |
|   // trust whatever we get as an update. It either comes from a shared config wrapper or one of that user's message. But in any case we should trust it, even if it gets resetted.
 | |
|   const prevPointer = conversation.get('avatarPointer');
 | |
|   const prevProfileKey = conversation.get('profileKey');
 | |
| 
 | |
|   // we have to set it right away and not in the async download job, as the next .commit will save it to the
 | |
|   // database and wrapper (and we do not want to override anything in the wrapper's content
 | |
|   // with what we have locally, so we need the commit to have already the right values in pointer and profileKey)
 | |
|   if (prevPointer !== profileUrl || prevProfileKey !== profileKeyHex) {
 | |
|     conversation.set({
 | |
|       avatarPointer: profileUrl || undefined,
 | |
|       profileKey: profileKeyHex || undefined,
 | |
|     });
 | |
| 
 | |
|     // if the avatar data we had before is not the same of what we received, we need to schedule a new avatar download job.
 | |
|     avatarChanged = true; // allow changes from strings to null/undefined to trigger a AvatarDownloadJob. If that happens, we want to remove the local attachment file.
 | |
|   }
 | |
| 
 | |
|   // if we have a local path to an downloaded  avatar, but no corresponding url/key for it, it means that
 | |
|   // the avatar was most likely removed so let's remove our link to that file.
 | |
|   if ((!profileUrl || !profileKeyHex) && conversation.get('avatarInProfile')) {
 | |
|     conversation.set({ avatarInProfile: undefined });
 | |
|     changes = true;
 | |
|   }
 | |
| 
 | |
|   if (changes) {
 | |
|     await conversation.commit();
 | |
|   }
 | |
| 
 | |
|   if (avatarChanged) {
 | |
|     // this call will download the new avatar or reset the local filepath if needed
 | |
|     await AvatarDownload.addAvatarDownloadJob({
 | |
|       conversationId: pubkey,
 | |
|     });
 | |
|   }
 | |
| }
 | |
| 
 | |
| export async function updateOurProfileDisplayName(newName: string, onboarding?: true) {
 | |
|   const cleanName = sanitizeSessionUsername(newName).trim();
 | |
| 
 | |
|   if (onboarding) {
 | |
|     try {
 | |
|       const tempKeyPair = generateFakeECKeyPair();
 | |
|       await UserConfigWrapperActions.init(tempKeyPair.privateKeyData, null);
 | |
|       const userInfoName = await UserConfigWrapperActions.setUserInfo(
 | |
|         cleanName,
 | |
|         CONVERSATION_PRIORITIES.default,
 | |
|         null
 | |
|       );
 | |
|       return userInfoName;
 | |
|     } finally {
 | |
|       await UserConfigWrapperActions.free();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const ourNumber = UserUtils.getOurPubKeyStrFromCache();
 | |
|   const conversation = await getConversationController().getOrCreateAndWait(
 | |
|     ourNumber,
 | |
|     ConversationTypeEnum.PRIVATE
 | |
|   );
 | |
| 
 | |
|   const dbProfileUrl = conversation.get('avatarPointer');
 | |
|   const dbProfileKey = conversation.get('profileKey')
 | |
|     ? fromHexToArray(conversation.get('profileKey')!)
 | |
|     : null;
 | |
|   const dbPriority = conversation.get('priority') || CONVERSATION_PRIORITIES.default;
 | |
| 
 | |
|   await UserConfigWrapperActions.setUserInfo(
 | |
|     cleanName,
 | |
|     dbPriority,
 | |
|     dbProfileUrl && dbProfileKey
 | |
|       ? {
 | |
|           url: dbProfileUrl,
 | |
|           key: dbProfileKey,
 | |
|         }
 | |
|       : null
 | |
|   );
 | |
| 
 | |
|   conversation.setSessionDisplayNameNoCommit(newName);
 | |
| 
 | |
|   // might be good to not trigger a sync if the name did not change
 | |
|   await conversation.commit();
 | |
|   await setLastProfileUpdateTimestamp(Date.now());
 | |
|   await SyncUtils.forceSyncConfigurationNowIfNeeded(true);
 | |
| 
 | |
|   return cleanName;
 | |
| }
 | |
| 
 | |
| export const ProfileManager = {
 | |
|   updateOurProfileSync,
 | |
|   updateProfileOfContact,
 | |
|   updateOurProfileDisplayName,
 | |
| };
 |