2022-05-18 10:42:51 +00:00
|
|
|
package protocol
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"testing"
|
|
|
|
|
2023-07-16 11:11:48 +00:00
|
|
|
"github.com/status-im/status-go/eth-node/types"
|
2022-05-18 10:42:51 +00:00
|
|
|
"github.com/status-im/status-go/multiaccounts/accounts"
|
|
|
|
"github.com/status-im/status-go/protocol/encryption/multidevice"
|
|
|
|
"github.com/status-im/status-go/protocol/tt"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestMessengerSyncWalletSuite(t *testing.T) {
|
|
|
|
suite.Run(t, new(MessengerSyncWalletSuite))
|
|
|
|
}
|
|
|
|
|
|
|
|
type MessengerSyncWalletSuite struct {
|
2023-07-13 10:28:34 +00:00
|
|
|
MessengerBaseTestSuite
|
2022-05-18 10:42:51 +00:00
|
|
|
}
|
|
|
|
|
2023-05-24 14:42:31 +00:00
|
|
|
// user should not be able to change a keypair name directly, it follows display name
|
|
|
|
func (s *MessengerSyncWalletSuite) TestProfileKeypairNameChange() {
|
|
|
|
profileKp := accounts.GetProfileKeypairForTest(true, false, false)
|
|
|
|
profileKp.KeyUID = s.m.account.KeyUID
|
|
|
|
profileKp.Name = s.m.account.Name
|
|
|
|
profileKp.Accounts[0].KeyUID = s.m.account.KeyUID
|
|
|
|
|
|
|
|
// Create a main account on alice
|
|
|
|
err := s.m.settings.SaveOrUpdateKeypair(profileKp)
|
|
|
|
s.Require().NoError(err, "profile keypair alice.settings.SaveOrUpdateKeypair")
|
|
|
|
|
|
|
|
// Check account is present in the db
|
|
|
|
dbProfileKp, err := s.m.settings.GetKeypairByKeyUID(profileKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(accounts.SameKeypairs(profileKp, dbProfileKp))
|
|
|
|
|
|
|
|
// Try to change profile keypair name using `SaveOrUpdateKeypair` function
|
|
|
|
profileKp1 := accounts.GetProfileKeypairForTest(true, false, false)
|
|
|
|
profileKp1.Name = profileKp1.Name + "updated"
|
|
|
|
profileKp1.KeyUID = s.m.account.KeyUID
|
|
|
|
profileKp1.Accounts[0].KeyUID = s.m.account.KeyUID
|
|
|
|
|
|
|
|
err = s.m.SaveOrUpdateKeypair(profileKp1)
|
|
|
|
s.Require().Error(err)
|
|
|
|
s.Require().True(err == ErrCannotChangeKeypairName)
|
|
|
|
|
|
|
|
// Check the db
|
|
|
|
dbProfileKp, err = s.m.settings.GetKeypairByKeyUID(profileKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(accounts.SameKeypairs(profileKp, dbProfileKp))
|
|
|
|
|
|
|
|
// Try to change profile keypair name using `UpdateKeypairName` function
|
|
|
|
err = s.m.UpdateKeypairName(profileKp1.KeyUID, profileKp1.Name)
|
|
|
|
s.Require().Error(err)
|
|
|
|
s.Require().True(err == ErrCannotChangeKeypairName)
|
|
|
|
|
|
|
|
// Check the db
|
|
|
|
dbProfileKp, err = s.m.settings.GetKeypairByKeyUID(profileKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(accounts.SameKeypairs(profileKp, dbProfileKp))
|
|
|
|
}
|
|
|
|
|
2023-04-19 14:44:57 +00:00
|
|
|
func (s *MessengerSyncWalletSuite) TestSyncWallets() {
|
2023-05-24 14:40:40 +00:00
|
|
|
profileKp := accounts.GetProfileKeypairForTest(true, true, true)
|
2023-06-28 19:45:36 +00:00
|
|
|
// set clocks for accounts
|
|
|
|
profileKp.Clock = uint64(len(profileKp.Accounts) - 1)
|
|
|
|
for i, acc := range profileKp.Accounts {
|
|
|
|
acc.Clock = uint64(i)
|
|
|
|
}
|
2022-05-18 10:42:51 +00:00
|
|
|
|
|
|
|
// Create a main account on alice
|
2023-05-16 10:50:04 +00:00
|
|
|
err := s.m.settings.SaveOrUpdateKeypair(profileKp)
|
|
|
|
s.Require().NoError(err, "profile keypair alice.settings.SaveOrUpdateKeypair")
|
2022-05-18 10:42:51 +00:00
|
|
|
|
|
|
|
// Check account is present in the db
|
2023-05-16 10:50:04 +00:00
|
|
|
dbProfileKp1, err := s.m.settings.GetKeypairByKeyUID(profileKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(accounts.SameKeypairs(profileKp, dbProfileKp1))
|
2022-05-18 10:42:51 +00:00
|
|
|
|
|
|
|
// Create new device and add main account to
|
|
|
|
alicesOtherDevice, err := newMessengerWithKey(s.shh, s.m.identity, s.logger, nil)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
2023-05-16 10:50:04 +00:00
|
|
|
// Store only chat and default wallet account on other device
|
2023-05-24 14:40:40 +00:00
|
|
|
profileKpOtherDevice := accounts.GetProfileKeypairForTest(true, true, false)
|
2023-05-16 10:50:04 +00:00
|
|
|
err = alicesOtherDevice.settings.SaveOrUpdateKeypair(profileKpOtherDevice)
|
|
|
|
s.Require().NoError(err, "profile keypair alicesOtherDevice.settings.SaveOrUpdateKeypair")
|
2022-05-18 10:42:51 +00:00
|
|
|
|
2023-05-16 10:50:04 +00:00
|
|
|
// Check account is present in the db
|
|
|
|
dbProfileKp2, err := alicesOtherDevice.settings.GetKeypairByKeyUID(profileKpOtherDevice.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(accounts.SameKeypairs(profileKpOtherDevice, dbProfileKp2))
|
2022-05-18 10:42:51 +00:00
|
|
|
|
|
|
|
// Pair devices
|
|
|
|
im1 := &multidevice.InstallationMetadata{
|
|
|
|
Name: "alice's-other-device",
|
|
|
|
DeviceType: "alice's-other-device-type",
|
|
|
|
}
|
|
|
|
err = alicesOtherDevice.SetInstallationMetadata(alicesOtherDevice.installationID, im1)
|
|
|
|
s.Require().NoError(err)
|
2023-02-28 12:32:45 +00:00
|
|
|
response, err := alicesOtherDevice.SendPairInstallation(context.Background(), nil)
|
2022-05-18 10:42:51 +00:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Chats(), 1)
|
|
|
|
s.Require().False(response.Chats()[0].Active)
|
|
|
|
|
|
|
|
// Wait for the message to reach its destination
|
|
|
|
response, err = WaitOnMessengerResponse(
|
|
|
|
s.m,
|
|
|
|
func(r *MessengerResponse) bool { return len(r.Installations) > 0 },
|
|
|
|
"installation not received",
|
|
|
|
)
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
actualInstallation := response.Installations[0]
|
|
|
|
s.Require().Equal(alicesOtherDevice.installationID, actualInstallation.ID)
|
|
|
|
s.Require().NotNil(actualInstallation.InstallationMetadata)
|
|
|
|
s.Require().Equal("alice's-other-device", actualInstallation.InstallationMetadata.Name)
|
|
|
|
s.Require().Equal("alice's-other-device-type", actualInstallation.InstallationMetadata.DeviceType)
|
|
|
|
|
|
|
|
err = s.m.EnableInstallation(alicesOtherDevice.installationID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
2023-05-16 10:50:04 +00:00
|
|
|
// Store seed phrase keypair with accounts on alice's device
|
|
|
|
seedPhraseKp := accounts.GetSeedImportedKeypair1ForTest()
|
|
|
|
err = s.m.settings.SaveOrUpdateKeypair(seedPhraseKp)
|
|
|
|
s.Require().NoError(err, "seed phrase keypair alice.settings.SaveOrUpdateKeypair")
|
|
|
|
|
|
|
|
dbSeedPhraseKp1, err := s.m.settings.GetKeypairByKeyUID(seedPhraseKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(accounts.SameKeypairs(seedPhraseKp, dbSeedPhraseKp1))
|
|
|
|
|
|
|
|
// Store private key keypair with accounts on alice's device
|
|
|
|
privKeyKp := accounts.GetPrivKeyImportedKeypairForTest()
|
|
|
|
err = s.m.settings.SaveOrUpdateKeypair(privKeyKp)
|
|
|
|
s.Require().NoError(err, "private key keypair alice.settings.SaveOrUpdateKeypair")
|
|
|
|
|
|
|
|
dbPrivKeyKp1, err := s.m.settings.GetKeypairByKeyUID(privKeyKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(accounts.SameKeypairs(privKeyKp, dbPrivKeyKp1))
|
|
|
|
|
|
|
|
// Store watch only accounts on alice's device
|
|
|
|
woAccounts := accounts.GetWatchOnlyAccountsForTest()
|
2023-06-28 19:45:36 +00:00
|
|
|
err = s.m.settings.SaveOrUpdateAccounts(woAccounts, false)
|
2023-05-16 10:50:04 +00:00
|
|
|
s.Require().NoError(err)
|
|
|
|
dbWoAccounts1, err := s.m.settings.GetWatchOnlyAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(woAccounts), len(dbWoAccounts1))
|
|
|
|
s.Require().True(haveSameElements(woAccounts, dbWoAccounts1, accounts.SameAccounts))
|
|
|
|
|
|
|
|
dbAccounts1, err := s.m.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(profileKp.Accounts)+len(seedPhraseKp.Accounts)+len(privKeyKp.Accounts)+len(woAccounts), len(dbAccounts1))
|
2022-05-18 10:42:51 +00:00
|
|
|
|
|
|
|
// Trigger's a sync between devices
|
2023-01-06 12:21:14 +00:00
|
|
|
err = s.m.SyncDevices(context.Background(), "ens-name", "profile-image", nil)
|
2022-05-18 10:42:51 +00:00
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
2023-04-19 14:44:57 +00:00
|
|
|
response, err := alicesOtherDevice.RetrieveAll()
|
2022-05-18 10:42:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-05-16 10:50:04 +00:00
|
|
|
if len(response.Keypairs) != 3 || // 3 keypairs (profile, seed, priv key)
|
2023-06-28 19:45:36 +00:00
|
|
|
len(response.WatchOnlyAccounts) != len(woAccounts) {
|
2022-05-18 10:42:51 +00:00
|
|
|
return errors.New("no sync wallet account received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
2023-06-28 19:45:36 +00:00
|
|
|
dbProfileKp2, err = alicesOtherDevice.settings.GetKeypairByKeyUID(profileKp.KeyUID)
|
2023-05-16 10:50:04 +00:00
|
|
|
s.Require().NoError(err)
|
2023-06-28 19:45:36 +00:00
|
|
|
s.Require().True(profileKp.KeyUID == dbProfileKp2.KeyUID &&
|
|
|
|
profileKp.Name == dbProfileKp2.Name &&
|
|
|
|
profileKp.Type == dbProfileKp2.Type &&
|
|
|
|
profileKp.DerivedFrom == dbProfileKp2.DerivedFrom &&
|
|
|
|
profileKp.LastUsedDerivationIndex == dbProfileKp2.LastUsedDerivationIndex &&
|
|
|
|
profileKp.Clock == dbProfileKp2.Clock &&
|
|
|
|
len(profileKp.Accounts) == len(dbProfileKp2.Accounts))
|
|
|
|
// chat and default wallet account should be fully operable, other accounts partially operable
|
|
|
|
for i := range profileKp.Accounts {
|
|
|
|
match := false
|
|
|
|
expectedOperableValue := accounts.AccountPartiallyOperable
|
|
|
|
if profileKp.Accounts[i].Chat || profileKp.Accounts[i].Wallet {
|
|
|
|
expectedOperableValue = accounts.AccountFullyOperable
|
|
|
|
}
|
|
|
|
for j := range dbProfileKp2.Accounts {
|
|
|
|
if accounts.SameAccountsWithDifferentOperable(profileKp.Accounts[i], dbProfileKp2.Accounts[j], expectedOperableValue) {
|
|
|
|
match = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().True(match)
|
|
|
|
}
|
2023-04-19 14:44:57 +00:00
|
|
|
|
2023-05-16 10:50:04 +00:00
|
|
|
dbSeedPhraseKp2, err := alicesOtherDevice.settings.GetKeypairByKeyUID(seedPhraseKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(accounts.SameKeypairsWithDifferentSyncedFrom(seedPhraseKp, dbSeedPhraseKp2, true, "", accounts.AccountNonOperable))
|
2022-05-18 10:42:51 +00:00
|
|
|
|
2023-05-16 10:50:04 +00:00
|
|
|
dbPrivKeyKp2, err := alicesOtherDevice.settings.GetKeypairByKeyUID(privKeyKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(accounts.SameKeypairsWithDifferentSyncedFrom(privKeyKp, dbPrivKeyKp2, true, "", accounts.AccountNonOperable))
|
|
|
|
|
|
|
|
dbWoAccounts2, err := alicesOtherDevice.settings.GetWatchOnlyAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(woAccounts), len(dbWoAccounts2))
|
|
|
|
s.Require().True(haveSameElements(woAccounts, dbWoAccounts2, accounts.SameAccounts))
|
|
|
|
|
|
|
|
dbAccounts2, err := alicesOtherDevice.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(profileKp.Accounts)+len(seedPhraseKp.Accounts)+len(privKeyKp.Accounts)+len(woAccounts), len(dbAccounts2))
|
|
|
|
|
|
|
|
s.Require().True(haveSameElements(dbAccounts1, dbAccounts2, accounts.SameAccounts))
|
|
|
|
|
|
|
|
// Update keypair name on alice's primary device
|
2023-05-24 14:40:40 +00:00
|
|
|
profileKpUpdated := accounts.GetProfileKeypairForTest(true, true, false)
|
2023-05-16 10:50:04 +00:00
|
|
|
profileKpUpdated.Name = profileKp.Name + "Updated"
|
|
|
|
profileKpUpdated.Accounts = profileKp.Accounts[:0]
|
|
|
|
err = s.m.SaveOrUpdateKeypair(profileKpUpdated)
|
|
|
|
s.Require().NoError(err, "updated keypair name on alice primary device")
|
2022-05-18 10:42:51 +00:00
|
|
|
|
|
|
|
// Sync between devices is triggered automatically
|
|
|
|
// via watch account changes subscription
|
|
|
|
// Retrieve community link & community
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
2023-04-19 14:44:57 +00:00
|
|
|
response, err := alicesOtherDevice.RetrieveAll()
|
2022-05-18 10:42:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-05-16 10:50:04 +00:00
|
|
|
if len(response.Keypairs) != 1 {
|
2023-06-28 19:45:36 +00:00
|
|
|
return errors.New("no sync keypairs received")
|
2022-05-18 10:42:51 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
2023-05-16 10:50:04 +00:00
|
|
|
// check on alice's other device
|
|
|
|
dbProfileKp2, err = alicesOtherDevice.settings.GetKeypairByKeyUID(profileKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(profileKpUpdated.Name, dbProfileKp2.Name)
|
|
|
|
|
|
|
|
// Update accounts on alice's primary device
|
2023-05-24 14:40:40 +00:00
|
|
|
profileKpUpdated = accounts.GetProfileKeypairForTest(true, true, true)
|
2023-05-16 10:50:04 +00:00
|
|
|
accountsToUpdate := profileKpUpdated.Accounts[2:]
|
|
|
|
for _, acc := range accountsToUpdate {
|
|
|
|
acc.Name = acc.Name + "Updated"
|
2023-06-02 15:06:51 +00:00
|
|
|
acc.ColorID = acc.ColorID + "Updated"
|
2023-05-16 10:50:04 +00:00
|
|
|
acc.Emoji = acc.Emoji + "Updated"
|
|
|
|
err = s.m.SaveOrUpdateAccount(acc)
|
|
|
|
s.Require().NoError(err, "updated account on alice primary device")
|
|
|
|
}
|
2023-04-19 14:44:57 +00:00
|
|
|
|
2023-05-16 10:50:04 +00:00
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err := alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2023-04-19 14:44:57 +00:00
|
|
|
}
|
2023-05-16 10:50:04 +00:00
|
|
|
|
2023-06-28 19:45:36 +00:00
|
|
|
if len(response.Keypairs) != 2 {
|
|
|
|
return errors.New("no sync keypairs received")
|
2022-05-18 10:42:51 +00:00
|
|
|
}
|
2023-05-16 10:50:04 +00:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// check on alice's other device
|
|
|
|
dbProfileKp2, err = alicesOtherDevice.settings.GetKeypairByKeyUID(profileKp.KeyUID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
for _, acc := range accountsToUpdate {
|
|
|
|
s.Require().True(contains(dbProfileKp2.Accounts, acc, accounts.SameAccounts))
|
2022-05-18 10:42:51 +00:00
|
|
|
}
|
|
|
|
}
|
2023-07-16 11:11:48 +00:00
|
|
|
|
|
|
|
func (s *MessengerSyncWalletSuite) TestSyncWalletAccountsReorder() {
|
|
|
|
profileKp := accounts.GetProfileKeypairForTest(true, false, false)
|
|
|
|
profileKp.Accounts[0].Position = -1 // Chat account must be at position -1 always
|
|
|
|
|
|
|
|
woAccounts := []*accounts.Account{
|
|
|
|
{Address: types.Address{0x11}, Type: accounts.AccountTypeWatch, Position: 0},
|
|
|
|
{Address: types.Address{0x12}, Type: accounts.AccountTypeWatch, Position: 1},
|
|
|
|
{Address: types.Address{0x13}, Type: accounts.AccountTypeWatch, Position: 2},
|
|
|
|
{Address: types.Address{0x14}, Type: accounts.AccountTypeWatch, Position: 3},
|
|
|
|
{Address: types.Address{0x15}, Type: accounts.AccountTypeWatch, Position: 4},
|
|
|
|
{Address: types.Address{0x16}, Type: accounts.AccountTypeWatch, Position: 5},
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a main account on alice
|
|
|
|
err := s.m.settings.SaveOrUpdateKeypair(profileKp)
|
|
|
|
s.Require().NoError(err, "profile keypair alice.settings.SaveOrUpdateKeypair")
|
|
|
|
// Store watch only accounts on alice's device
|
|
|
|
err = s.m.settings.SaveOrUpdateAccounts(woAccounts, false)
|
|
|
|
s.Require().NoError(err, "wo accounts alice.settings.SaveOrUpdateKeypair")
|
|
|
|
|
|
|
|
dbAccounts, err := s.m.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
|
|
|
|
|
|
|
|
// Create a main account on alice's other device
|
|
|
|
alicesOtherDevice, err := newMessengerWithKey(s.shh, s.m.identity, s.logger, nil)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
err = alicesOtherDevice.settings.SaveOrUpdateKeypair(profileKp)
|
|
|
|
s.Require().NoError(err, "profile keypair alice.settings.SaveOrUpdateKeypair")
|
|
|
|
// Store watch only accounts on alice's other device
|
|
|
|
err = alicesOtherDevice.settings.SaveOrUpdateAccounts(woAccounts, false)
|
|
|
|
s.Require().NoError(err, "wo accounts alice.settings.SaveOrUpdateKeypair")
|
|
|
|
|
|
|
|
dbAccounts, err = alicesOtherDevice.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
|
|
|
|
|
|
|
|
// Pair devices
|
|
|
|
im1 := &multidevice.InstallationMetadata{
|
|
|
|
Name: "alice's-other-device",
|
|
|
|
DeviceType: "alice's-other-device-type",
|
|
|
|
}
|
|
|
|
err = alicesOtherDevice.SetInstallationMetadata(alicesOtherDevice.installationID, im1)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
response, err := alicesOtherDevice.SendPairInstallation(context.Background(), nil)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Chats(), 1)
|
|
|
|
s.Require().False(response.Chats()[0].Active)
|
|
|
|
|
|
|
|
// Wait for the message to reach its destination
|
|
|
|
response, err = WaitOnMessengerResponse(
|
|
|
|
s.m,
|
|
|
|
func(r *MessengerResponse) bool { return len(r.Installations) > 0 },
|
|
|
|
"installation not received",
|
|
|
|
)
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
actualInstallation := response.Installations[0]
|
|
|
|
s.Require().Equal(alicesOtherDevice.installationID, actualInstallation.ID)
|
|
|
|
s.Require().NotNil(actualInstallation.InstallationMetadata)
|
|
|
|
s.Require().Equal("alice's-other-device", actualInstallation.InstallationMetadata.Name)
|
|
|
|
s.Require().Equal("alice's-other-device-type", actualInstallation.InstallationMetadata.DeviceType)
|
|
|
|
|
|
|
|
err = s.m.EnableInstallation(alicesOtherDevice.installationID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Move down account from position 1 to position 4
|
|
|
|
err = s.m.MoveWalletAccount(1, 4)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Expected after moving down
|
|
|
|
woAccounts = []*accounts.Account{
|
|
|
|
{Address: types.Address{0x11}, Type: accounts.AccountTypeWatch, Position: 0},
|
|
|
|
{Address: types.Address{0x13}, Type: accounts.AccountTypeWatch, Position: 1},
|
|
|
|
{Address: types.Address{0x14}, Type: accounts.AccountTypeWatch, Position: 2},
|
|
|
|
{Address: types.Address{0x15}, Type: accounts.AccountTypeWatch, Position: 3},
|
|
|
|
{Address: types.Address{0x12}, Type: accounts.AccountTypeWatch, Position: 4}, // acc with addr 0x12 is at position 4 (moved from position 1)
|
|
|
|
{Address: types.Address{0x16}, Type: accounts.AccountTypeWatch, Position: 5},
|
|
|
|
}
|
|
|
|
|
|
|
|
dbAccounts, err = s.m.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
|
|
|
|
for i := 0; i < len(woAccounts); i++ {
|
2023-07-19 11:50:16 +00:00
|
|
|
s.Require().True(accounts.SameAccountsIncludingPosition(woAccounts[i], dbAccounts[i+1]))
|
2023-07-16 11:11:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sync between devices is triggered automatically
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err := alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.AccountsPositions) != len(woAccounts) {
|
|
|
|
return errors.New("no sync message received for accounts reordering")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// check on alice's other device
|
|
|
|
dbAccounts, err = alicesOtherDevice.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
|
|
|
|
for i := 0; i < len(woAccounts); i++ {
|
2023-07-19 11:50:16 +00:00
|
|
|
s.Require().True(accounts.SameAccountsIncludingPosition(woAccounts[i], dbAccounts[i+1]))
|
2023-07-16 11:11:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// compare times
|
|
|
|
dbClock, err := s.m.settings.GetClockOfLastAccountsPositionChange()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
dbClockOtherDevice, err := s.m.settings.GetClockOfLastAccountsPositionChange()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(dbClock, dbClockOtherDevice)
|
|
|
|
|
|
|
|
// Move up account from position 5 to position 0
|
|
|
|
err = s.m.MoveWalletAccount(5, 0)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Expected after moving down
|
|
|
|
woAccounts = []*accounts.Account{
|
|
|
|
{Address: types.Address{0x16}, Type: accounts.AccountTypeWatch, Position: 0}, // acc with addr 0x16 is at position 0 (moved from position 5)
|
|
|
|
{Address: types.Address{0x11}, Type: accounts.AccountTypeWatch, Position: 1},
|
|
|
|
{Address: types.Address{0x13}, Type: accounts.AccountTypeWatch, Position: 2},
|
|
|
|
{Address: types.Address{0x14}, Type: accounts.AccountTypeWatch, Position: 3},
|
|
|
|
{Address: types.Address{0x15}, Type: accounts.AccountTypeWatch, Position: 4},
|
|
|
|
{Address: types.Address{0x12}, Type: accounts.AccountTypeWatch, Position: 5},
|
|
|
|
}
|
|
|
|
|
|
|
|
dbAccounts, err = s.m.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
|
|
|
|
for i := 0; i < len(woAccounts); i++ {
|
2023-07-19 11:50:16 +00:00
|
|
|
s.Require().True(accounts.SameAccountsIncludingPosition(woAccounts[i], dbAccounts[i+1]))
|
2023-07-16 11:11:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sync between devices is triggered automatically
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err := alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.AccountsPositions) != len(woAccounts) {
|
|
|
|
return errors.New("no sync message received for accounts reordering")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// check on alice's other device
|
|
|
|
dbAccounts, err = alicesOtherDevice.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
|
|
|
|
for i := 0; i < len(woAccounts); i++ {
|
2023-07-19 11:50:16 +00:00
|
|
|
s.Require().True(accounts.SameAccountsIncludingPosition(woAccounts[i], dbAccounts[i+1]))
|
2023-07-16 11:11:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// compare times
|
|
|
|
dbClock, err = s.m.settings.GetClockOfLastAccountsPositionChange()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
dbClockOtherDevice, err = s.m.settings.GetClockOfLastAccountsPositionChange()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(dbClock, dbClockOtherDevice)
|
|
|
|
}
|
2023-07-19 11:50:16 +00:00
|
|
|
|
|
|
|
func (s *MessengerSyncWalletSuite) TestSyncWalletAccountOrderAfterDeletion() {
|
|
|
|
profileKp := accounts.GetProfileKeypairForTest(true, true, true)
|
|
|
|
// set clocks for accounts
|
|
|
|
profileKp.Clock = uint64(len(profileKp.Accounts) - 1)
|
|
|
|
i := -1
|
|
|
|
for _, acc := range profileKp.Accounts {
|
|
|
|
acc.Clock = uint64(i + 1)
|
|
|
|
acc.Position = int64(i)
|
|
|
|
acc.Operable = accounts.AccountNonOperable
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a main account on alice
|
|
|
|
err := s.m.settings.SaveOrUpdateKeypair(profileKp)
|
|
|
|
s.Require().NoError(err, "profile keypair alice.settings.SaveOrUpdateKeypair")
|
|
|
|
// Store seed phrase keypair with accounts on alice's device
|
|
|
|
seedPhraseKp := accounts.GetSeedImportedKeypair1ForTest()
|
|
|
|
for _, acc := range seedPhraseKp.Accounts {
|
|
|
|
acc.Clock = uint64(i + 1)
|
|
|
|
acc.Position = int64(i)
|
|
|
|
acc.Operable = accounts.AccountNonOperable
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
err = s.m.settings.SaveOrUpdateKeypair(seedPhraseKp)
|
|
|
|
s.Require().NoError(err, "seed phrase keypair alice.settings.SaveOrUpdateKeypair")
|
|
|
|
// Store private key keypair with accounts on alice's device
|
|
|
|
privKeyKp := accounts.GetPrivKeyImportedKeypairForTest()
|
|
|
|
for _, acc := range privKeyKp.Accounts {
|
|
|
|
acc.Clock = uint64(i + 1)
|
|
|
|
acc.Position = int64(i)
|
|
|
|
acc.Operable = accounts.AccountNonOperable
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
err = s.m.settings.SaveOrUpdateKeypair(privKeyKp)
|
|
|
|
s.Require().NoError(err, "private key keypair alice.settings.SaveOrUpdateKeypair")
|
|
|
|
// Store watch only accounts on alice's device
|
|
|
|
woAccounts := accounts.GetWatchOnlyAccountsForTest()
|
|
|
|
for _, acc := range woAccounts {
|
|
|
|
acc.Clock = uint64(i + 1)
|
|
|
|
acc.Position = int64(i)
|
|
|
|
acc.Operable = accounts.AccountFullyOperable
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
err = s.m.settings.SaveOrUpdateAccounts(woAccounts, false)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
// Check accounts
|
|
|
|
dbAccounts1, err := s.m.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
totalNumOfAccounts := len(profileKp.Accounts) + len(seedPhraseKp.Accounts) + len(privKeyKp.Accounts) + len(woAccounts)
|
|
|
|
s.Require().Equal(totalNumOfAccounts, len(dbAccounts1))
|
|
|
|
|
|
|
|
// Create new device and add main account to
|
|
|
|
alicesOtherDevice, err := newMessengerWithKey(s.shh, s.m.identity, s.logger, nil)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
// Store only chat and default wallet account on other device
|
|
|
|
profileKpOtherDevice := accounts.GetProfileKeypairForTest(true, true, false)
|
|
|
|
err = alicesOtherDevice.settings.SaveOrUpdateKeypair(profileKpOtherDevice)
|
|
|
|
s.Require().NoError(err, "profile keypair alicesOtherDevice.settings.SaveOrUpdateKeypair")
|
|
|
|
|
|
|
|
// Pair devices
|
|
|
|
im1 := &multidevice.InstallationMetadata{
|
|
|
|
Name: "alice's-other-device",
|
|
|
|
DeviceType: "alice's-other-device-type",
|
|
|
|
}
|
|
|
|
err = alicesOtherDevice.SetInstallationMetadata(alicesOtherDevice.installationID, im1)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
response, err := alicesOtherDevice.SendPairInstallation(context.Background(), nil)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Chats(), 1)
|
|
|
|
s.Require().False(response.Chats()[0].Active)
|
|
|
|
|
|
|
|
// Wait for the message to reach its destination
|
|
|
|
response, err = WaitOnMessengerResponse(
|
|
|
|
s.m,
|
|
|
|
func(r *MessengerResponse) bool { return len(r.Installations) > 0 },
|
|
|
|
"installation not received",
|
|
|
|
)
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
actualInstallation := response.Installations[0]
|
|
|
|
s.Require().Equal(alicesOtherDevice.installationID, actualInstallation.ID)
|
|
|
|
s.Require().NotNil(actualInstallation.InstallationMetadata)
|
|
|
|
s.Require().Equal("alice's-other-device", actualInstallation.InstallationMetadata.Name)
|
|
|
|
s.Require().Equal("alice's-other-device-type", actualInstallation.InstallationMetadata.DeviceType)
|
|
|
|
|
|
|
|
err = s.m.EnableInstallation(alicesOtherDevice.installationID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Trigger's a sync between devices
|
|
|
|
err = s.m.SyncDevices(context.Background(), "ens-name", "profile-image", nil)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err := alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.Keypairs) != 3 || // 3 keypairs (profile, seed, priv key)
|
|
|
|
len(response.WatchOnlyAccounts) != len(woAccounts) ||
|
|
|
|
len(response.AccountsPositions) != totalNumOfAccounts-1 /* we don't include chat account in position ordering*/ {
|
|
|
|
return errors.New("no sync wallet account received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
dbAccounts2, err := alicesOtherDevice.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(totalNumOfAccounts, len(dbAccounts2))
|
|
|
|
|
|
|
|
s.Require().True(haveSameElements(dbAccounts1, dbAccounts2, accounts.SameAccountsIncludingPosition))
|
|
|
|
|
|
|
|
// Delete keypair related account on alice's primary device
|
|
|
|
accToDelete := seedPhraseKp.Accounts[1]
|
|
|
|
err = s.m.DeleteAccount(accToDelete.Address)
|
|
|
|
s.Require().NoError(err, "delete account on alice primary device")
|
|
|
|
|
|
|
|
totalNumOfAccounts-- //one acc less
|
|
|
|
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err := alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.Keypairs) != 1 {
|
|
|
|
return errors.New("no sync keypairs received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
dbAccounts1, err = s.m.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(totalNumOfAccounts, len(dbAccounts1))
|
|
|
|
|
|
|
|
dbAccounts2, err = alicesOtherDevice.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(totalNumOfAccounts, len(dbAccounts2))
|
|
|
|
|
|
|
|
s.Require().True(haveSameElements(dbAccounts1, dbAccounts2, accounts.SameAccountsIncludingPosition))
|
|
|
|
|
|
|
|
// Delete watch only account on alice's primary device
|
|
|
|
accToDelete = woAccounts[1]
|
|
|
|
err = s.m.DeleteAccount(accToDelete.Address)
|
|
|
|
s.Require().NoError(err, "delete account on alice primary device")
|
|
|
|
|
|
|
|
totalNumOfAccounts-- //one acc less
|
|
|
|
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err := alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.WatchOnlyAccounts) != 1 {
|
|
|
|
return errors.New("no sync keypairs received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
dbAccounts1, err = s.m.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(totalNumOfAccounts, len(dbAccounts1))
|
|
|
|
|
|
|
|
dbAccounts2, err = alicesOtherDevice.settings.GetAccounts()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(totalNumOfAccounts, len(dbAccounts2))
|
|
|
|
|
|
|
|
s.Require().True(haveSameElements(dbAccounts1, dbAccounts2, accounts.SameAccountsIncludingPosition))
|
|
|
|
}
|