|
|
|
import { SignalService } from './../protobuf';
|
|
|
|
import { removeFromCache } from './cache';
|
|
|
|
import { EnvelopePlus } from './types';
|
|
|
|
import { getEnvelopeId } from './common';
|
|
|
|
|
|
|
|
import { PubKey } from '../session/types';
|
|
|
|
import { handleMessageJob } from './queuedJob';
|
|
|
|
import { downloadAttachment } from './attachments';
|
|
|
|
import _ from 'lodash';
|
|
|
|
import { StringUtils, UserUtils } from '../session/utils';
|
|
|
|
import { getConversationController } from '../session/conversations';
|
|
|
|
import { handleClosedGroupControlMessage } from './closedGroups';
|
|
|
|
import { MessageModel } from '../models/message';
|
|
|
|
import { MessageModelType } from '../models/messageType';
|
Session 1.7.5 (#2094)
* Added message requests disabled for now
* no longer showing empty space for conversations moved from list.
* Added syncing accepting of contact between running instances.
* Adding blocking of individual requests and syncing of block to devices. Added approval by replying to a message.
* fixed typos for translations and method name.
* Blocking, accepting on click and accepting on msg send
working across clients.
* adding setting of active_at to hide unapproved messages.
* adding feature flag for config message receiving
* fix archlinux pw unused issue
on archlinux, the appimage links to the system sqlite by default which
does not support sqlcipher
* hide activeAt = 0 convo from search results
Fixes #2033
* opengroup messages from blocked user are dropped
Fixes #2019
* opengroup messages from blocked user are dropped
Fixes #2019
* dismiss a call when answered from another of our devices
* add data-testid for leftpane sections and edit profile dialog
* update turn servers
* cleanup sessionprotobuf
* move the state of calling to its own slice
* no video track by default and will be turn ON if asked to
* message request refactoring.
* create offer and answer ourselves and do not use the negotiation needed
event.
this event is causing us to loop in negotiation needed when each side
try to create one, gets the answer and so on...
* auto select the first audio input on connection success webrtc
* add a way to choose the audioouput/mute a webrtc call
* mute audio from bg when video is in fullscreen
this is to avoid having two times the remote sound playing
one in the bg and one in the fullscreen
* Adding improvements to message request handling.
* Only updating approval when it is a true value as we consider a block a decline.
* Linting and formatting.
* More formatting and linting
* fixing merge conflicts
* linting and formatting changes
* darken a bit the green of sent message box in light theme
* disable deduplication based serverId+sender
only use the serverTimestamp+sender for searching because
serverId+sender might have false positive
* Fixing up block all logic.
* speed up fetching closed group's members avatar
* Applying PR changes.
* cleanup props passing of avatar and name with a custom hook
* fix a bug releasing the decrypted attachment blobs too early
* Adding trigger logic for conversation filtering of requests.
* Fixing rimraf transpile bug. Adding PR fixes - icon buttons.
* Minor call tweaks (#2051)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* Fetch translations (#2056)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* add type for i18n to run update after crowdin fetch with tools/updateI18nKeysType.py
* update to latest translations
* Open group regex fixes (#2058)
* Open group URL regex fixes
- Capital letters in room tokens were not being accepted (it eventually
gets lower-cased internally, which works fine, but that happens
*after* the URL is tested for acceptability).
- `-` in room was not being allowed (it is and always has been on SOGS,
session-android, and session-ios).
- single-letter room ids are valid, but only 2+ letter ids were being
accepted.
- complete URL regex wasn't anchored so something like
`garbagehttps://example.com/room?public_key=<64hex>moregarbage` was
being accepted in the GUI input (it fails later when other code tries
to parse it as a URL).
- removed `m` modifier from open group regex: without anchors it wasn't
doing anything anyway, but *with* anchors it would still allow
leading/trailing garbage if delineated by newlines.
- public key regex was accepting g-z letters, and not accepting A-F.
- various regex cleanups:
- use non-capture groups (?:...) rather than capturing groups (...)
- avoid repetition in host segment matching
- tightened up host pattern matching a bit:
- DNS host segments have a max length of 63
- Limit port max length to 5, and disallow starting with 0
* Show an error when the open group URL is invalid
It's quite disconcerting when you have a bad open group URL and try to
add it and the join button just "doesn't work" without any feedback at
all. Fix it to show an error message. (There is already an i18n entry
for this because this same message is thrown if the URL can't be parsed
later on).
* Add call duration (#2059)
* add call duration once connected
* close incoming call dialog if endCall from same sender
* disable message request toggle if featureFlag is OFF
* Cleanup message request (#2063)
* close incoming call dialog if endCall from seame sender
* disable message request toggle if featureFlag is OFF
* cleanup UI of message requests
* mark all existing conversations as approved in a migration
* fix regex with conversationID for opengroups
* Various UI fixes (#2070)
* cleanup unused convo json fields in db
* display a toast if the user is not approved yet on call OFFER received
* enable CBR for calls
* do not update active_at on configMessage if !!active_at
* remove mkdirp dependency
* disable call button if focused convo is blocked
* quote: do not include the full body in quote, but just the first 100
* click on the edit profile qr code padding
* Allow longer input for opengroup join overlay
Fixes #2068
* Fix overlay feature for start new session button
* make ringing depend on redux CALL status
* turn ON read-receipt by default
* keep read-receipts disabled by default (#2071)
* refactor most of the components to outside of their Session folder (#2072)
* refactor most of the components to outside of their Session folder
* finish moving overlay and memberListItem to react hook
* fix bug with kicked member len >2 not being displayed
also sort admins first in UpdateGroupMembers dialog
* fix admin leaving text of groupNotification
* add a useFocusMount hook to focus input fields on mount
* make click avatar convo item open only user dialog
* cleanup config default.json
* make sure to use convoController to build sync message
* disable showing pubkey on opengroups
* add a pause on audio playback
Fixes #2079
* Minor styling fix for large amount of message requests (#2080)
* Minor styling fix for large amount of message requests
* Vertical center fix for message request banner.
* removing top margin from banner again.
* reactify group updates text bubble from redux store (#2083)
* add crown icon for closed group admins (#2084)
* disable call for now + fix left pane actions overflow (#2085)
* Fix attachment dl freeze (#2086)
* fix attachment download freezing app for some opengroups
* make registration page work with smaller height
* Unban UI (#2091)
* adding basic functionaliy for unbanning a user
* merge ban and unban user dialog in to one dialog
Co-authored-by: warrickct <warrickct@gmail.com>
* use React Provider for convoListItem (#2088)
this is to avoid passing down the prop to all the components
* fix closed group updates undefined on no names (#2092)
Co-authored-by: Warrick Corfe-Tan <warrickct@gmail.com>
Co-authored-by: Jason Rhinelander <jason@imaginary.ca>
Co-authored-by: Warrick <wcor690@aucklanduni.ac.nz>
3 years ago
|
|
|
import { getMessageBySender, getMessageBySenderAndServerTimestamp } from '../../ts/data/data';
|
|
|
|
import { ConversationModel, ConversationTypeEnum } from '../models/conversation';
|
|
|
|
import { allowOnlyOneAtATime } from '../session/utils/Promise';
|
|
|
|
import { toHex } from '../session/utils/String';
|
|
|
|
|
|
|
|
export async function updateProfileOneAtATime(
|
|
|
|
conversation: ConversationModel,
|
|
|
|
profile: SignalService.DataMessage.ILokiProfile,
|
|
|
|
profileKey?: Uint8Array | null // was any
|
|
|
|
) {
|
|
|
|
if (!conversation?.id) {
|
|
|
|
window?.log?.warn('Cannot update profile with empty convoid');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const oneAtaTimeStr = `updateProfileOneAtATime:${conversation.id}`;
|
|
|
|
return allowOnlyOneAtATime(oneAtaTimeStr, async () => {
|
Session 1.7.5 (#2094)
* Added message requests disabled for now
* no longer showing empty space for conversations moved from list.
* Added syncing accepting of contact between running instances.
* Adding blocking of individual requests and syncing of block to devices. Added approval by replying to a message.
* fixed typos for translations and method name.
* Blocking, accepting on click and accepting on msg send
working across clients.
* adding setting of active_at to hide unapproved messages.
* adding feature flag for config message receiving
* fix archlinux pw unused issue
on archlinux, the appimage links to the system sqlite by default which
does not support sqlcipher
* hide activeAt = 0 convo from search results
Fixes #2033
* opengroup messages from blocked user are dropped
Fixes #2019
* opengroup messages from blocked user are dropped
Fixes #2019
* dismiss a call when answered from another of our devices
* add data-testid for leftpane sections and edit profile dialog
* update turn servers
* cleanup sessionprotobuf
* move the state of calling to its own slice
* no video track by default and will be turn ON if asked to
* message request refactoring.
* create offer and answer ourselves and do not use the negotiation needed
event.
this event is causing us to loop in negotiation needed when each side
try to create one, gets the answer and so on...
* auto select the first audio input on connection success webrtc
* add a way to choose the audioouput/mute a webrtc call
* mute audio from bg when video is in fullscreen
this is to avoid having two times the remote sound playing
one in the bg and one in the fullscreen
* Adding improvements to message request handling.
* Only updating approval when it is a true value as we consider a block a decline.
* Linting and formatting.
* More formatting and linting
* fixing merge conflicts
* linting and formatting changes
* darken a bit the green of sent message box in light theme
* disable deduplication based serverId+sender
only use the serverTimestamp+sender for searching because
serverId+sender might have false positive
* Fixing up block all logic.
* speed up fetching closed group's members avatar
* Applying PR changes.
* cleanup props passing of avatar and name with a custom hook
* fix a bug releasing the decrypted attachment blobs too early
* Adding trigger logic for conversation filtering of requests.
* Fixing rimraf transpile bug. Adding PR fixes - icon buttons.
* Minor call tweaks (#2051)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* Fetch translations (#2056)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* add type for i18n to run update after crowdin fetch with tools/updateI18nKeysType.py
* update to latest translations
* Open group regex fixes (#2058)
* Open group URL regex fixes
- Capital letters in room tokens were not being accepted (it eventually
gets lower-cased internally, which works fine, but that happens
*after* the URL is tested for acceptability).
- `-` in room was not being allowed (it is and always has been on SOGS,
session-android, and session-ios).
- single-letter room ids are valid, but only 2+ letter ids were being
accepted.
- complete URL regex wasn't anchored so something like
`garbagehttps://example.com/room?public_key=<64hex>moregarbage` was
being accepted in the GUI input (it fails later when other code tries
to parse it as a URL).
- removed `m` modifier from open group regex: without anchors it wasn't
doing anything anyway, but *with* anchors it would still allow
leading/trailing garbage if delineated by newlines.
- public key regex was accepting g-z letters, and not accepting A-F.
- various regex cleanups:
- use non-capture groups (?:...) rather than capturing groups (...)
- avoid repetition in host segment matching
- tightened up host pattern matching a bit:
- DNS host segments have a max length of 63
- Limit port max length to 5, and disallow starting with 0
* Show an error when the open group URL is invalid
It's quite disconcerting when you have a bad open group URL and try to
add it and the join button just "doesn't work" without any feedback at
all. Fix it to show an error message. (There is already an i18n entry
for this because this same message is thrown if the URL can't be parsed
later on).
* Add call duration (#2059)
* add call duration once connected
* close incoming call dialog if endCall from same sender
* disable message request toggle if featureFlag is OFF
* Cleanup message request (#2063)
* close incoming call dialog if endCall from seame sender
* disable message request toggle if featureFlag is OFF
* cleanup UI of message requests
* mark all existing conversations as approved in a migration
* fix regex with conversationID for opengroups
* Various UI fixes (#2070)
* cleanup unused convo json fields in db
* display a toast if the user is not approved yet on call OFFER received
* enable CBR for calls
* do not update active_at on configMessage if !!active_at
* remove mkdirp dependency
* disable call button if focused convo is blocked
* quote: do not include the full body in quote, but just the first 100
* click on the edit profile qr code padding
* Allow longer input for opengroup join overlay
Fixes #2068
* Fix overlay feature for start new session button
* make ringing depend on redux CALL status
* turn ON read-receipt by default
* keep read-receipts disabled by default (#2071)
* refactor most of the components to outside of their Session folder (#2072)
* refactor most of the components to outside of their Session folder
* finish moving overlay and memberListItem to react hook
* fix bug with kicked member len >2 not being displayed
also sort admins first in UpdateGroupMembers dialog
* fix admin leaving text of groupNotification
* add a useFocusMount hook to focus input fields on mount
* make click avatar convo item open only user dialog
* cleanup config default.json
* make sure to use convoController to build sync message
* disable showing pubkey on opengroups
* add a pause on audio playback
Fixes #2079
* Minor styling fix for large amount of message requests (#2080)
* Minor styling fix for large amount of message requests
* Vertical center fix for message request banner.
* removing top margin from banner again.
* reactify group updates text bubble from redux store (#2083)
* add crown icon for closed group admins (#2084)
* disable call for now + fix left pane actions overflow (#2085)
* Fix attachment dl freeze (#2086)
* fix attachment download freezing app for some opengroups
* make registration page work with smaller height
* Unban UI (#2091)
* adding basic functionaliy for unbanning a user
* merge ban and unban user dialog in to one dialog
Co-authored-by: warrickct <warrickct@gmail.com>
* use React Provider for convoListItem (#2088)
this is to avoid passing down the prop to all the components
* fix closed group updates undefined on no names (#2092)
Co-authored-by: Warrick Corfe-Tan <warrickct@gmail.com>
Co-authored-by: Jason Rhinelander <jason@imaginary.ca>
Co-authored-by: Warrick <wcor690@aucklanduni.ac.nz>
3 years ago
|
|
|
return createOrUpdateProfile(conversation, profile, profileKey);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
Session 1.7.5 (#2094)
* Added message requests disabled for now
* no longer showing empty space for conversations moved from list.
* Added syncing accepting of contact between running instances.
* Adding blocking of individual requests and syncing of block to devices. Added approval by replying to a message.
* fixed typos for translations and method name.
* Blocking, accepting on click and accepting on msg send
working across clients.
* adding setting of active_at to hide unapproved messages.
* adding feature flag for config message receiving
* fix archlinux pw unused issue
on archlinux, the appimage links to the system sqlite by default which
does not support sqlcipher
* hide activeAt = 0 convo from search results
Fixes #2033
* opengroup messages from blocked user are dropped
Fixes #2019
* opengroup messages from blocked user are dropped
Fixes #2019
* dismiss a call when answered from another of our devices
* add data-testid for leftpane sections and edit profile dialog
* update turn servers
* cleanup sessionprotobuf
* move the state of calling to its own slice
* no video track by default and will be turn ON if asked to
* message request refactoring.
* create offer and answer ourselves and do not use the negotiation needed
event.
this event is causing us to loop in negotiation needed when each side
try to create one, gets the answer and so on...
* auto select the first audio input on connection success webrtc
* add a way to choose the audioouput/mute a webrtc call
* mute audio from bg when video is in fullscreen
this is to avoid having two times the remote sound playing
one in the bg and one in the fullscreen
* Adding improvements to message request handling.
* Only updating approval when it is a true value as we consider a block a decline.
* Linting and formatting.
* More formatting and linting
* fixing merge conflicts
* linting and formatting changes
* darken a bit the green of sent message box in light theme
* disable deduplication based serverId+sender
only use the serverTimestamp+sender for searching because
serverId+sender might have false positive
* Fixing up block all logic.
* speed up fetching closed group's members avatar
* Applying PR changes.
* cleanup props passing of avatar and name with a custom hook
* fix a bug releasing the decrypted attachment blobs too early
* Adding trigger logic for conversation filtering of requests.
* Fixing rimraf transpile bug. Adding PR fixes - icon buttons.
* Minor call tweaks (#2051)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* Fetch translations (#2056)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* add type for i18n to run update after crowdin fetch with tools/updateI18nKeysType.py
* update to latest translations
* Open group regex fixes (#2058)
* Open group URL regex fixes
- Capital letters in room tokens were not being accepted (it eventually
gets lower-cased internally, which works fine, but that happens
*after* the URL is tested for acceptability).
- `-` in room was not being allowed (it is and always has been on SOGS,
session-android, and session-ios).
- single-letter room ids are valid, but only 2+ letter ids were being
accepted.
- complete URL regex wasn't anchored so something like
`garbagehttps://example.com/room?public_key=<64hex>moregarbage` was
being accepted in the GUI input (it fails later when other code tries
to parse it as a URL).
- removed `m` modifier from open group regex: without anchors it wasn't
doing anything anyway, but *with* anchors it would still allow
leading/trailing garbage if delineated by newlines.
- public key regex was accepting g-z letters, and not accepting A-F.
- various regex cleanups:
- use non-capture groups (?:...) rather than capturing groups (...)
- avoid repetition in host segment matching
- tightened up host pattern matching a bit:
- DNS host segments have a max length of 63
- Limit port max length to 5, and disallow starting with 0
* Show an error when the open group URL is invalid
It's quite disconcerting when you have a bad open group URL and try to
add it and the join button just "doesn't work" without any feedback at
all. Fix it to show an error message. (There is already an i18n entry
for this because this same message is thrown if the URL can't be parsed
later on).
* Add call duration (#2059)
* add call duration once connected
* close incoming call dialog if endCall from same sender
* disable message request toggle if featureFlag is OFF
* Cleanup message request (#2063)
* close incoming call dialog if endCall from seame sender
* disable message request toggle if featureFlag is OFF
* cleanup UI of message requests
* mark all existing conversations as approved in a migration
* fix regex with conversationID for opengroups
* Various UI fixes (#2070)
* cleanup unused convo json fields in db
* display a toast if the user is not approved yet on call OFFER received
* enable CBR for calls
* do not update active_at on configMessage if !!active_at
* remove mkdirp dependency
* disable call button if focused convo is blocked
* quote: do not include the full body in quote, but just the first 100
* click on the edit profile qr code padding
* Allow longer input for opengroup join overlay
Fixes #2068
* Fix overlay feature for start new session button
* make ringing depend on redux CALL status
* turn ON read-receipt by default
* keep read-receipts disabled by default (#2071)
* refactor most of the components to outside of their Session folder (#2072)
* refactor most of the components to outside of their Session folder
* finish moving overlay and memberListItem to react hook
* fix bug with kicked member len >2 not being displayed
also sort admins first in UpdateGroupMembers dialog
* fix admin leaving text of groupNotification
* add a useFocusMount hook to focus input fields on mount
* make click avatar convo item open only user dialog
* cleanup config default.json
* make sure to use convoController to build sync message
* disable showing pubkey on opengroups
* add a pause on audio playback
Fixes #2079
* Minor styling fix for large amount of message requests (#2080)
* Minor styling fix for large amount of message requests
* Vertical center fix for message request banner.
* removing top margin from banner again.
* reactify group updates text bubble from redux store (#2083)
* add crown icon for closed group admins (#2084)
* disable call for now + fix left pane actions overflow (#2085)
* Fix attachment dl freeze (#2086)
* fix attachment download freezing app for some opengroups
* make registration page work with smaller height
* Unban UI (#2091)
* adding basic functionaliy for unbanning a user
* merge ban and unban user dialog in to one dialog
Co-authored-by: warrickct <warrickct@gmail.com>
* use React Provider for convoListItem (#2088)
this is to avoid passing down the prop to all the components
* fix closed group updates undefined on no names (#2092)
Co-authored-by: Warrick Corfe-Tan <warrickct@gmail.com>
Co-authored-by: Jason Rhinelander <jason@imaginary.ca>
Co-authored-by: Warrick <wcor690@aucklanduni.ac.nz>
3 years ago
|
|
|
/**
|
|
|
|
* Creates a new profile from the profile provided. Creates the profile if it doesn't exist.
|
|
|
|
*/
|
|
|
|
async function createOrUpdateProfile(
|
|
|
|
conversation: ConversationModel,
|
|
|
|
profile: SignalService.DataMessage.ILokiProfile,
|
|
|
|
profileKey?: Uint8Array | null // was any
|
|
|
|
) {
|
|
|
|
const { dcodeIO, textsecure, Signal } = window;
|
|
|
|
|
|
|
|
// Retain old values unless changed:
|
|
|
|
const newProfile = conversation.get('profile') || {};
|
|
|
|
|
|
|
|
newProfile.displayName = profile.displayName;
|
|
|
|
|
|
|
|
if (profile.profilePicture && profileKey) {
|
|
|
|
const prevPointer = conversation.get('avatarPointer');
|
|
|
|
const needsUpdate = !prevPointer || !_.isEqual(prevPointer, profile.profilePicture);
|
|
|
|
|
|
|
|
if (needsUpdate) {
|
|
|
|
try {
|
|
|
|
const downloaded = await downloadAttachment({
|
|
|
|
url: profile.profilePicture,
|
|
|
|
isRaw: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
// null => use placeholder with color and first letter
|
|
|
|
let path = null;
|
|
|
|
if (profileKey) {
|
|
|
|
// Convert profileKey to ArrayBuffer, if needed
|
|
|
|
const encoding = typeof profileKey === 'string' ? 'base64' : null;
|
|
|
|
try {
|
|
|
|
const profileKeyArrayBuffer = dcodeIO.ByteBuffer.wrap(
|
|
|
|
profileKey,
|
|
|
|
encoding
|
|
|
|
).toArrayBuffer();
|
|
|
|
const decryptedData = await textsecure.crypto.decryptProfile(
|
|
|
|
downloaded.data,
|
|
|
|
profileKeyArrayBuffer
|
|
|
|
);
|
|
|
|
const upgraded = await Signal.Migrations.processNewAttachment({
|
|
|
|
...downloaded,
|
|
|
|
data: decryptedData,
|
|
|
|
});
|
|
|
|
// Only update the convo if the download and decrypt is a success
|
|
|
|
conversation.set('avatarPointer', profile.profilePicture);
|
|
|
|
conversation.set('profileKey', toHex(profileKey));
|
|
|
|
({ path } = upgraded);
|
|
|
|
} catch (e) {
|
|
|
|
window?.log?.error(`Could not decrypt profile image: ${e}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
newProfile.avatar = path;
|
|
|
|
} catch (e) {
|
|
|
|
window.log.warn('Failed to download attachment at', profile.profilePicture);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (profileKey) {
|
|
|
|
newProfile.avatar = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const conv = await getConversationController().getOrCreateAndWait(
|
|
|
|
conversation.id,
|
|
|
|
ConversationTypeEnum.PRIVATE
|
|
|
|
);
|
|
|
|
await conv.setLokiProfile(newProfile);
|
|
|
|
await conv.commit();
|
|
|
|
}
|
|
|
|
|
|
|
|
function cleanAttachment(attachment: any) {
|
|
|
|
return {
|
|
|
|
..._.omit(attachment, 'thumbnail'),
|
|
|
|
id: attachment.id.toString(),
|
|
|
|
key: attachment.key ? StringUtils.decode(attachment.key, 'base64') : null,
|
|
|
|
digest:
|
|
|
|
attachment.digest && attachment.digest.length > 0
|
|
|
|
? StringUtils.decode(attachment.digest, 'base64')
|
|
|
|
: null,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function cleanAttachments(decrypted: any) {
|
|
|
|
const { quote, group } = decrypted;
|
|
|
|
|
|
|
|
// Here we go from binary to string/base64 in all AttachmentPointer digest/key fields
|
|
|
|
|
|
|
|
if (group && group.type === SignalService.GroupContext.Type.UPDATE) {
|
|
|
|
if (group.avatar !== null) {
|
|
|
|
group.avatar = cleanAttachment(group.avatar);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
decrypted.attachments = (decrypted.attachments || []).map(cleanAttachment);
|
|
|
|
decrypted.preview = (decrypted.preview || []).map((item: any) => {
|
|
|
|
const { image } = item;
|
|
|
|
|
|
|
|
if (!image) {
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...item,
|
|
|
|
image: cleanAttachment(image),
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
if (quote) {
|
|
|
|
if (quote.id) {
|
|
|
|
quote.id = _.toNumber(quote.id);
|
|
|
|
}
|
|
|
|
|
|
|
|
quote.attachments = (quote.attachments || []).map((item: any) => {
|
|
|
|
const { thumbnail } = item;
|
|
|
|
|
|
|
|
if (!thumbnail || thumbnail.length === 0) {
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...item,
|
|
|
|
thumbnail: cleanAttachment(item.thumbnail),
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function processDecrypted(
|
|
|
|
envelope: EnvelopePlus,
|
|
|
|
decrypted: SignalService.IDataMessage
|
|
|
|
) {
|
|
|
|
/* tslint:disable:no-bitwise */
|
|
|
|
const FLAGS = SignalService.DataMessage.Flags;
|
|
|
|
|
|
|
|
// Now that its decrypted, validate the message and clean it up for consumer
|
|
|
|
// processing
|
|
|
|
// Note that messages may (generally) only perform one action and we ignore remaining
|
|
|
|
// fields after the first action.
|
|
|
|
|
|
|
|
if (decrypted.flags == null) {
|
|
|
|
decrypted.flags = 0;
|
|
|
|
}
|
|
|
|
if (decrypted.expireTimer == null) {
|
|
|
|
decrypted.expireTimer = 0;
|
|
|
|
}
|
|
|
|
if (decrypted.flags & FLAGS.EXPIRATION_TIMER_UPDATE) {
|
|
|
|
decrypted.body = '';
|
|
|
|
decrypted.attachments = [];
|
|
|
|
} else if (decrypted.flags !== 0) {
|
|
|
|
throw new Error('Unknown flags in message');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (decrypted.group) {
|
|
|
|
// decrypted.group.id = new TextDecoder('utf-8').decode(decrypted.group.id);
|
|
|
|
|
|
|
|
switch (decrypted.group.type) {
|
|
|
|
case SignalService.GroupContext.Type.UPDATE:
|
|
|
|
decrypted.body = '';
|
|
|
|
decrypted.attachments = [];
|
|
|
|
break;
|
|
|
|
case SignalService.GroupContext.Type.QUIT:
|
|
|
|
decrypted.body = '';
|
|
|
|
decrypted.attachments = [];
|
|
|
|
break;
|
|
|
|
case SignalService.GroupContext.Type.DELIVER:
|
|
|
|
decrypted.group.name = null;
|
|
|
|
decrypted.group.members = [];
|
|
|
|
decrypted.group.avatar = null;
|
|
|
|
break;
|
|
|
|
case SignalService.GroupContext.Type.REQUEST_INFO:
|
|
|
|
decrypted.body = '';
|
|
|
|
decrypted.attachments = [];
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
await removeFromCache(envelope);
|
|
|
|
throw new Error('Unknown group message type');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const attachmentCount = decrypted?.attachments?.length || 0;
|
|
|
|
const ATTACHMENT_MAX = 32;
|
|
|
|
if (attachmentCount > ATTACHMENT_MAX) {
|
|
|
|
await removeFromCache(envelope);
|
|
|
|
throw new Error(
|
|
|
|
`Too many attachments: ${attachmentCount} included in one message, max is ${ATTACHMENT_MAX}`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanAttachments(decrypted);
|
|
|
|
|
|
|
|
// if the decrypted dataMessage timestamp is not set, copy the one from the envelope
|
|
|
|
if (!_.toNumber(decrypted?.timestamp)) {
|
|
|
|
decrypted.timestamp = envelope.timestamp;
|
|
|
|
}
|
|
|
|
|
|
|
|
return decrypted as SignalService.DataMessage;
|
|
|
|
/* tslint:disable:no-bitwise */
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isMessageEmpty(message: SignalService.DataMessage) {
|
|
|
|
const { flags, body, attachments, group, quote, preview, openGroupInvitation } = message;
|
|
|
|
|
|
|
|
return (
|
|
|
|
!flags &&
|
|
|
|
// FIXME remove this hack to drop auto friend requests messages in a few weeks 15/07/2020
|
|
|
|
isBodyEmpty(body) &&
|
|
|
|
_.isEmpty(attachments) &&
|
|
|
|
_.isEmpty(group) &&
|
|
|
|
_.isEmpty(quote) &&
|
|
|
|
_.isEmpty(preview) &&
|
|
|
|
_.isEmpty(openGroupInvitation)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isBodyEmpty(body: string) {
|
|
|
|
return _.isEmpty(body);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* We have a few origins possible
|
|
|
|
* - if the message is from a private conversation with a friend and he wrote to us,
|
|
|
|
* the conversation to add the message to is our friend pubkey, so envelope.source
|
|
|
|
* - if the message is from a medium group conversation
|
|
|
|
* * envelope.source is the medium group pubkey
|
|
|
|
* * envelope.senderIdentity is the author pubkey (the one who sent the message)
|
|
|
|
* - at last, if the message is a syncMessage,
|
|
|
|
* * envelope.source is our pubkey (our other device has the same pubkey as us)
|
|
|
|
* * dataMessage.syncTarget is either the group public key OR the private conversation this message is about.
|
|
|
|
*/
|
|
|
|
export async function handleDataMessage(
|
|
|
|
envelope: EnvelopePlus,
|
|
|
|
dataMessage: SignalService.IDataMessage,
|
|
|
|
messageHash?: string
|
|
|
|
): Promise<void> {
|
|
|
|
// we handle group updates from our other devices in handleClosedGroupControlMessage()
|
|
|
|
if (dataMessage.closedGroupControlMessage) {
|
|
|
|
await handleClosedGroupControlMessage(
|
|
|
|
envelope,
|
|
|
|
dataMessage.closedGroupControlMessage as SignalService.DataMessage.ClosedGroupControlMessage
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const message = await processDecrypted(envelope, dataMessage);
|
|
|
|
const source = dataMessage.syncTarget || envelope.source;
|
|
|
|
const senderPubKey = envelope.senderIdentity || envelope.source;
|
|
|
|
const isMe = UserUtils.isUsFromCache(senderPubKey);
|
|
|
|
const isSyncMessage = Boolean(dataMessage.syncTarget?.length);
|
|
|
|
|
|
|
|
window?.log?.info(`Handle dataMessage from ${source} `);
|
|
|
|
|
|
|
|
if (isSyncMessage && !isMe) {
|
|
|
|
window?.log?.warn('Got a sync message from someone else than me. Dropping it.');
|
|
|
|
return removeFromCache(envelope);
|
|
|
|
} else if (isSyncMessage && dataMessage.syncTarget) {
|
|
|
|
// override the envelope source
|
|
|
|
envelope.source = dataMessage.syncTarget;
|
|
|
|
}
|
|
|
|
|
|
|
|
const senderConversation = await getConversationController().getOrCreateAndWait(
|
|
|
|
senderPubKey,
|
|
|
|
ConversationTypeEnum.PRIVATE
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check if we need to update any profile names
|
|
|
|
if (!isMe && senderConversation && message.profile) {
|
|
|
|
// do not await this
|
|
|
|
void updateProfileOneAtATime(senderConversation, message.profile, message.profileKey);
|
|
|
|
}
|
|
|
|
if (isMessageEmpty(message)) {
|
|
|
|
window?.log?.warn(`Message ${getEnvelopeId(envelope)} ignored; it was empty`);
|
|
|
|
return removeFromCache(envelope);
|
|
|
|
}
|
|
|
|
|
|
|
|
const ev: any = {};
|
|
|
|
if (isMe) {
|
|
|
|
// Data messages for medium groups don't arrive as sync messages. Instead,
|
|
|
|
// linked devices poll for group messages independently, thus they need
|
|
|
|
// to recognise some of those messages at their own.
|
|
|
|
ev.type = 'sent';
|
|
|
|
} else {
|
|
|
|
ev.type = 'message';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (envelope.senderIdentity) {
|
|
|
|
message.group = {
|
|
|
|
id: envelope.source as any, // FIXME Uint8Array vs string
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
ev.confirm = () => removeFromCache(envelope);
|
|
|
|
|
|
|
|
ev.data = {
|
|
|
|
source: senderPubKey,
|
|
|
|
destination: isMe ? message.syncTarget : undefined,
|
|
|
|
sourceDevice: 1,
|
|
|
|
timestamp: _.toNumber(envelope.timestamp),
|
|
|
|
receivedAt: envelope.receivedAt,
|
|
|
|
message,
|
|
|
|
messageHash,
|
|
|
|
};
|
|
|
|
|
|
|
|
await handleMessageEvent(ev); // dataMessage
|
|
|
|
}
|
|
|
|
|
|
|
|
type MessageDuplicateSearchType = {
|
|
|
|
body: string;
|
|
|
|
id: string;
|
|
|
|
timestamp: number;
|
|
|
|
serverId?: number;
|
|
|
|
};
|
|
|
|
|
|
|
|
export type MessageId = {
|
|
|
|
source: string;
|
|
|
|
serverId: number;
|
|
|
|
serverTimestamp: number;
|
|
|
|
sourceDevice: number;
|
|
|
|
timestamp: number;
|
|
|
|
message: MessageDuplicateSearchType;
|
|
|
|
};
|
|
|
|
const PUBLICCHAT_MIN_TIME_BETWEEN_DUPLICATE_MESSAGES = 10 * 1000; // 10s
|
|
|
|
|
|
|
|
export async function isMessageDuplicate({
|
|
|
|
source,
|
|
|
|
sourceDevice,
|
|
|
|
timestamp,
|
|
|
|
message,
|
|
|
|
serverId,
|
|
|
|
serverTimestamp,
|
|
|
|
}: MessageId) {
|
|
|
|
const { Errors } = window.Signal.Types;
|
|
|
|
// serverId is only used for opengroupv2
|
|
|
|
try {
|
|
|
|
let result;
|
|
|
|
if (serverId || serverTimestamp) {
|
Session 1.7.5 (#2094)
* Added message requests disabled for now
* no longer showing empty space for conversations moved from list.
* Added syncing accepting of contact between running instances.
* Adding blocking of individual requests and syncing of block to devices. Added approval by replying to a message.
* fixed typos for translations and method name.
* Blocking, accepting on click and accepting on msg send
working across clients.
* adding setting of active_at to hide unapproved messages.
* adding feature flag for config message receiving
* fix archlinux pw unused issue
on archlinux, the appimage links to the system sqlite by default which
does not support sqlcipher
* hide activeAt = 0 convo from search results
Fixes #2033
* opengroup messages from blocked user are dropped
Fixes #2019
* opengroup messages from blocked user are dropped
Fixes #2019
* dismiss a call when answered from another of our devices
* add data-testid for leftpane sections and edit profile dialog
* update turn servers
* cleanup sessionprotobuf
* move the state of calling to its own slice
* no video track by default and will be turn ON if asked to
* message request refactoring.
* create offer and answer ourselves and do not use the negotiation needed
event.
this event is causing us to loop in negotiation needed when each side
try to create one, gets the answer and so on...
* auto select the first audio input on connection success webrtc
* add a way to choose the audioouput/mute a webrtc call
* mute audio from bg when video is in fullscreen
this is to avoid having two times the remote sound playing
one in the bg and one in the fullscreen
* Adding improvements to message request handling.
* Only updating approval when it is a true value as we consider a block a decline.
* Linting and formatting.
* More formatting and linting
* fixing merge conflicts
* linting and formatting changes
* darken a bit the green of sent message box in light theme
* disable deduplication based serverId+sender
only use the serverTimestamp+sender for searching because
serverId+sender might have false positive
* Fixing up block all logic.
* speed up fetching closed group's members avatar
* Applying PR changes.
* cleanup props passing of avatar and name with a custom hook
* fix a bug releasing the decrypted attachment blobs too early
* Adding trigger logic for conversation filtering of requests.
* Fixing rimraf transpile bug. Adding PR fixes - icon buttons.
* Minor call tweaks (#2051)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* Fetch translations (#2056)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* add type for i18n to run update after crowdin fetch with tools/updateI18nKeysType.py
* update to latest translations
* Open group regex fixes (#2058)
* Open group URL regex fixes
- Capital letters in room tokens were not being accepted (it eventually
gets lower-cased internally, which works fine, but that happens
*after* the URL is tested for acceptability).
- `-` in room was not being allowed (it is and always has been on SOGS,
session-android, and session-ios).
- single-letter room ids are valid, but only 2+ letter ids were being
accepted.
- complete URL regex wasn't anchored so something like
`garbagehttps://example.com/room?public_key=<64hex>moregarbage` was
being accepted in the GUI input (it fails later when other code tries
to parse it as a URL).
- removed `m` modifier from open group regex: without anchors it wasn't
doing anything anyway, but *with* anchors it would still allow
leading/trailing garbage if delineated by newlines.
- public key regex was accepting g-z letters, and not accepting A-F.
- various regex cleanups:
- use non-capture groups (?:...) rather than capturing groups (...)
- avoid repetition in host segment matching
- tightened up host pattern matching a bit:
- DNS host segments have a max length of 63
- Limit port max length to 5, and disallow starting with 0
* Show an error when the open group URL is invalid
It's quite disconcerting when you have a bad open group URL and try to
add it and the join button just "doesn't work" without any feedback at
all. Fix it to show an error message. (There is already an i18n entry
for this because this same message is thrown if the URL can't be parsed
later on).
* Add call duration (#2059)
* add call duration once connected
* close incoming call dialog if endCall from same sender
* disable message request toggle if featureFlag is OFF
* Cleanup message request (#2063)
* close incoming call dialog if endCall from seame sender
* disable message request toggle if featureFlag is OFF
* cleanup UI of message requests
* mark all existing conversations as approved in a migration
* fix regex with conversationID for opengroups
* Various UI fixes (#2070)
* cleanup unused convo json fields in db
* display a toast if the user is not approved yet on call OFFER received
* enable CBR for calls
* do not update active_at on configMessage if !!active_at
* remove mkdirp dependency
* disable call button if focused convo is blocked
* quote: do not include the full body in quote, but just the first 100
* click on the edit profile qr code padding
* Allow longer input for opengroup join overlay
Fixes #2068
* Fix overlay feature for start new session button
* make ringing depend on redux CALL status
* turn ON read-receipt by default
* keep read-receipts disabled by default (#2071)
* refactor most of the components to outside of their Session folder (#2072)
* refactor most of the components to outside of their Session folder
* finish moving overlay and memberListItem to react hook
* fix bug with kicked member len >2 not being displayed
also sort admins first in UpdateGroupMembers dialog
* fix admin leaving text of groupNotification
* add a useFocusMount hook to focus input fields on mount
* make click avatar convo item open only user dialog
* cleanup config default.json
* make sure to use convoController to build sync message
* disable showing pubkey on opengroups
* add a pause on audio playback
Fixes #2079
* Minor styling fix for large amount of message requests (#2080)
* Minor styling fix for large amount of message requests
* Vertical center fix for message request banner.
* removing top margin from banner again.
* reactify group updates text bubble from redux store (#2083)
* add crown icon for closed group admins (#2084)
* disable call for now + fix left pane actions overflow (#2085)
* Fix attachment dl freeze (#2086)
* fix attachment download freezing app for some opengroups
* make registration page work with smaller height
* Unban UI (#2091)
* adding basic functionaliy for unbanning a user
* merge ban and unban user dialog in to one dialog
Co-authored-by: warrickct <warrickct@gmail.com>
* use React Provider for convoListItem (#2088)
this is to avoid passing down the prop to all the components
* fix closed group updates undefined on no names (#2092)
Co-authored-by: Warrick Corfe-Tan <warrickct@gmail.com>
Co-authored-by: Jason Rhinelander <jason@imaginary.ca>
Co-authored-by: Warrick <wcor690@aucklanduni.ac.nz>
3 years ago
|
|
|
// first try to find a duplicate with the same serverTimestamp from this sender
|
|
|
|
if (!result && serverTimestamp) {
|
|
|
|
result = await getMessageBySenderAndServerTimestamp({
|
|
|
|
source,
|
|
|
|
serverTimestamp,
|
|
|
|
});
|
|
|
|
}
|
Session 1.7.5 (#2094)
* Added message requests disabled for now
* no longer showing empty space for conversations moved from list.
* Added syncing accepting of contact between running instances.
* Adding blocking of individual requests and syncing of block to devices. Added approval by replying to a message.
* fixed typos for translations and method name.
* Blocking, accepting on click and accepting on msg send
working across clients.
* adding setting of active_at to hide unapproved messages.
* adding feature flag for config message receiving
* fix archlinux pw unused issue
on archlinux, the appimage links to the system sqlite by default which
does not support sqlcipher
* hide activeAt = 0 convo from search results
Fixes #2033
* opengroup messages from blocked user are dropped
Fixes #2019
* opengroup messages from blocked user are dropped
Fixes #2019
* dismiss a call when answered from another of our devices
* add data-testid for leftpane sections and edit profile dialog
* update turn servers
* cleanup sessionprotobuf
* move the state of calling to its own slice
* no video track by default and will be turn ON if asked to
* message request refactoring.
* create offer and answer ourselves and do not use the negotiation needed
event.
this event is causing us to loop in negotiation needed when each side
try to create one, gets the answer and so on...
* auto select the first audio input on connection success webrtc
* add a way to choose the audioouput/mute a webrtc call
* mute audio from bg when video is in fullscreen
this is to avoid having two times the remote sound playing
one in the bg and one in the fullscreen
* Adding improvements to message request handling.
* Only updating approval when it is a true value as we consider a block a decline.
* Linting and formatting.
* More formatting and linting
* fixing merge conflicts
* linting and formatting changes
* darken a bit the green of sent message box in light theme
* disable deduplication based serverId+sender
only use the serverTimestamp+sender for searching because
serverId+sender might have false positive
* Fixing up block all logic.
* speed up fetching closed group's members avatar
* Applying PR changes.
* cleanup props passing of avatar and name with a custom hook
* fix a bug releasing the decrypted attachment blobs too early
* Adding trigger logic for conversation filtering of requests.
* Fixing rimraf transpile bug. Adding PR fixes - icon buttons.
* Minor call tweaks (#2051)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* Fetch translations (#2056)
* show missed-call,started-call and answered call notification in chat
* fix types for createLastMessageUpdate
* show incoming dialog if we have a pending call when enable call receptio
* simplify a bit the avatar component
* move disableDrag to a custom hook
* speed up hash colors of avatarPlaceHolders
* fixup text selection and double click reply on message
* keep avatar decoded items longer before releasing memory
* add incoming/outgoing/missed call notification
also, merge that notification with the timer and group notification
component
* hangup call if no answer after 30sec
* refactor SessionInput using hook + add testid field for recovery
* disable message request feature flag for now
* fix merge issue
* force loading screen to be black instead of white
for our dark theme user's eyes safety
* add type for i18n to run update after crowdin fetch with tools/updateI18nKeysType.py
* update to latest translations
* Open group regex fixes (#2058)
* Open group URL regex fixes
- Capital letters in room tokens were not being accepted (it eventually
gets lower-cased internally, which works fine, but that happens
*after* the URL is tested for acceptability).
- `-` in room was not being allowed (it is and always has been on SOGS,
session-android, and session-ios).
- single-letter room ids are valid, but only 2+ letter ids were being
accepted.
- complete URL regex wasn't anchored so something like
`garbagehttps://example.com/room?public_key=<64hex>moregarbage` was
being accepted in the GUI input (it fails later when other code tries
to parse it as a URL).
- removed `m` modifier from open group regex: without anchors it wasn't
doing anything anyway, but *with* anchors it would still allow
leading/trailing garbage if delineated by newlines.
- public key regex was accepting g-z letters, and not accepting A-F.
- various regex cleanups:
- use non-capture groups (?:...) rather than capturing groups (...)
- avoid repetition in host segment matching
- tightened up host pattern matching a bit:
- DNS host segments have a max length of 63
- Limit port max length to 5, and disallow starting with 0
* Show an error when the open group URL is invalid
It's quite disconcerting when you have a bad open group URL and try to
add it and the join button just "doesn't work" without any feedback at
all. Fix it to show an error message. (There is already an i18n entry
for this because this same message is thrown if the URL can't be parsed
later on).
* Add call duration (#2059)
* add call duration once connected
* close incoming call dialog if endCall from same sender
* disable message request toggle if featureFlag is OFF
* Cleanup message request (#2063)
* close incoming call dialog if endCall from seame sender
* disable message request toggle if featureFlag is OFF
* cleanup UI of message requests
* mark all existing conversations as approved in a migration
* fix regex with conversationID for opengroups
* Various UI fixes (#2070)
* cleanup unused convo json fields in db
* display a toast if the user is not approved yet on call OFFER received
* enable CBR for calls
* do not update active_at on configMessage if !!active_at
* remove mkdirp dependency
* disable call button if focused convo is blocked
* quote: do not include the full body in quote, but just the first 100
* click on the edit profile qr code padding
* Allow longer input for opengroup join overlay
Fixes #2068
* Fix overlay feature for start new session button
* make ringing depend on redux CALL status
* turn ON read-receipt by default
* keep read-receipts disabled by default (#2071)
* refactor most of the components to outside of their Session folder (#2072)
* refactor most of the components to outside of their Session folder
* finish moving overlay and memberListItem to react hook
* fix bug with kicked member len >2 not being displayed
also sort admins first in UpdateGroupMembers dialog
* fix admin leaving text of groupNotification
* add a useFocusMount hook to focus input fields on mount
* make click avatar convo item open only user dialog
* cleanup config default.json
* make sure to use convoController to build sync message
* disable showing pubkey on opengroups
* add a pause on audio playback
Fixes #2079
* Minor styling fix for large amount of message requests (#2080)
* Minor styling fix for large amount of message requests
* Vertical center fix for message request banner.
* removing top margin from banner again.
* reactify group updates text bubble from redux store (#2083)
* add crown icon for closed group admins (#2084)
* disable call for now + fix left pane actions overflow (#2085)
* Fix attachment dl freeze (#2086)
* fix attachment download freezing app for some opengroups
* make registration page work with smaller height
* Unban UI (#2091)
* adding basic functionaliy for unbanning a user
* merge ban and unban user dialog in to one dialog
Co-authored-by: warrickct <warrickct@gmail.com>
* use React Provider for convoListItem (#2088)
this is to avoid passing down the prop to all the components
* fix closed group updates undefined on no names (#2092)
Co-authored-by: Warrick Corfe-Tan <warrickct@gmail.com>
Co-authored-by: Jason Rhinelander <jason@imaginary.ca>
Co-authored-by: Warrick <wcor690@aucklanduni.ac.nz>
3 years ago
|
|
|
// if we have a result, it means a specific user sent two messages either with the same serverTimestamp.
|
|
|
|
// no need to do anything else, those messages must be the same
|
|
|
|
// Note: this test is not based on which conversation the user sent the message
|
|
|
|
// but we consider that a user sending two messages with the same serverTimestamp is unlikely
|
|
|
|
return Boolean(result);
|
|
|
|
} else {
|
|
|
|
result = await getMessageBySender({
|
|
|
|
source,
|
|
|
|
sourceDevice,
|
|
|
|
sentAt: timestamp,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!result) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const filteredResult = [result].filter((m: any) => m.attributes.body === message.body);
|
|
|
|
return filteredResult.some(m => isDuplicate(m, message, source));
|
|
|
|
} catch (error) {
|
|
|
|
window?.log?.error('isMessageDuplicate error:', Errors.toLogFormat(error));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export const isDuplicate = (
|
|
|
|
m: MessageModel,
|
|
|
|
testedMessage: MessageDuplicateSearchType,
|
|
|
|
source: string
|
|
|
|
) => {
|
|
|
|
// The username in this case is the users pubKey
|
|
|
|
const sameUsername = m.attributes.source === source;
|
|
|
|
const sameText = m.attributes.body === testedMessage.body;
|
|
|
|
// Don't filter out messages that are too far apart from each other
|
|
|
|
const timestampsSimilar =
|
|
|
|
Math.abs(m.attributes.sent_at - testedMessage.timestamp) <=
|
|
|
|
PUBLICCHAT_MIN_TIME_BETWEEN_DUPLICATE_MESSAGES;
|
|
|
|
|
|
|
|
return sameUsername && sameText && timestampsSimilar;
|
|
|
|
};
|
|
|
|
|
|
|
|
async function handleProfileUpdate(
|
|
|
|
profileKeyBuffer: Uint8Array,
|
|
|
|
convoId: string,
|
|
|
|
isIncoming: boolean
|
|
|
|
) {
|
|
|
|
if (!isIncoming) {
|
|
|
|
// We update our own profileKey if it's different from what we have
|
|
|
|
const ourNumber = UserUtils.getOurPubKeyStrFromCache();
|
|
|
|
const me = getConversationController().getOrCreate(ourNumber, ConversationTypeEnum.PRIVATE);
|
|
|
|
|
|
|
|
// Will do the save for us if needed
|
|
|
|
await me.setProfileKey(profileKeyBuffer);
|
|
|
|
} else {
|
|
|
|
const sender = await getConversationController().getOrCreateAndWait(
|
|
|
|
convoId,
|
|
|
|
ConversationTypeEnum.PRIVATE
|
|
|
|
);
|
|
|
|
|
|
|
|
// Will do the save for us
|
|
|
|
await sender.setProfileKey(profileKeyBuffer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface MessageCreationData {
|
|
|
|
timestamp: number;
|
|
|
|
isPublic: boolean;
|
|
|
|
receivedAt: number;
|
|
|
|
sourceDevice: number; // always 1 isn't it?
|
|
|
|
source: string;
|
|
|
|
serverId: number;
|
|
|
|
message: any;
|
|
|
|
serverTimestamp: any;
|
|
|
|
|
|
|
|
// Needed for synced outgoing messages
|
|
|
|
expirationStartTimestamp: any; // ???
|
|
|
|
destination: string;
|
|
|
|
messageHash?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function initIncomingMessage(data: MessageCreationData): MessageModel {
|
|
|
|
const {
|
|
|
|
timestamp,
|
|
|
|
isPublic,
|
|
|
|
receivedAt,
|
|
|
|
sourceDevice,
|
|
|
|
source,
|
|
|
|
serverId,
|
|
|
|
message,
|
|
|
|
serverTimestamp,
|
|
|
|
messageHash,
|
|
|
|
} = data;
|
|
|
|
|
|
|
|
const messageGroupId = message?.group?.id;
|
|
|
|
let groupId = messageGroupId && messageGroupId.length > 0 ? messageGroupId : null;
|
|
|
|
|
|
|
|
if (groupId) {
|
|
|
|
groupId = PubKey.removeTextSecurePrefixIfNeeded(groupId);
|
|
|
|
}
|
|
|
|
|
|
|
|
const messageData: any = {
|
|
|
|
source,
|
|
|
|
sourceDevice,
|
|
|
|
serverId, // + (not present below in `createSentMessage`)
|
|
|
|
sent_at: timestamp,
|
|
|
|
serverTimestamp,
|
|
|
|
received_at: receivedAt || Date.now(),
|
|
|
|
conversationId: groupId ?? source,
|
|
|
|
type: 'incoming',
|
|
|
|
direction: 'incoming', // +
|
|
|
|
unread: 1, // +
|
|
|
|
isPublic, // +
|
|
|
|
messageHash: messageHash || null,
|
|
|
|
};
|
|
|
|
|
|
|
|
return new MessageModel(messageData);
|
|
|
|
}
|
|
|
|
|
|
|
|
function createSentMessage(data: MessageCreationData): MessageModel {
|
|
|
|
const now = Date.now();
|
|
|
|
|
|
|
|
const {
|
|
|
|
timestamp,
|
|
|
|
serverTimestamp,
|
|
|
|
serverId,
|
|
|
|
isPublic,
|
|
|
|
receivedAt,
|
|
|
|
sourceDevice,
|
|
|
|
expirationStartTimestamp,
|
|
|
|
destination,
|
|
|
|
message,
|
|
|
|
messageHash,
|
|
|
|
} = data;
|
|
|
|
|
|
|
|
const sentSpecificFields = {
|
|
|
|
sent_to: [],
|
|
|
|
sent: true,
|
|
|
|
expirationStartTimestamp: Math.min(expirationStartTimestamp || data.timestamp || now, now),
|
|
|
|
};
|
|
|
|
|
|
|
|
const messageGroupId = message?.group?.id;
|
|
|
|
let groupId = messageGroupId && messageGroupId.length > 0 ? messageGroupId : null;
|
|
|
|
|
|
|
|
if (groupId) {
|
|
|
|
groupId = PubKey.removeTextSecurePrefixIfNeeded(groupId);
|
|
|
|
}
|
|
|
|
|
|
|
|
const messageData = {
|
|
|
|
source: UserUtils.getOurPubKeyStrFromCache(),
|
|
|
|
sourceDevice,
|
|
|
|
serverTimestamp,
|
|
|
|
serverId,
|
|
|
|
sent_at: timestamp,
|
|
|
|
received_at: isPublic ? receivedAt : now,
|
|
|
|
isPublic,
|
|
|
|
conversationId: groupId ?? destination,
|
|
|
|
type: 'outgoing' as MessageModelType,
|
|
|
|
messageHash,
|
|
|
|
...sentSpecificFields,
|
|
|
|
};
|
|
|
|
|
|
|
|
return new MessageModel(messageData);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function createMessage(data: MessageCreationData, isIncoming: boolean): MessageModel {
|
|
|
|
if (isIncoming) {
|
|
|
|
return initIncomingMessage(data);
|
|
|
|
} else {
|
|
|
|
return createSentMessage(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface MessageEvent {
|
|
|
|
data: any;
|
|
|
|
type: string;
|
|
|
|
confirm: () => void;
|
|
|
|
}
|
|
|
|
|
|
|
|
// tslint:disable:cyclomatic-complexity max-func-body-length */
|
|
|
|
export async function handleMessageEvent(event: MessageEvent): Promise<void> {
|
|
|
|
const { data, confirm } = event;
|
|
|
|
|
|
|
|
const isIncoming = event.type === 'message';
|
|
|
|
|
|
|
|
if (!data || !data.message) {
|
|
|
|
window?.log?.warn('Invalid data passed to handleMessageEvent.', event);
|
|
|
|
confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const { message, destination, messageHash } = data;
|
|
|
|
|
|
|
|
let { source } = data;
|
|
|
|
|
|
|
|
const isGroupMessage = Boolean(message.group);
|
|
|
|
|
|
|
|
const type = isGroupMessage ? ConversationTypeEnum.GROUP : ConversationTypeEnum.PRIVATE;
|
|
|
|
|
|
|
|
let conversationId = isIncoming ? source : destination || source; // for synced message
|
|
|
|
if (!conversationId) {
|
|
|
|
window?.log?.error('We cannot handle a message without a conversationId');
|
|
|
|
confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (message.profileKey?.length) {
|
|
|
|
await handleProfileUpdate(message.profileKey, conversationId, isIncoming);
|
|
|
|
}
|
|
|
|
|
|
|
|
const msg = createMessage(data, isIncoming);
|
|
|
|
|
|
|
|
// if the message is `sent` (from secondary device) we have to set the sender manually... (at least for now)
|
|
|
|
source = source || msg.get('source');
|
|
|
|
|
|
|
|
// Conversation Id is:
|
|
|
|
// - primarySource if it is an incoming DM message,
|
|
|
|
// - destination if it is an outgoing message,
|
|
|
|
// - group.id if it is a group message
|
|
|
|
if (isGroupMessage) {
|
|
|
|
// remove the prefix from the source object so this is correct for all other
|
|
|
|
message.group.id = PubKey.removeTextSecurePrefixIfNeeded(message.group.id);
|
|
|
|
|
|
|
|
conversationId = message.group.id;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!conversationId) {
|
|
|
|
window?.log?.warn('Invalid conversation id for incoming message', conversationId);
|
|
|
|
}
|
|
|
|
const ourNumber = UserUtils.getOurPubKeyStrFromCache();
|
|
|
|
|
|
|
|
// =========================================
|
|
|
|
|
|
|
|
if (!isGroupMessage && source !== ourNumber) {
|
|
|
|
// Ignore auth from our devices
|
|
|
|
conversationId = source;
|
|
|
|
}
|
|
|
|
|
|
|
|
const conversation = await getConversationController().getOrCreateAndWait(conversationId, type);
|
|
|
|
|
|
|
|
if (!conversation) {
|
|
|
|
window?.log?.warn('Skipping handleJob for unknown convo: ', conversationId);
|
|
|
|
confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void conversation.queueJob(async () => {
|
|
|
|
if (await isMessageDuplicate(data)) {
|
|
|
|
window?.log?.info('Received duplicate message. Dropping it.');
|
|
|
|
confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
await handleMessageJob(msg, conversation, message, ourNumber, confirm, source, messageHash);
|
|
|
|
});
|
|
|
|
}
|