status-go/protocol/messenger_pairing_and_syncing_test.go
frank a08319f615
fix(sync)_: sync fallback notification (#5888)
* fix(sync)_: Improve EnableInstallationAndSync and add EnableInstallationV2

- Refactor EnableInstallationAndSync for better error handling and response merging
- Add new EnableInstallationV2 method returning the installation
- Update tests to check for installation in response
- Deprecate old EnableInstallation method

* chore_: remove EnableInstallationV2

* fix(sync)_: create/delete AC notification only when targetInstallationID match

- Add targetInstallationID parameter to SendPairInstallation function
- Update protobuf SyncPairInstallation struct with TargetInstallationId field
- Modify method calls across multiple test files to include new parameter
- Update pairing.proto and pairing.pb.go with new field for local pairing

* chore_: rename stubEnableInstallationAndPair

chore_: move InstallationIDProvider

chore_: revert endpoints.go

test_: check AC with resp

chore_: rename ModifiedInstallationsTargetedToThisDevice

test_: add InstallationIDProvider

chore_: revert endpoints.go

chore_: remove comment

test_: add TestNewInstallationCreatedIsNotDeleted
2024-10-07 22:05:37 +08:00

237 lines
7.6 KiB
Go

package protocol
import (
"errors"
"testing"
"github.com/google/uuid"
"github.com/stretchr/testify/suite"
"github.com/status-im/status-go/eth-node/types"
"github.com/status-im/status-go/multiaccounts/accounts"
"github.com/status-im/status-go/multiaccounts/settings"
"github.com/status-im/status-go/protocol/encryption/multidevice"
"github.com/status-im/status-go/protocol/requests"
"github.com/status-im/status-go/protocol/tt"
)
func TestMessengerPairingTest(t *testing.T) {
suite.Run(t, new(MessengerPairingSuite))
}
type MessengerPairingSuite struct {
MessengerBaseTestSuite
}
func (s *MessengerPairingSuite) TestEnableNonExistingInstallation() {
installationID := uuid.New().String()
_, err := s.m.EnableInstallationAndPair(&requests.EnableInstallationAndPair{InstallationID: installationID})
s.Require().NoError(err)
installations := s.m.Installations()
s.Require().NoError(err)
s.Require().Len(installations, 2)
var theirInstallation *multidevice.Installation
for _, i := range installations {
if i.ID == installationID {
theirInstallation = i
break
} else {
s.Require().NotNil(i.InstallationMetadata)
}
}
s.Require().NotNil(theirInstallation)
s.Require().True(theirInstallation.Enabled)
installationsFromDB, err := s.m.encryptor.GetOurActiveInstallations(&s.m.identity.PublicKey)
s.Require().NoError(err)
s.Require().Len(installationsFromDB, 2)
for _, i := range installationsFromDB {
s.Require().True(i.Enabled)
if i.ID == installationID {
theirInstallation = i
break
}
}
s.Require().NotNil(theirInstallation)
s.Require().True(theirInstallation.Enabled)
}
type stubEnableInstallationAndPair struct {
installationID string
getInstallationIDInvokedFirst bool
}
func (m *stubEnableInstallationAndPair) GetInstallationID() string {
if !m.getInstallationIDInvokedFirst {
m.getInstallationIDInvokedFirst = true
return m.installationID
}
return "wrong installation ID"
}
func (m *stubEnableInstallationAndPair) Validate() error {
return nil
}
// TestNewInstallationReceivedIsNotCreated tests the scenario where alice2 wants to pair with alice1
// but the target installation ID is wrong so no AC notification(ActivityCenterNotificationTypeNewInstallationReceived)
// should be created for alice1
func (s *MessengerPairingSuite) TestNewInstallationReceivedIsNotCreated() {
alice1 := s.m
alice2, err := newMessengerWithKey(s.shh, s.privateKey, s.logger, nil)
s.Require().NoError(err)
defer TearDownMessenger(&s.Suite, alice2)
mockRequest := &stubEnableInstallationAndPair{installationID: alice1.installationID}
_, err = alice2.EnableInstallationAndPair(mockRequest)
s.Require().NoError(err)
resp, err := WaitOnMessengerResponse(
alice1,
func(r *MessengerResponse) bool {
for _, i := range r.Installations() {
if i.ID == alice2.installationID {
return true
}
}
return false
},
"no messages",
)
s.Require().NoError(err)
s.Require().Len(resp.ActivityCenterNotifications(), 0)
}
// TestNewInstallationCreatedIsNotDeleted tests the scenario
// 1. prepare AC NewInstallationCreated for alice2
// 2. prepare AC NewInstallationCreated for alice3
// 3. alice1.EnableInstallationAndSync
// 4. check AC NewInstallationCreated for alice3 is not deleted
func (s *MessengerPairingSuite) TestNewInstallationCreatedIsNotDeleted() {
alice1 := s.m
alice2, err := newMessengerWithKey(s.shh, s.privateKey, s.logger, nil)
s.Require().NoError(err)
alice3, err := newMessengerWithKey(s.shh, s.privateKey, s.logger, nil)
s.Require().NoError(err)
defer TearDownMessenger(&s.Suite, alice2)
defer TearDownMessenger(&s.Suite, alice3)
// prepare AC NewInstallationCreated for alice2
_, err = alice2.EnableInstallationAndPair(&requests.EnableInstallationAndPair{InstallationID: alice1.installationID})
s.Require().NoError(err)
// alice1 should get the installationID2 from alice2
s.expectInstallationReceived(alice1, alice2.installationID)
// prepare AC NewInstallationCreated for alice3
_, err = alice3.EnableInstallationAndPair(&requests.EnableInstallationAndPair{InstallationID: alice1.installationID})
s.Require().NoError(err)
// alice1 should get the installationID3 from alice3
s.expectInstallationReceived(alice1, alice3.installationID)
_, err = alice1.EnableInstallationAndSync(&requests.EnableInstallationAndSync{InstallationID: alice2.installationID})
s.Require().NoError(err)
_, err = WaitOnMessengerResponse(
alice3,
func(r *MessengerResponse) bool {
for _, i := range r.Installations() {
if i.ID == alice1.installationID {
return true
}
}
return false
},
"alice3 should get the installationID from alice1",
)
s.Require().NoError(err)
ac, err := alice3.ActivityCenterNotification(types.FromHex(alice1.installationID))
s.Require().NoError(err)
s.Require().Equal(ActivityCenterNotificationTypeNewInstallationCreated, ac.Type)
s.Require().False(ac.Deleted)
}
func (s *MessengerPairingSuite) expectInstallationReceived(m *Messenger, installationID string) {
resp, err := WaitOnMessengerResponse(
m,
func(r *MessengerResponse) bool {
for _, i := range r.Installations() {
if i.ID == installationID {
return true
}
}
return false
},
"installation not received",
)
s.Require().NoError(err)
s.Require().Len(resp.ActivityCenterNotifications(), 1)
s.Require().Equal(ActivityCenterNotificationTypeNewInstallationReceived, resp.ActivityCenterNotifications()[0].Type)
}
// TestMessengerSyncFallback tests the scenario where alice2 wants to sync with alice1
// alice1 generated the connection string for bootstraping alice2
// alice2 failed to connect to alice1 and restored from seed phrase
// alice2 get the installationID1 from alice1 via parsing the connection string
// alice2 should get the display name from alice1 after pairing
func (s *MessengerPairingSuite) TestMessengerSyncFallback() {
alice1 := s.m
alice2, err := newMessengerWithKey(s.shh, s.privateKey, s.logger, nil)
s.Require().NoError(err)
defer TearDownMessenger(&s.Suite, alice2)
alice1ProfileKp := accounts.GetProfileKeypairForTest(true, false, false)
alice1ProfileKp.KeyUID = alice1.account.KeyUID
alice1ProfileKp.Accounts[0].KeyUID = alice1.account.KeyUID
err = alice1.settings.SaveOrUpdateKeypair(alice1ProfileKp)
s.Require().NoError(err)
expectedDisplayName := "alice1"
s.Require().NoError(alice1.SetDisplayName(expectedDisplayName))
ss, err := alice2.getSettings()
s.Require().NoError(err)
s.Require().NotEqual(expectedDisplayName, ss.DisplayName)
installationID1 := alice1.installationID
installationID2 := alice2.installationID
s.Require().NotEqual(installationID1, installationID2)
_, err = alice2.EnableInstallationAndPair(&requests.EnableInstallationAndPair{InstallationID: installationID1})
s.Require().NoError(err)
// alice1 should get the installationID2 from alice2
s.expectInstallationReceived(alice1, installationID2)
// check response from alice1
resp, err := alice1.EnableInstallationAndSync(&requests.EnableInstallationAndSync{InstallationID: installationID2})
s.Require().NoError(err)
installationID2Exist := false
for _, i := range resp.Installations() {
if i.ID == installationID2 {
installationID2Exist = true
break
}
}
s.Require().True(installationID2Exist)
// check if the display name is synced
err = tt.RetryWithBackOff(func() error {
r, err := alice2.RetrieveAll()
s.Require().NoError(err)
for _, ss := range r.Settings {
if ss.GetDBName() == settings.DisplayName.GetDBName() {
return nil
}
}
return errors.New("display name setting not received yet")
})
s.Require().NoError(err)
ss, err = alice2.getSettings()
s.Require().NoError(err)
s.Require().Equal(expectedDisplayName, ss.DisplayName)
}