From a1f0afdae82dec79e89876bdce61096b5e5545c4 Mon Sep 17 00:00:00 2001 From: Scott Nonnenberg Date: Fri, 3 May 2024 07:28:36 -0700 Subject: [PATCH] Fix rendering of group joins and remove checkServiceIdEquivalence --- .../conversation/GroupV2Change.stories.tsx | 42 ++++++++++--- ts/components/conversation/GroupV2Change.tsx | 6 -- .../conversation/Timeline.stories.tsx | 1 - .../conversation/TimelineItem.stories.tsx | 1 - ts/components/conversation/TimelineItem.tsx | 5 -- ts/groupChange.ts | 14 ++--- ts/groups.ts | 2 + ts/services/backups/export.ts | 8 ++- ts/state/selectors/conversations.ts | 18 ------ ts/state/smart/TimelineItem.tsx | 8 +-- .../backup_groupv2_notifications_test.ts | 62 ++++++++++++++++++- ts/util/getNotificationDataForMessage.ts | 6 -- 12 files changed, 108 insertions(+), 65 deletions(-) diff --git a/ts/components/conversation/GroupV2Change.stories.tsx b/ts/components/conversation/GroupV2Change.stories.tsx index 387b16906ff7..be47418951fb 100644 --- a/ts/components/conversation/GroupV2Change.stories.tsx +++ b/ts/components/conversation/GroupV2Change.stories.tsx @@ -50,13 +50,6 @@ const renderContact: SmartContactRendererType = ( ); -function checkServiceIdEquivalence( - left: ServiceIdString | undefined, - right: ServiceIdString | undefined -): boolean { - return Boolean(left && right && contactMap[left] === contactMap[right]); -} - const renderChange = ( change: GroupV2ChangeType, { @@ -79,7 +72,6 @@ const renderChange = ( blockGroupLinkRequests={action('blockGroupLinkRequests')} conversationId="some-conversation-id" change={change} - checkServiceIdEquivalence={checkServiceIdEquivalence} groupBannedMemberships={groupBannedMemberships} groupMemberships={groupMemberships} groupName={groupName} @@ -615,7 +607,39 @@ export function MemberAddFromInvited(): JSX.Element { }, ], })} - ACI accepts PNI invite: + ACI accepts PNI invite (X joined the group) + {renderChange({ + from: OUR_PNI, + details: [ + { + type: 'member-add-from-invite', + aci: OUR_ACI, + pni: OUR_PNI, + inviter: CONTACT_B, + }, + ], + })} + {renderChange({ + from: OUR_PNI, + details: [ + { + type: 'member-add-from-invite', + aci: OUR_ACI, + pni: OUR_PNI, + }, + ], + })} + {renderChange({ + from: CONTACT_A_PNI, + details: [ + { + type: 'member-add-from-invite', + aci: CONTACT_A, + pni: CONTACT_A_PNI, + }, + ], + })} + ACI accepts PNI invite, the old way (X added X to group) {renderChange({ from: OUR_PNI, details: [ diff --git a/ts/components/conversation/GroupV2Change.tsx b/ts/components/conversation/GroupV2Change.tsx index 2a2888c4ab04..d7221a2a391d 100644 --- a/ts/components/conversation/GroupV2Change.tsx +++ b/ts/components/conversation/GroupV2Change.tsx @@ -49,10 +49,6 @@ export type PropsActionsType = { }; export type PropsHousekeepingType = { - checkServiceIdEquivalence( - left: ServiceIdString | undefined, - right: ServiceIdString | undefined - ): boolean; i18n: LocalizerType; renderContact: SmartContactRendererType; }; @@ -297,7 +293,6 @@ export function GroupV2Change(props: PropsType): ReactElement { areWeAdmin, blockGroupLinkRequests, change, - checkServiceIdEquivalence, conversationId, groupBannedMemberships, groupMemberships, @@ -311,7 +306,6 @@ export function GroupV2Change(props: PropsType): ReactElement { return ( <> {renderChange(change, { - checkServiceIdEquivalence, i18n, ourAci, ourPni, diff --git a/ts/components/conversation/Timeline.stories.tsx b/ts/components/conversation/Timeline.stories.tsx index ec08bd9afcb2..316ad4977f09 100644 --- a/ts/components/conversation/Timeline.stories.tsx +++ b/ts/components/conversation/Timeline.stories.tsx @@ -361,7 +361,6 @@ const renderItem = ({ isNextItemCallingNotification={false} theme={ThemeType.light} platform="darwin" - checkServiceIdEquivalence={() => false} containerElementRef={containerElementRef} containerWidthBreakpoint={containerWidthBreakpoint} conversationId="" diff --git a/ts/components/conversation/TimelineItem.stories.tsx b/ts/components/conversation/TimelineItem.stories.tsx index 97458232c741..1e3828c6b9d7 100644 --- a/ts/components/conversation/TimelineItem.stories.tsx +++ b/ts/components/conversation/TimelineItem.stories.tsx @@ -69,7 +69,6 @@ const getDefaultProps = () => ({ toggleSelectMessage: action('toggleSelectMessage'), reactToMessage: action('reactToMessage'), checkForAccount: action('checkForAccount'), - checkServiceIdEquivalence: () => false, clearTargetedMessage: action('clearTargetedMessage'), setMessageToEdit: action('setMessageToEdit'), setQuoteByMessageId: action('setQuoteByMessageId'), diff --git a/ts/components/conversation/TimelineItem.tsx b/ts/components/conversation/TimelineItem.tsx index d72812c0c852..4883a8f13e61 100644 --- a/ts/components/conversation/TimelineItem.tsx +++ b/ts/components/conversation/TimelineItem.tsx @@ -61,7 +61,6 @@ import { type MessageRequestResponseNotificationData, } from './MessageRequestResponseNotification'; import type { MessageRequestState } from './MessageRequestActionsConfirmation'; -import type { ServiceIdString } from '../../types/ServiceId'; type CallHistoryType = { type: 'callHistory'; @@ -173,10 +172,6 @@ export type TimelineItemType = ( ) & { timestamp: number }; type PropsLocalType = { - checkServiceIdEquivalence( - left: ServiceIdString | undefined, - right: ServiceIdString | undefined - ): boolean; containerElementRef: RefObject; conversationId: string; item?: TimelineItemType; diff --git a/ts/groupChange.ts b/ts/groupChange.ts index 35a304a5732d..4b4c1fdd68b0 100644 --- a/ts/groupChange.ts +++ b/ts/groupChange.ts @@ -37,10 +37,6 @@ export type RenderOptionsType = { ourAci: AciString | undefined; ourPni: PniString | undefined; renderContact: SmartContactRendererType; - checkServiceIdEquivalence( - left: ServiceIdString | undefined, - right: ServiceIdString | undefined - ): boolean; renderIntl: StringRendererType; }; @@ -87,7 +83,6 @@ function renderChangeDetail( options: RenderOptionsType ): string | T | ReadonlyArray { const { - checkServiceIdEquivalence, from, i18n: localizer, ourAci, @@ -301,11 +296,14 @@ function renderChangeDetail( }); } if (detail.type === 'member-add-from-invite') { - const { aci, inviter } = detail; + const { aci, inviter, pni } = detail; const weAreJoiner = isOurServiceId(aci); const weAreInviter = isOurServiceId(inviter); - if (!from || !checkServiceIdEquivalence(from, aci)) { + const fromPni = pni && from === pni; + const fromAci = from === aci; + + if (!from || (!fromPni && !fromAci)) { if (weAreJoiner) { // They can't be the same, no fromYou check here if (from) { @@ -433,7 +431,7 @@ function renderChangeDetail( memberName: renderContact(aci), }); } - if (from && fromYou) { + if (from && from === aci) { return i18n('icu:GroupV2--member-remove--other--self', { memberName: renderContact(from), }); diff --git a/ts/groups.ts b/ts/groups.ts index 8dd76cc5cd73..64f43cffa280 100644 --- a/ts/groups.ts +++ b/ts/groups.ts @@ -149,6 +149,7 @@ type GroupV2MemberAddChangeType = { type GroupV2MemberAddFromInviteChangeType = { type: 'member-add-from-invite'; aci: AciString; + pni?: PniString; inviter?: AciString; }; type GroupV2MemberAddFromLinkChangeType = { @@ -4430,6 +4431,7 @@ function extractDiffs({ details.push({ type: 'member-add-from-invite', aci, + pni, inviter: pendingMember.addedByUserId, }); } else if (currentMember.joinedFromLink) { diff --git a/ts/services/backups/export.ts b/ts/services/backups/export.ts index badcafe886a8..216892ba625b 100644 --- a/ts/services/backups/export.ts +++ b/ts/services/backups/export.ts @@ -1253,7 +1253,13 @@ export class BackupExportStream extends Readable { update.groupMemberAddedUpdate = innerUpdate; updates.push(update); } else if (type === 'member-add-from-invite') { - if (from && checkServiceIdEquivalence(from, detail.aci)) { + const { aci, pni } = detail; + if ( + from && + ((pni && from === pni) || + (aci && from === aci) || + checkServiceIdEquivalence(from, aci)) + ) { const innerUpdate = new Backups.GroupInvitationAcceptedUpdate(); innerUpdate.newMemberAci = this.aciToBytes(detail.aci); if (detail.inviter) { diff --git a/ts/state/selectors/conversations.ts b/ts/state/selectors/conversations.ts index 4eaaa5857c5f..153ce5a7a91b 100644 --- a/ts/state/selectors/conversations.ts +++ b/ts/state/selectors/conversations.ts @@ -931,24 +931,6 @@ export const getConversationSelector = createSelector( } ); -export type CheckServiceIdEquivalenceType = ( - left: ServiceIdString | undefined, - right: ServiceIdString | undefined -) => boolean; -export const getCheckServiceIdEquivalence = createSelector( - getConversationByAnyIdSelector, - ( - getById: GetConversationByAnyIdSelectorType - ): CheckServiceIdEquivalenceType => { - return ( - left: ServiceIdString | undefined, - right: ServiceIdString | undefined - ): boolean => { - return Boolean(left && right && getById(left) === getById(right)); - }; - } -); - export const getConversationByIdSelector = createSelector( getConversationLookup, conversationLookup => diff --git a/ts/state/smart/TimelineItem.tsx b/ts/state/smart/TimelineItem.tsx index 6010a1477503..4fc2534a7900 100644 --- a/ts/state/smart/TimelineItem.tsx +++ b/ts/state/smart/TimelineItem.tsx @@ -21,10 +21,7 @@ import { getTheme, getPlatform, } from '../selectors/user'; -import { - getTargetedMessage, - getCheckServiceIdEquivalence, -} from '../selectors/conversations'; +import { getTargetedMessage } from '../selectors/conversations'; import { useTimelineItem } from '../selectors/timeline'; import { areMessagesInSameGroup, @@ -89,8 +86,6 @@ export const SmartTimelineItem = memo(function SmartTimelineItem( const isTargeted = Boolean( targetedMessage && messageId === targetedMessage.id ); - const checkServiceIdEquivalence = useSelector(getCheckServiceIdEquivalence); - const isNextItemCallingNotification = nextItem?.type === 'callHistory'; const shouldCollapseAbove = areMessagesInSameGroup( @@ -180,7 +175,6 @@ export const SmartTimelineItem = memo(function SmartTimelineItem( { }, ], }), - // ACI accepts PNI invite: + // ACI accepts PNI invite (X joined the group) + // These don't roundtrip; the PNI from is replaced with ACI. See next test below. + // createMessage({ + // from: OUR_PNI, + // details: [ + // { + // type: 'member-add-from-invite', + // aci: OUR_ACI, + // pni: OUR_PNI, + // inviter: CONTACT_B, + // }, + // ], + // }), + // createMessage({ + // from: OUR_PNI, + // details: [ + // { + // type: 'member-add-from-invite', + // aci: OUR_ACI, + // pni: OUR_PNI, + // }, + // ], + // }), + // createMessage({ + // from: CONTACT_A_PNI, + // details: [ + // { + // type: 'member-add-from-invite', + // aci: CONTACT_A, + // pni: CONTACT_A_PNI, + // }, + // ], + // }), + // ACI accepts PNI invite, the old way (X added X to group) // These don't roundtrip; the PNI is replaced with ACI. See next test below. // createMessage({ // from: OUR_PNI, @@ -813,8 +846,31 @@ describe('backup/groupv2/notifications', () => { { disableIncrement: true } ); - const before = [firstBefore, secondBefore, thirdBefore]; - const after = [firstAfter, secondAfter, thirdAfter]; + const fourthBefore = createMessage({ + from: CONTACT_A_PNI, + details: [ + { + type: 'member-add-from-invite', + aci: CONTACT_A, + pni: CONTACT_A_PNI, + }, + ], + }); + const fourthAfter = createMessage( + { + from: CONTACT_A, + details: [ + { + type: 'member-add-from-invite', + aci: CONTACT_A, + }, + ], + }, + { disableIncrement: true } + ); + + const before = [firstBefore, secondBefore, thirdBefore, fourthBefore]; + const after = [firstAfter, secondAfter, thirdAfter, fourthAfter]; await asymmetricRoundtripHarness(before, after); }); diff --git a/ts/util/getNotificationDataForMessage.ts b/ts/util/getNotificationDataForMessage.ts index b463bdfbcf68..e6f5416045a3 100644 --- a/ts/util/getNotificationDataForMessage.ts +++ b/ts/util/getNotificationDataForMessage.ts @@ -143,12 +143,6 @@ export function getNotificationDataForMessage( ); const changes = GroupChange.renderChange(change, { - checkServiceIdEquivalence: (left, right) => { - return ( - window.ConversationController.get(left) === - window.ConversationController.get(right) - ); - }, i18n: window.i18n, ourAci: window.textsecure.storage.user.getCheckedAci(), ourPni: window.textsecure.storage.user.getCheckedPni(),