status-im/status-go

View on GitHub
protocol/messenger_sync_wallets_test.go

Summary

Maintainability
A
0 mins
Test Coverage
package protocol

import (
    "context"
    "errors"
    "testing"

    "github.com/status-im/status-go/eth-node/types"
    "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 {
    MessengerBaseTestSuite
}

// 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))
}

func (s *MessengerSyncWalletSuite) TestSyncWallets() {
    profileKp := accounts.GetProfileKeypairForTest(true, true, true)
    // set clocks for accounts
    profileKp.Clock = uint64(len(profileKp.Accounts) - 1)
    for i, acc := range profileKp.Accounts {
        acc.Clock = uint64(i)
    }

    // 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
    dbProfileKp1, err := s.m.settings.GetKeypairByKeyUID(profileKp.KeyUID)
    s.Require().NoError(err)
    s.Require().True(accounts.SameKeypairs(profileKp, dbProfileKp1))

    // 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")

    // 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))

    // 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)

    // 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()
    err = s.m.settings.SaveOrUpdateAccounts(woAccounts, false)
    s.Require().NoError(err)
    dbWoAccounts1, err := s.m.settings.GetActiveWatchOnlyAccounts()
    s.Require().NoError(err)
    s.Require().Equal(len(woAccounts), len(dbWoAccounts1))
    s.Require().True(haveSameElements(woAccounts, dbWoAccounts1, accounts.SameAccounts))

    dbAccounts1, err := s.m.settings.GetActiveAccounts()
    s.Require().NoError(err)
    s.Require().Equal(len(profileKp.Accounts)+len(seedPhraseKp.Accounts)+len(privKeyKp.Accounts)+len(woAccounts), len(dbAccounts1))

    // 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) {
            return errors.New("no sync wallet account received")
        }
        return nil
    })
    s.Require().NoError(err)

    dbProfileKp2, err = alicesOtherDevice.settings.GetKeypairByKeyUID(profileKp.KeyUID)
    s.Require().NoError(err)
    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)
    }

    dbSeedPhraseKp2, err := alicesOtherDevice.settings.GetKeypairByKeyUID(seedPhraseKp.KeyUID)
    s.Require().NoError(err)
    s.Require().True(accounts.SameKeypairsWithDifferentSyncedFrom(seedPhraseKp, dbSeedPhraseKp2, true, "", accounts.AccountNonOperable))

    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.GetActiveWatchOnlyAccounts()
    s.Require().NoError(err)
    s.Require().Equal(len(woAccounts), len(dbWoAccounts2))
    s.Require().True(haveSameElements(woAccounts, dbWoAccounts2, accounts.SameAccounts))

    dbAccounts2, err := alicesOtherDevice.settings.GetActiveAccounts()
    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
    profileKpUpdated := accounts.GetProfileKeypairForTest(true, true, false)
    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")

    // Sync between devices is triggered automatically
    // via watch account changes subscription
    // Retrieve community link & community
    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)

    // 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
    profileKpUpdated = accounts.GetProfileKeypairForTest(true, true, true)
    accountsToUpdate := profileKpUpdated.Accounts[2:]
    for _, acc := range accountsToUpdate {
        acc.Name = acc.Name + "Updated"
        acc.ColorID = acc.ColorID + "Updated"
        acc.Emoji = acc.Emoji + "Updated"
        err = s.m.SaveOrUpdateAccount(acc)
        s.Require().NoError(err, "updated account on alice primary device")
    }

    err = tt.RetryWithBackOff(func() error {
        response, err := alicesOtherDevice.RetrieveAll()
        if err != nil {
            return err
        }

        if len(response.Keypairs) != 2 {
            return errors.New("no sync keypairs received")
        }
        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))
    }
}

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.GetActiveAccounts()
    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.GetActiveAccounts()
    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.GetActiveAccounts()
    s.Require().NoError(err)
    s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
    for i := 0; i < len(woAccounts); i++ {
        s.Require().True(accounts.SameAccountsIncludingPosition(woAccounts[i], dbAccounts[i+1]))
    }

    // 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.GetActiveAccounts()
    s.Require().NoError(err)
    s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
    for i := 0; i < len(woAccounts); i++ {
        s.Require().True(accounts.SameAccountsIncludingPosition(woAccounts[i], dbAccounts[i+1]))
    }

    // 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.GetActiveAccounts()
    s.Require().NoError(err)
    s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
    for i := 0; i < len(woAccounts); i++ {
        s.Require().True(accounts.SameAccountsIncludingPosition(woAccounts[i], dbAccounts[i+1]))
    }

    // 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.GetActiveAccounts()
    s.Require().NoError(err)
    s.Require().Equal(len(woAccounts), len(dbAccounts)-1)
    for i := 0; i < len(woAccounts); i++ {
        s.Require().True(accounts.SameAccountsIncludingPosition(woAccounts[i], dbAccounts[i+1]))
    }

    // 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)
}

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.GetActiveAccounts()
    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.GetActiveAccounts()
    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.GetActiveAccounts()
    s.Require().NoError(err)
    s.Require().Equal(totalNumOfAccounts, len(dbAccounts1))

    dbAccounts2, err = alicesOtherDevice.settings.GetActiveAccounts()
    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.GetActiveAccounts()
    s.Require().NoError(err)
    s.Require().Equal(totalNumOfAccounts, len(dbAccounts1))

    dbAccounts2, err = alicesOtherDevice.settings.GetActiveAccounts()
    s.Require().NoError(err)
    s.Require().Equal(totalNumOfAccounts, len(dbAccounts2))

    s.Require().True(haveSameElements(dbAccounts1, dbAccounts2, accounts.SameAccountsIncludingPosition))
}