Deprecate phone number discovery notification

This commit is contained in:
Fedor Indutny 2023-01-12 14:18:08 -08:00 committed by GitHub
parent 63509b8965
commit d7b09b9703
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
21 changed files with 196 additions and 517 deletions

View file

@ -1235,9 +1235,13 @@
"messageformat": "{obsoleteConversationTitle} and {conversationTitle} are the same account. Your message history for both chats are here.", "messageformat": "{obsoleteConversationTitle} and {conversationTitle} are the same account. Your message history for both chats are here.",
"description": "Shown when we've discovered that two local conversations are the same remote account in an unusual way" "description": "Shown when we've discovered that two local conversations are the same remote account in an unusual way"
}, },
"icu:ConversationMerge--notification--no-e164": { "icu:ConversationMerge--notification--with-e164": {
"messageformat": "Your message history with {conversationTitle} and their number {obsoleteConversationNumber} has been merged.",
"description": "Shown when we've discovered that two local conversations are the same remote account in an unusual way, but we have the phone number for the old conversation"
},
"icu:ConversationMerge--notification--no-title": {
"messageformat": "Your message history with {conversationTitle} and another chat that belonged to them has been merged.", "messageformat": "Your message history with {conversationTitle} and another chat that belonged to them has been merged.",
"description": "Shown when we've discovered that two local conversations are the same remote account in an unusual way, but we don't have the phone number for the old conversation" "description": "Shown when we've discovered that two local conversations are the same remote account in an unusual way, but we don't have the title for the old conversation"
}, },
"icu:ConversationMerge--learn-more": { "icu:ConversationMerge--learn-more": {
"messageformat": "Learn More", "messageformat": "Learn More",
@ -1253,11 +1257,11 @@
}, },
"icu:PhoneNumberDiscovery--notification--withSharedGroup": { "icu:PhoneNumberDiscovery--notification--withSharedGroup": {
"messageformat": "{phoneNumber} belongs to {conversationTitle}. You're both members of {sharedGroup}.", "messageformat": "{phoneNumber} belongs to {conversationTitle}. You're both members of {sharedGroup}.",
"description": "Shown when we've discovered a phone number for a contact you've been communicating with." "description": "(deleted 2023/01/11) Shown when we've discovered a phone number for a contact you've been communicating with."
}, },
"icu:PhoneNumberDiscovery--notification--noSharedGroup": { "icu:PhoneNumberDiscovery--notification--noSharedGroup": {
"messageformat": "{phoneNumber} belongs to {conversationTitle}", "messageformat": "{phoneNumber} belongs to {conversationTitle}",
"description": "Shown when we've discovered a phone number for a contact you've been communicating with, but you have no shared groups." "description": "(deleted 2023/01/11) Shown when we've discovered a phone number for a contact you've been communicating with, but you have no shared groups."
}, },
"quoteThumbnailAlt": { "quoteThumbnailAlt": {
"message": "Thumbnail of image from quoted message", "message": "Thumbnail of image from quoted message",

View file

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="none"><path fill="#000" fill-rule="evenodd" d="M7.6 2.533a.5.5 0 0 1 .8 0l2 2.667a.5.5 0 0 1-.4.8H8.75v.028c0 1.424.01 2.813.457 4.075.432 1.22 1.29 2.361 3.112 3.218a.75.75 0 1 1-.638 1.358C9.74 13.765 8.622 12.518 8 11.122c-.622 1.396-1.74 2.643-3.68 3.557a.75.75 0 1 1-.64-1.358c1.822-.857 2.681-1.998 3.113-3.218.447-1.262.457-2.65.457-4.075V6H6a.5.5 0 0 1-.4-.8l2-2.667Z" clip-rule="evenodd"/></svg>

After

Width:  |  Height:  |  Size: 473 B

View file

@ -281,6 +281,13 @@
'../images/icons/v2/credit-card-16.svg' '../images/icons/v2/credit-card-16.svg'
); );
} }
&--icon-merge::before {
@include system-message-icon(
'../images/icons/v2/merge-16.svg',
'../images/icons/v2/merge-16.svg'
);
}
} }
&--error { &--error {

View file

@ -49,8 +49,7 @@ function applyChangeToConversation(
conversation: ConversationModel, conversation: ConversationModel,
suggestedChange: Partial< suggestedChange: Partial<
Pick<ConversationAttributesType, 'uuid' | 'e164' | 'pni'> Pick<ConversationAttributesType, 'uuid' | 'e164' | 'pni'>
>, >
disableDiscoveryNotification?: boolean
) { ) {
const change = { ...suggestedChange }; const change = { ...suggestedChange };
@ -84,9 +83,7 @@ function applyChangeToConversation(
conversation.updateUuid(change.uuid); conversation.updateUuid(change.uuid);
} }
if (hasOwnProperty.call(change, 'e164')) { if (hasOwnProperty.call(change, 'e164')) {
conversation.updateE164(change.e164, { conversation.updateE164(change.e164);
disableDiscoveryNotification,
});
} }
if (hasOwnProperty.call(change, 'pni')) { if (hasOwnProperty.call(change, 'pni')) {
conversation.updatePni(change.pni); conversation.updatePni(change.pni);
@ -485,7 +482,6 @@ export class ConversationController {
const aci = providedAci ? UUID.cast(providedAci) : undefined; const aci = providedAci ? UUID.cast(providedAci) : undefined;
const pni = providedPni ? UUID.cast(providedPni) : undefined; const pni = providedPni ? UUID.cast(providedPni) : undefined;
let targetConversationWasCreated = false;
const mergePromises: Array<Promise<void>> = []; const mergePromises: Array<Promise<void>> = [];
if (!aci && !e164 && !pni) { if (!aci && !e164 && !pni) {
@ -524,13 +520,9 @@ export class ConversationController {
`${logId}: No match for ${key}, applying to target conversation` `${logId}: No match for ${key}, applying to target conversation`
); );
// Note: This line might erase a known e164 or PNI // Note: This line might erase a known e164 or PNI
applyChangeToConversation( applyChangeToConversation(targetConversation, {
targetConversation, [key]: value,
{ });
[key]: value,
},
targetConversationWasCreated
);
} else { } else {
unusedMatches.push(item); unusedMatches.push(item);
} }
@ -578,7 +570,6 @@ export class ConversationController {
// If PNI match already has an ACI, then we need to create a new one // If PNI match already has an ACI, then we need to create a new one
if (!targetConversation) { if (!targetConversation) {
targetConversation = this.getOrCreate(unused.value, 'private'); targetConversation = this.getOrCreate(unused.value, 'private');
targetConversationWasCreated = true;
log.info( log.info(
`${logId}: Match on ${key} already had ${unused.key}, ` + `${logId}: Match on ${key} already had ${unused.key}, ` +
`so created new target conversation - ${targetConversation.idForLogging()}` `so created new target conversation - ${targetConversation.idForLogging()}`
@ -588,13 +579,9 @@ export class ConversationController {
log.info( log.info(
`${logId}: Applying new value for ${unused.key} to target conversation` `${logId}: Applying new value for ${unused.key} to target conversation`
); );
applyChangeToConversation( applyChangeToConversation(targetConversation, {
targetConversation, [unused.key]: unused.value,
{ });
[unused.key]: unused.value,
},
targetConversationWasCreated
);
}); });
unusedMatches = []; unusedMatches = [];
@ -633,20 +620,16 @@ export class ConversationController {
if ((key === 'pni' || key === 'e164') && match.get('uuid') === pni) { if ((key === 'pni' || key === 'e164') && match.get('uuid') === pni) {
change.uuid = undefined; change.uuid = undefined;
} }
applyChangeToConversation(match, change, targetConversationWasCreated); applyChangeToConversation(match, change);
// Note: The PNI check here is just to be bulletproof; if we know a UUID is a PNI, // Note: The PNI check here is just to be bulletproof; if we know a UUID is a PNI,
// then that should be put in the UUID field as well! // then that should be put in the UUID field as well!
const willMerge = const willMerge =
!match.get('uuid') && !match.get('e164') && !match.get('pni'); !match.get('uuid') && !match.get('e164') && !match.get('pni');
applyChangeToConversation( applyChangeToConversation(targetConversation, {
targetConversation, [key]: value,
{ });
[key]: value,
},
willMerge || targetConversationWasCreated
);
if (willMerge) { if (willMerge) {
log.warn( log.warn(
@ -666,13 +649,9 @@ export class ConversationController {
} else if (targetConversation && !targetConversation?.get(key)) { } else if (targetConversation && !targetConversation?.get(key)) {
// This is mostly for the situation where PNI was erased when updating e164 // This is mostly for the situation where PNI was erased when updating e164
log.debug(`${logId}: Re-adding ${key} on target conversation`); log.debug(`${logId}: Re-adding ${key} on target conversation`);
applyChangeToConversation( applyChangeToConversation(targetConversation, {
targetConversation, [key]: value,
{ });
[key]: value,
},
targetConversationWasCreated
);
} }
if (!targetConversation) { if (!targetConversation) {
@ -739,7 +718,7 @@ export class ConversationController {
// `identifier` would resolve to uuid if we had both, so fix up e164 // `identifier` would resolve to uuid if we had both, so fix up e164
if (normalizedUuid && e164) { if (normalizedUuid && e164) {
newConvo.updateE164(e164, { disableDiscoveryNotification: true }); newConvo.updateE164(e164);
} }
return newConvo; return newConvo;
@ -1131,7 +1110,7 @@ export class ConversationController {
const titleIsUseful = Boolean( const titleIsUseful = Boolean(
obsoleteTitleInfo && getTitleNoDefault(obsoleteTitleInfo) obsoleteTitleInfo && getTitleNoDefault(obsoleteTitleInfo)
); );
if (!fromPniSignature && obsoleteTitleInfo && titleIsUseful) { if (obsoleteTitleInfo && titleIsUseful && !fromPniSignature) {
drop(current.addConversationMerge(obsoleteTitleInfo)); drop(current.addConversationMerge(obsoleteTitleInfo));
} }

View file

@ -338,6 +338,7 @@ export function ConversationList({
'typingContactId', 'typingContactId',
'unblurredAvatarPath', 'unblurredAvatarPath',
'unreadCount', 'unreadCount',
'uuid',
]); ]);
const { badges, title, unreadCount, lastMessage } = itemProps; const { badges, title, unreadCount, lastMessage } = itemProps;
result = ( result = (

View file

@ -17,14 +17,25 @@ export default {
const createProps = (overrideProps: Partial<PropsType> = {}): PropsType => ({ const createProps = (overrideProps: Partial<PropsType> = {}): PropsType => ({
i18n, i18n,
conversationTitle: overrideProps.conversationTitle || 'John Fire', conversationTitle: overrideProps.conversationTitle || 'John Fire',
obsoleteConversationNumber:
overrideProps.obsoleteConversationNumber || '(555) 333-1111',
obsoleteConversationTitle: obsoleteConversationTitle:
overrideProps.obsoleteConversationTitle || '(555) 333-1111', overrideProps.obsoleteConversationTitle || 'John Obsolete',
}); });
export function Basic(): JSX.Element { export function Basic(): JSX.Element {
return <ConversationMergeNotification {...createProps()} />; return <ConversationMergeNotification {...createProps()} />;
} }
export function WithNoObsoleteNumber(): JSX.Element {
return (
<ConversationMergeNotification
{...createProps()}
obsoleteConversationNumber={undefined}
/>
);
}
export function WithNoObsoleteTitle(): JSX.Element { export function WithNoObsoleteTitle(): JSX.Element {
return ( return (
<ConversationMergeNotification <ConversationMergeNotification

View file

@ -14,16 +14,23 @@ import { Intl } from '../Intl';
export type PropsDataType = { export type PropsDataType = {
conversationTitle: string; conversationTitle: string;
obsoleteConversationTitle: string | undefined; obsoleteConversationTitle: string | undefined;
obsoleteConversationNumber: string | undefined;
}; };
export type PropsType = PropsDataType & { export type PropsType = PropsDataType & {
i18n: LocalizerType; i18n: LocalizerType;
}; };
export function ConversationMergeNotification(props: PropsType): JSX.Element { export function ConversationMergeNotification(props: PropsType): JSX.Element {
const { conversationTitle, obsoleteConversationTitle, i18n } = props; const {
conversationTitle,
obsoleteConversationTitle,
obsoleteConversationNumber,
i18n,
} = props;
const message = getStringForConversationMerge({ const message = getStringForConversationMerge({
conversationTitle, conversationTitle,
obsoleteConversationTitle, obsoleteConversationTitle,
obsoleteConversationNumber,
i18n, i18n,
}); });
@ -40,7 +47,7 @@ export function ConversationMergeNotification(props: PropsType): JSX.Element {
return ( return (
<> <>
<SystemMessage <SystemMessage
icon="profile" icon="merge"
contents={<Emojify text={message} />} contents={<Emojify text={message} />}
button={ button={
obsoleteConversationTitle ? ( obsoleteConversationTitle ? (

View file

@ -1,36 +0,0 @@
// Copyright 2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import * as React from 'react';
import { setupI18n } from '../../util/setupI18n';
import enMessages from '../../../_locales/en/messages.json';
import type { PropsType } from './PhoneNumberDiscoveryNotification';
import { PhoneNumberDiscoveryNotification } from './PhoneNumberDiscoveryNotification';
const i18n = setupI18n('en', enMessages);
export default {
title: 'Components/Conversation/PhoneNumberDiscoveryNotification',
};
const createProps = (overrideProps: Partial<PropsType> = {}): PropsType => ({
i18n,
conversationTitle: overrideProps.conversationTitle || 'Mr. Fire',
phoneNumber: overrideProps.phoneNumber || '+1 (000) 123-4567',
sharedGroup: overrideProps.sharedGroup,
});
export function Basic(): JSX.Element {
return <PhoneNumberDiscoveryNotification {...createProps()} />;
}
export function WithSharedGroup(): JSX.Element {
return (
<PhoneNumberDiscoveryNotification
{...createProps({
sharedGroup: 'Animal Lovers',
})}
/>
);
}

View file

@ -1,32 +0,0 @@
// Copyright 2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import React from 'react';
import type { LocalizerType } from '../../types/Util';
import { SystemMessage } from './SystemMessage';
import { Emojify } from './Emojify';
import { getStringForPhoneNumberDiscovery } from '../../util/getStringForPhoneNumberDiscovery';
export type PropsDataType = {
conversationTitle: string;
phoneNumber: string;
sharedGroup?: string;
};
export type PropsType = PropsDataType & {
i18n: LocalizerType;
};
export function PhoneNumberDiscoveryNotification(
props: PropsType
): JSX.Element {
const { conversationTitle, i18n, sharedGroup, phoneNumber } = props;
const message = getStringForPhoneNumberDiscovery({
conversationTitle,
i18n,
phoneNumber,
sharedGroup,
});
return <SystemMessage icon="profile" contents={<Emojify text={message} />} />;
}

View file

@ -50,8 +50,6 @@ import type { PropsType as PaymentEventNotificationPropsType } from './PaymentEv
import { PaymentEventNotification } from './PaymentEventNotification'; import { PaymentEventNotification } from './PaymentEventNotification';
import type { PropsDataType as ConversationMergeNotificationPropsType } from './ConversationMergeNotification'; import type { PropsDataType as ConversationMergeNotificationPropsType } from './ConversationMergeNotification';
import { ConversationMergeNotification } from './ConversationMergeNotification'; import { ConversationMergeNotification } from './ConversationMergeNotification';
import type { PropsDataType as PhoneNumberDiscoveryNotificationPropsType } from './PhoneNumberDiscoveryNotification';
import { PhoneNumberDiscoveryNotification } from './PhoneNumberDiscoveryNotification';
import type { FullJSXType } from '../Intl'; import type { FullJSXType } from '../Intl';
import { TimelineMessage } from './TimelineMessage'; import { TimelineMessage } from './TimelineMessage';
@ -119,10 +117,6 @@ type ConversationMergeNotificationType = {
type: 'conversationMerge'; type: 'conversationMerge';
data: ConversationMergeNotificationPropsType; data: ConversationMergeNotificationPropsType;
}; };
type PhoneNumberDiscoveryNotificationType = {
type: 'phoneNumberDiscovery';
data: PhoneNumberDiscoveryNotificationPropsType;
};
type PaymentEventType = { type PaymentEventType = {
type: 'paymentEvent'; type: 'paymentEvent';
data: Omit<PaymentEventNotificationPropsType, 'i18n'>; data: Omit<PaymentEventNotificationPropsType, 'i18n'>;
@ -138,7 +132,6 @@ export type TimelineItemType = (
| GroupV1MigrationType | GroupV1MigrationType
| GroupV2ChangeType | GroupV2ChangeType
| MessageType | MessageType
| PhoneNumberDiscoveryNotificationType
| ProfileChangeNotificationType | ProfileChangeNotificationType
| ResetSessionNotificationType | ResetSessionNotificationType
| SafetyNumberNotificationType | SafetyNumberNotificationType
@ -319,14 +312,6 @@ export class TimelineItem extends React.PureComponent<PropsType> {
i18n={i18n} i18n={i18n}
/> />
); );
} else if (item.type === 'phoneNumberDiscovery') {
notification = (
<PhoneNumberDiscoveryNotification
{...reducedProps}
{...item.data}
i18n={i18n}
/>
);
} else if (item.type === 'resetSessionNotification') { } else if (item.type === 'resetSessionNotification') {
notification = ( notification = (
<ResetSessionNotification {...reducedProps} i18n={i18n} /> <ResetSessionNotification {...reducedProps} i18n={i18n} />

4
ts/model-types.d.ts vendored
View file

@ -176,7 +176,6 @@ export type MessageAttributesType = {
| 'incoming' | 'incoming'
| 'keychange' | 'keychange'
| 'outgoing' | 'outgoing'
| 'phone-number-discovery'
| 'profile-change' | 'profile-change'
| 'story' | 'story'
| 'timer-notification' | 'timer-notification'
@ -209,9 +208,6 @@ export type MessageAttributesType = {
source?: string; source?: string;
sourceUuid?: string; sourceUuid?: string;
}; };
phoneNumberDiscovery?: {
e164: string;
};
conversationMerge?: { conversationMerge?: {
renderInfo: ConversationRenderInfoType; renderInfo: ConversationRenderInfoType;
}; };

View file

@ -1912,14 +1912,7 @@ export class ConversationModel extends window.Backbone
}; };
} }
updateE164( updateE164(e164?: string | null): void {
e164?: string | null,
{
disableDiscoveryNotification,
}: {
disableDiscoveryNotification?: boolean;
} = {}
): void {
const oldValue = this.get('e164'); const oldValue = this.get('e164');
if (e164 === oldValue) { if (e164 === oldValue) {
return; return;
@ -1927,15 +1920,6 @@ export class ConversationModel extends window.Backbone
this.set('e164', e164 || undefined); this.set('e164', e164 || undefined);
// We just discovered a new phone number for this account. If we're not merging
// then we'll add a standalone notification here.
const haveSentMessage = Boolean(
this.get('profileSharing') || this.get('sentMessageCount')
);
if (!oldValue && e164 && haveSentMessage && !disableDiscoveryNotification) {
void this.addPhoneNumberDiscovery(e164);
}
// This user changed their phone number // This user changed their phone number
if (oldValue && e164) { if (oldValue && e164) {
void this.addChangeNumberNotification(oldValue, e164); void this.addChangeNumberNotification(oldValue, e164);
@ -3083,43 +3067,6 @@ export class ConversationModel extends window.Backbone
}); });
} }
async addPhoneNumberDiscovery(e164: string): Promise<void> {
log.info(
`addPhoneNumberDiscovery/${this.idForLogging()}: Adding for ${e164}`
);
const timestamp = Date.now();
const message: MessageAttributesType = {
id: generateGuid(),
conversationId: this.id,
type: 'phone-number-discovery',
sent_at: timestamp,
timestamp,
received_at: window.Signal.Util.incrementMessageCounter(),
received_at_ms: timestamp,
phoneNumberDiscovery: {
e164,
},
readStatus: ReadStatus.Read,
seenStatus: SeenStatus.Unseen,
schemaVersion: Message.VERSION_NEEDED_FOR_DISPLAY,
};
const id = await window.Signal.Data.saveMessage(message, {
ourUuid: window.textsecure.storage.user.getCheckedUuid().toString(),
forceSave: true,
});
const model = window.MessageController.register(
id,
new window.Whisper.Message({
...message,
id,
})
);
this.trigger('newmessage', model);
}
async addConversationMerge( async addConversationMerge(
renderInfo: ConversationRenderInfoType renderInfo: ConversationRenderInfoType
): Promise<void> { ): Promise<void> {
@ -4845,7 +4792,6 @@ export class ConversationModel extends window.Backbone
// same before/after string, even if someone is moving from just first name to // same before/after string, even if someone is moving from just first name to
// first/last name in their profile data. // first/last name in their profile data.
const nameChanged = oldName !== newName; const nameChanged = oldName !== newName;
if (!isMe(this.attributes) && hadPreviousName && nameChanged) { if (!isMe(this.attributes) && hadPreviousName && nameChanged) {
const change = { const change = {
type: 'name', type: 'name',
@ -4902,8 +4848,10 @@ export class ConversationModel extends window.Backbone
profileKey: string | undefined, profileKey: string | undefined,
{ viaStorageServiceSync = false } = {} { viaStorageServiceSync = false } = {}
): Promise<boolean> { ): Promise<boolean> {
const oldProfileKey = this.get('profileKey');
// profileKey is a string so we can compare it directly // profileKey is a string so we can compare it directly
if (this.get('profileKey') !== profileKey) { if (oldProfileKey !== profileKey) {
log.info( log.info(
`Setting sealedSender to UNKNOWN for conversation ${this.idForLogging()}` `Setting sealedSender to UNKNOWN for conversation ${this.idForLogging()}`
); );

View file

@ -115,7 +115,6 @@ import {
isVerifiedChange, isVerifiedChange,
processBodyRanges, processBodyRanges,
isConversationMerge, isConversationMerge,
isPhoneNumberDiscovery,
} from '../state/selectors/message'; } from '../state/selectors/message';
import { import {
isInCall, isInCall,
@ -174,8 +173,7 @@ import { GiftBadgeStates } from '../components/conversation/Message';
import { downloadAttachment } from '../util/downloadAttachment'; import { downloadAttachment } from '../util/downloadAttachment';
import type { StickerWithHydratedData } from '../types/Stickers'; import type { StickerWithHydratedData } from '../types/Stickers';
import { getStringForConversationMerge } from '../util/getStringForConversationMerge'; import { getStringForConversationMerge } from '../util/getStringForConversationMerge';
import { getStringForPhoneNumberDiscovery } from '../util/getStringForPhoneNumberDiscovery'; import { getTitleNoDefault, getNumber } from '../util/getTitle';
import { getTitle, renderNumber } from '../util/getTitle';
import dataInterface from '../sql/Client'; import dataInterface from '../sql/Client';
function isSameUuid( function isSameUuid(
@ -409,7 +407,6 @@ export class MessageModel extends window.Backbone.Model<MessageAttributesType> {
!isGroupV1Migration(attributes) && !isGroupV1Migration(attributes) &&
!isGroupV2Change(attributes) && !isGroupV2Change(attributes) &&
!isKeyChange(attributes) && !isKeyChange(attributes) &&
!isPhoneNumberDiscovery(attributes) &&
!isProfileChange(attributes) && !isProfileChange(attributes) &&
!isUniversalTimerNotification(attributes) && !isUniversalTimerNotification(attributes) &&
!isUnsupportedMessage(attributes) && !isUnsupportedMessage(attributes) &&
@ -499,7 +496,10 @@ export class MessageModel extends window.Backbone.Model<MessageAttributesType> {
return { return {
text: getStringForConversationMerge({ text: getStringForConversationMerge({
obsoleteConversationTitle: getTitle( obsoleteConversationTitle: getTitleNoDefault(
attributes.conversationMerge.renderInfo
),
obsoleteConversationNumber: getNumber(
attributes.conversationMerge.renderInfo attributes.conversationMerge.renderInfo
), ),
conversationTitle: conversation.getTitle(), conversationTitle: conversation.getTitle(),
@ -508,24 +508,6 @@ export class MessageModel extends window.Backbone.Model<MessageAttributesType> {
}; };
} }
if (isPhoneNumberDiscovery(attributes)) {
const conversation = this.getConversation();
strictAssert(conversation, 'getNotificationData/isPhoneNumberDiscovery');
strictAssert(
attributes.phoneNumberDiscovery,
'getNotificationData/isPhoneNumberDiscovery/phoneNumberDiscovery'
);
return {
text: getStringForPhoneNumberDiscovery({
phoneNumber: renderNumber(attributes.phoneNumberDiscovery.e164),
conversationTitle: conversation.getTitle(),
sharedGroup: conversation.get('sharedGroupNames')?.[0],
i18n: window.i18n,
}),
};
}
if (isChatSessionRefreshed(attributes)) { if (isChatSessionRefreshed(attributes)) {
return { return {
emoji: '🔁', emoji: '🔁',
@ -1219,7 +1201,6 @@ export class MessageModel extends window.Backbone.Model<MessageAttributesType> {
const isUniversalTimerNotificationValue = const isUniversalTimerNotificationValue =
isUniversalTimerNotification(attributes); isUniversalTimerNotification(attributes);
const isConversationMergeValue = isConversationMerge(attributes); const isConversationMergeValue = isConversationMerge(attributes);
const isPhoneNumberDiscoveryValue = isPhoneNumberDiscovery(attributes);
const isPayment = messageHasPaymentEvent(attributes); const isPayment = messageHasPaymentEvent(attributes);
@ -1251,8 +1232,7 @@ export class MessageModel extends window.Backbone.Model<MessageAttributesType> {
isKeyChangeValue || isKeyChangeValue ||
isProfileChangeValue || isProfileChangeValue ||
isUniversalTimerNotificationValue || isUniversalTimerNotificationValue ||
isConversationMergeValue || isConversationMergeValue;
isPhoneNumberDiscoveryValue;
return !hasSomethingToDisplay; return !hasSomethingToDisplay;
} }

View file

@ -0,0 +1,112 @@
// Copyright 2023 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { Database } from '@signalapp/better-sqlite3';
import type { LoggerType } from '../../types/Logging';
export default function updateToSchemaVersion73(
currentVersion: number,
db: Database,
logger: LoggerType
): void {
if (currentVersion >= 73) {
return;
}
db.transaction(() => {
db.exec(
`
--- Delete deprecated notifications
DELETE FROM messages WHERE type IS 'phone-number-discovery';
--- These will be re-added below
DROP INDEX messages_preview;
DROP INDEX messages_activity;
DROP INDEX message_user_initiated;
--- These will also be re-added below
ALTER TABLE messages DROP COLUMN shouldAffectActivity;
ALTER TABLE messages DROP COLUMN shouldAffectPreview;
ALTER TABLE messages DROP COLUMN isUserInitiatedMessage;
--- Note: These generated columns were originally introduced in migration 71, and
--- are mostly the same
--- (change: removed phone-number-discovery)
ALTER TABLE messages
ADD COLUMN shouldAffectActivity INTEGER
GENERATED ALWAYS AS (
type IS NULL
OR
type NOT IN (
'change-number-notification',
'conversation-merge',
'group-v1-migration',
'keychange',
'message-history-unsynced',
'profile-change',
'story',
'universal-timer-notification',
'verified-change'
)
);
--- (change: removed phone-number-discovery
--- (now matches the above list)
ALTER TABLE messages
ADD COLUMN shouldAffectPreview INTEGER
GENERATED ALWAYS AS (
type IS NULL
OR
type NOT IN (
'change-number-notification',
'conversation-merge',
'group-v1-migration',
'keychange',
'message-history-unsynced',
'profile-change',
'story',
'universal-timer-notification',
'verified-change'
)
);
--- Note: This list only differs from the above on these types:
--- group-v2-change
--- (change: removed phone-number-discovery
ALTER TABLE messages
ADD COLUMN isUserInitiatedMessage INTEGER
GENERATED ALWAYS AS (
type IS NULL
OR
type NOT IN (
'change-number-notification',
'conversation-merge',
'group-v1-migration',
'group-v2-change',
'keychange',
'message-history-unsynced',
'profile-change',
'story',
'universal-timer-notification',
'verified-change'
)
);
CREATE INDEX messages_preview ON messages
(conversationId, shouldAffectPreview, isGroupLeaveEventFromOther, expiresAt, received_at, sent_at);
CREATE INDEX messages_activity ON messages
(conversationId, shouldAffectActivity, isTimerChangeFromSync, isGroupLeaveEventFromOther, received_at, sent_at);
CREATE INDEX message_user_initiated ON messages (isUserInitiatedMessage);
`
);
db.pragma('user_version = 73');
})();
logger.info('updateToSchemaVersion73: success!');
}

View file

@ -48,6 +48,7 @@ import updateToSchemaVersion69 from './69-group-call-ring-cancellations';
import updateToSchemaVersion70 from './70-story-reply-index'; import updateToSchemaVersion70 from './70-story-reply-index';
import updateToSchemaVersion71 from './71-merge-notifications'; import updateToSchemaVersion71 from './71-merge-notifications';
import updateToSchemaVersion72 from './72-optimize-call-id-message-lookup'; import updateToSchemaVersion72 from './72-optimize-call-id-message-lookup';
import updateToSchemaVersion73 from './73-remove-phone-number-discovery';
function updateToSchemaVersion1( function updateToSchemaVersion1(
currentVersion: number, currentVersion: number,
@ -1965,6 +1966,7 @@ export const SCHEMA_VERSIONS = [
updateToSchemaVersion70, updateToSchemaVersion70,
updateToSchemaVersion71, updateToSchemaVersion71,
updateToSchemaVersion72, updateToSchemaVersion72,
updateToSchemaVersion73,
]; ];
export function updateSchema(db: Database, logger: LoggerType): void { export function updateSchema(db: Database, logger: LoggerType): void {

View file

@ -34,7 +34,6 @@ import type { PropsDataType as GroupV1MigrationPropsType } from '../../component
import type { PropsDataType as DeliveryIssuePropsType } from '../../components/conversation/DeliveryIssueNotification'; import type { PropsDataType as DeliveryIssuePropsType } from '../../components/conversation/DeliveryIssueNotification';
import type { PropsType as PaymentEventNotificationPropsType } from '../../components/conversation/PaymentEventNotification'; import type { PropsType as PaymentEventNotificationPropsType } from '../../components/conversation/PaymentEventNotification';
import type { PropsDataType as ConversationMergePropsType } from '../../components/conversation/ConversationMergeNotification'; import type { PropsDataType as ConversationMergePropsType } from '../../components/conversation/ConversationMergeNotification';
import type { PropsDataType as PhoneNumberDiscoveryPropsType } from '../../components/conversation/PhoneNumberDiscoveryNotification';
import type { import type {
PropsData as GroupNotificationProps, PropsData as GroupNotificationProps,
ChangeType, ChangeType,
@ -119,7 +118,7 @@ import { calculateExpirationTimestamp } from '../../util/expirationTimer';
import { isSignalConversation } from '../../util/isSignalConversation'; import { isSignalConversation } from '../../util/isSignalConversation';
import type { AnyPaymentEvent } from '../../types/Payment'; import type { AnyPaymentEvent } from '../../types/Payment';
import { isPaymentNotificationEvent } from '../../types/Payment'; import { isPaymentNotificationEvent } from '../../types/Payment';
import { getTitle, renderNumber } from '../../util/getTitle'; import { getTitleNoDefault, getNumber } from '../../util/getTitle';
export { isIncoming, isOutgoing, isStory }; export { isIncoming, isOutgoing, isStory };
@ -895,13 +894,6 @@ export function getPropsForBubble(
timestamp, timestamp,
}; };
} }
if (isPhoneNumberDiscovery(message)) {
return {
type: 'phoneNumberDiscovery',
data: getPhoneNumberDiscovery(message, options),
timestamp,
};
}
if ( if (
messageHasPaymentEvent(message) && messageHasPaymentEvent(message) &&
@ -1422,37 +1414,16 @@ export function getPropsForConversationMerge(
const conversation = getConversation(message, conversationSelector); const conversation = getConversation(message, conversationSelector);
const conversationTitle = conversation.title; const conversationTitle = conversation.title;
const { type, e164 } = conversationMerge.renderInfo; const { renderInfo } = conversationMerge;
const obsoleteConversationTitle = e164 ? getTitle({ type, e164 }) : undefined; const obsoleteConversationTitle = getTitleNoDefault(renderInfo);
const obsoleteConversationNumber = getNumber(renderInfo);
return { return {
conversationTitle, conversationTitle,
obsoleteConversationTitle, obsoleteConversationTitle,
obsoleteConversationNumber,
}; };
} }
export function isPhoneNumberDiscovery(
message: MessageWithUIFieldsType
): boolean {
return message.type === 'phone-number-discovery';
}
export function getPhoneNumberDiscovery(
message: MessageWithUIFieldsType,
{ conversationSelector }: GetPropsForBubbleOptions
): PhoneNumberDiscoveryPropsType {
const { phoneNumberDiscovery } = message;
if (!phoneNumberDiscovery) {
throw new Error(
'getPhoneNumberDiscovery: message is missing phoneNumberDiscovery!'
);
}
const conversation = getConversation(message, conversationSelector);
const conversationTitle = conversation.title;
const sharedGroup = conversation.sharedGroupNames[0];
const phoneNumber = renderNumber(phoneNumberDiscovery.e164);
return { conversationTitle, sharedGroup, phoneNumber };
}
// Delivery Issue // Delivery Issue

View file

@ -1,238 +0,0 @@
// Copyright 2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import { assert } from 'chai';
import { UUIDKind, Proto, StorageState } from '@signalapp/mock-server';
import type { PrimaryDevice } from '@signalapp/mock-server';
import createDebug from 'debug';
import * as durations from '../../util/durations';
import { Bootstrap } from '../bootstrap';
import type { App } from '../bootstrap';
export const debug = createDebug('mock:test:pni-signature');
describe('pnp/learn', function needsName() {
this.timeout(durations.MINUTE);
let bootstrap: Bootstrap;
let app: App;
let contactA: PrimaryDevice;
beforeEach(async () => {
bootstrap = new Bootstrap();
await bootstrap.init();
const { server, phone } = bootstrap;
contactA = await server.createPrimaryDevice({
profileName: 'contactA',
});
let state = StorageState.getEmpty();
state = state.updateAccount({
profileKey: phone.profileKey.serialize(),
e164: phone.device.number,
});
state = state.addContact(
contactA,
{
whitelisted: false,
identityKey: contactA.getPublicKey(UUIDKind.ACI).serialize(),
serviceE164: undefined,
givenName: 'ContactA',
},
UUIDKind.ACI
);
// Just to make PNI Contact visible in the left pane
state = state.pin(contactA, UUIDKind.ACI);
await phone.setStorageState(state);
app = await bootstrap.link();
});
afterEach(async function after() {
if (this.currentTest?.state !== 'passed') {
await bootstrap.saveLogs(app);
}
await app.close();
await bootstrap.teardown();
});
it('shows Learned Number notification if we find out number later', async () => {
const { desktop, phone } = bootstrap;
const window = await app.getWindow();
debug('Open conversation with contactA');
{
const leftPane = window.locator('.left-pane-wrapper');
await leftPane
.locator('_react=ConversationListItem[title = "ContactA"]')
.click();
await window.locator('.module-conversation-hero').waitFor();
}
debug('Verify starting state');
{
// No messages
const messages = window.locator('.module-message__text');
assert.strictEqual(await messages.count(), 0, 'message count');
// No notifications
const notifications = window.locator('.SystemMessage');
assert.strictEqual(await notifications.count(), 0, 'notification count');
}
debug('Send message to contactA');
{
const composeArea = window.locator(
'.composition-area-wrapper, .conversation .ConversationView'
);
const compositionInput = composeArea.locator('_react=CompositionInput');
await compositionInput.type('message to contactA');
await compositionInput.press('Enter');
}
debug('Wait for the message to contactA');
{
const { source, body } = await contactA.waitForMessage();
assert.strictEqual(
source,
desktop,
'first message must have valid source'
);
assert.strictEqual(
body,
'message to contactA',
'message must have correct body'
);
}
debug('Add phone number to contactA via storage service');
{
const state = await phone.expectStorageState('consistency check');
const updated = await phone.setStorageState(
state
.removeRecord(
item =>
item.record.contact?.serviceUuid ===
contactA.device.getUUIDByKind(UUIDKind.ACI)
)
.addContact(
contactA,
{
identityState: Proto.ContactRecord.IdentityState.DEFAULT,
whitelisted: true,
identityKey: contactA.getPublicKey(UUIDKind.ACI).serialize(),
givenName: 'ContactA',
serviceE164: contactA.device.number,
},
UUIDKind.ACI
)
);
const updatedStorageVersion = updated.version;
await phone.sendFetchStorage({
timestamp: bootstrap.getTimestamp(),
});
await app.waitForManifestVersion(updatedStorageVersion);
}
debug('Verify final state');
{
// One outgoing message
const messages = window.locator('.module-message__text');
assert.strictEqual(await messages.count(), 1, 'messages');
// One 'learned number' notification
const notifications = window.locator('.SystemMessage');
assert.strictEqual(await notifications.count(), 1, 'notifications');
const first = await notifications.first();
assert.match(await first.innerText(), /belongs to ContactA$/);
}
});
it('Does not show Learned Number notification if no sent, not in allowlist', async () => {
const { phone } = bootstrap;
const window = await app.getWindow();
debug('Open conversation with contactA');
{
const leftPane = window.locator('.left-pane-wrapper');
await leftPane
.locator('_react=ConversationListItem[title = "ContactA"]')
.click();
await window.locator('.module-conversation-hero').waitFor();
}
debug('Verify starting state');
{
// No messages
const messages = window.locator('.module-message__text');
assert.strictEqual(await messages.count(), 0, 'message count');
// No notifications
const notifications = window.locator('.SystemMessage');
assert.strictEqual(await notifications.count(), 0, 'notification count');
}
debug('Add phone number to contactA via storage service');
{
const state = await phone.expectStorageState('consistency check');
const updated = await phone.setStorageState(
state
.removeRecord(
item =>
item.record.contact?.serviceUuid ===
contactA.device.getUUIDByKind(UUIDKind.ACI)
)
.addContact(
contactA,
{
identityState: Proto.ContactRecord.IdentityState.DEFAULT,
whitelisted: false,
identityKey: contactA.getPublicKey(UUIDKind.ACI).serialize(),
givenName: 'ContactA',
serviceE164: contactA.device.number,
},
UUIDKind.ACI
)
);
const updatedStorageVersion = updated.version;
await phone.sendFetchStorage({
timestamp: bootstrap.getTimestamp(),
});
await app.waitForManifestVersion(updatedStorageVersion);
}
debug('Verify final state');
{
// No messages
const messages = window.locator('.module-message__text');
assert.strictEqual(await messages.count(), 0, 'messages');
// No 'learned number' notification
const notifications = window.locator('.SystemMessage');
assert.strictEqual(await notifications.count(), 0, 'notifications');
}
});
});

View file

@ -220,7 +220,7 @@ describe('pnp/merge', function needsName() {
const first = await notifications.first(); const first = await notifications.first();
assert.match( assert.match(
await first.innerText(), await first.innerText(),
/and ACI Contact are the same account. Your message history for both chats are here./ /Your message history with ACI Contact and their number .* has been merged./
); );
} }
}); });

View file

@ -339,6 +339,7 @@ describe('pnp/PNI Signature', function needsName() {
} }
debug('Verify final state'); debug('Verify final state');
{ {
const newState = await phone.waitForStorageState({ const newState = await phone.waitForStorageState({
after: state, after: state,

View file

@ -5,19 +5,28 @@ import type { LocalizerType } from '../types/Util';
export function getStringForConversationMerge({ export function getStringForConversationMerge({
obsoleteConversationTitle, obsoleteConversationTitle,
obsoleteConversationNumber,
conversationTitle, conversationTitle,
i18n, i18n,
}: { }: {
obsoleteConversationTitle: string | undefined; obsoleteConversationTitle: string | undefined;
obsoleteConversationNumber: string | undefined;
conversationTitle: string; conversationTitle: string;
i18n: LocalizerType; i18n: LocalizerType;
}): string { }): string {
if (!obsoleteConversationTitle) { if (!obsoleteConversationTitle) {
return i18n('icu:ConversationMerge--notification--no-e164', { return i18n('icu:ConversationMerge--notification--no-title', {
conversationTitle, conversationTitle,
}); });
} }
if (obsoleteConversationNumber) {
return i18n('icu:ConversationMerge--notification--with-e164', {
conversationTitle,
obsoleteConversationNumber,
});
}
return i18n('icu:ConversationMerge--notification', { return i18n('icu:ConversationMerge--notification', {
obsoleteConversationTitle, obsoleteConversationTitle,
conversationTitle, conversationTitle,

View file

@ -1,29 +0,0 @@
// Copyright 2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { LocalizerType } from '../types/Util';
export function getStringForPhoneNumberDiscovery({
phoneNumber,
i18n,
conversationTitle,
sharedGroup,
}: {
phoneNumber: string;
i18n: LocalizerType;
conversationTitle: string;
sharedGroup?: string;
}): string {
if (sharedGroup) {
return i18n('icu:PhoneNumberDiscovery--notification--withSharedGroup', {
phoneNumber,
conversationTitle,
sharedGroup,
});
}
return i18n('icu:PhoneNumberDiscovery--notification--noSharedGroup', {
phoneNumber,
conversationTitle,
});
}