288 lines
8.3 KiB
Go
288 lines
8.3 KiB
Go
package wallet
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/event"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
gomock "go.uber.org/mock/gomock"
|
|
|
|
"github.com/status-im/status-go/appdatabase"
|
|
"github.com/status-im/status-go/multiaccounts/accounts"
|
|
"github.com/status-im/status-go/params"
|
|
"github.com/status-im/status-go/rpc"
|
|
mock_reader "github.com/status-im/status-go/services/wallet/mock/reader"
|
|
"github.com/status-im/status-go/services/wallet/onramp"
|
|
mock_onramp "github.com/status-im/status-go/services/wallet/onramp/mock"
|
|
"github.com/status-im/status-go/services/wallet/requests"
|
|
"github.com/status-im/status-go/services/wallet/token"
|
|
"github.com/status-im/status-go/services/wallet/walletconnect"
|
|
"github.com/status-im/status-go/t/helpers"
|
|
"github.com/status-im/status-go/walletdatabase"
|
|
)
|
|
|
|
// TestAPI_GetWalletConnectActiveSessions tames coverage
|
|
func TestAPI_GetWalletConnectActiveSessions(t *testing.T) {
|
|
db, close := walletconnect.SetupTestDB(t)
|
|
defer close()
|
|
api := &API{
|
|
s: &Service{db: db},
|
|
}
|
|
|
|
sessions, err := api.GetWalletConnectActiveSessions(context.Background(), 0)
|
|
require.NoError(t, err)
|
|
require.Equal(t, 0, len(sessions))
|
|
}
|
|
|
|
// TestAPI_HashMessageEIP191
|
|
func TestAPI_HashMessageEIP191(t *testing.T) {
|
|
api := &API{}
|
|
|
|
res := api.HashMessageEIP191(context.Background(), []byte("test"))
|
|
require.Equal(t, "0x4a5c5d454721bbbb25540c3317521e71c373ae36458f960d2ad46ef088110e95", res.String())
|
|
}
|
|
|
|
func TestAPI_IsChecksumValidForAddress(t *testing.T) {
|
|
api := &API{}
|
|
|
|
res, err := api.IsChecksumValidForAddress("0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa")
|
|
require.NoError(t, err)
|
|
require.False(t, res)
|
|
|
|
res, err = api.IsChecksumValidForAddress("0xaAaAaAaaAaAaAaaAaAAAAAAAAaaaAaAaAaaAaaAa")
|
|
require.NoError(t, err)
|
|
require.True(t, res)
|
|
}
|
|
|
|
func TestAPI_GetCryptoOnRamps(t *testing.T) {
|
|
ctrl := gomock.NewController(t)
|
|
defer ctrl.Finish()
|
|
|
|
provider0 := mock_onramp.NewMockProvider(ctrl)
|
|
id0 := "provider0"
|
|
provider0.EXPECT().ID().Return(id0).AnyTimes()
|
|
provider1 := mock_onramp.NewMockProvider(ctrl)
|
|
id1 := "provider1"
|
|
provider1.EXPECT().ID().Return(id1).AnyTimes()
|
|
providers := []onramp.Provider{provider0, provider1}
|
|
onrampManager := onramp.NewManager(providers)
|
|
|
|
api := &API{
|
|
s: &Service{cryptoOnRampManager: onrampManager},
|
|
}
|
|
|
|
ctx := context.Background()
|
|
|
|
// Check returned providers
|
|
provider0.EXPECT().GetCryptoOnRamp(ctx).Return(onramp.CryptoOnRamp{ID: id0}, nil)
|
|
provider1.EXPECT().GetCryptoOnRamp(ctx).Return(onramp.CryptoOnRamp{ID: id1}, nil)
|
|
|
|
retProviders, err := api.GetCryptoOnRamps(ctx)
|
|
require.NoError(t, err)
|
|
require.Equal(t, len(providers), len(retProviders))
|
|
require.Equal(t, id0, retProviders[0].ID)
|
|
require.Equal(t, id1, retProviders[1].ID)
|
|
|
|
// Check error handling
|
|
provider0.EXPECT().GetCryptoOnRamp(ctx).Return(onramp.CryptoOnRamp{}, errors.New("error"))
|
|
provider1.EXPECT().GetCryptoOnRamp(ctx).Return(onramp.CryptoOnRamp{ID: id1}, nil)
|
|
retProviders, err = api.GetCryptoOnRamps(ctx)
|
|
require.NoError(t, err)
|
|
require.Equal(t, 1, len(retProviders))
|
|
require.Equal(t, id1, retProviders[0].ID)
|
|
|
|
// Check URL retrieval
|
|
provider1.EXPECT().GetURL(ctx, onramp.Parameters{}).Return("url", nil)
|
|
url, err := api.GetCryptoOnRampURL(ctx, id1, onramp.Parameters{})
|
|
require.NoError(t, err)
|
|
require.Equal(t, "url", url)
|
|
}
|
|
|
|
func TestAPI_GetAddressDetails(t *testing.T) {
|
|
appDB, err := helpers.SetupTestMemorySQLDB(appdatabase.DbInitializer{})
|
|
require.NoError(t, err)
|
|
defer appDB.Close()
|
|
|
|
accountsDb, err := accounts.NewDB(appDB)
|
|
require.NoError(t, err)
|
|
defer accountsDb.Close()
|
|
|
|
db, err := helpers.SetupTestMemorySQLDB(walletdatabase.DbInitializer{})
|
|
require.NoError(t, err)
|
|
defer db.Close()
|
|
|
|
accountFeed := &event.Feed{}
|
|
|
|
chainID := uint64(1)
|
|
address := "0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"
|
|
|
|
providerConfig := params.ProviderConfig{
|
|
Enabled: true,
|
|
Name: rpc.ProviderStatusProxy,
|
|
User: "user1",
|
|
Password: "pass1",
|
|
}
|
|
providerConfigs := []params.ProviderConfig{providerConfig}
|
|
|
|
// Create a new server that delays the response by 1 second
|
|
serverWith1SecDelay := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
time.Sleep(1 * time.Second)
|
|
fmt.Fprintln(w, `{"result": "0x10"}`)
|
|
}))
|
|
defer serverWith1SecDelay.Close()
|
|
|
|
networks := []params.Network{
|
|
{
|
|
ChainID: chainID,
|
|
DefaultRPCURL: serverWith1SecDelay.URL + "/nodefleet/",
|
|
},
|
|
}
|
|
config := rpc.ClientConfig{
|
|
Client: nil,
|
|
UpstreamChainID: chainID,
|
|
Networks: networks,
|
|
DB: appDB,
|
|
WalletFeed: nil,
|
|
ProviderConfigs: providerConfigs,
|
|
}
|
|
c, err := rpc.NewClient(config)
|
|
require.NoError(t, err)
|
|
|
|
chainClient, err := c.EthClient(chainID)
|
|
require.NoError(t, err)
|
|
chainClient.SetWalletNotifier(func(chainID uint64, message string) {})
|
|
c.SetWalletNotifier(func(chainID uint64, message string) {})
|
|
|
|
service := NewService(db, accountsDb, appDB, c, accountFeed, nil, nil, nil, ¶ms.NodeConfig{}, nil, nil, nil, nil, "")
|
|
|
|
api := &API{
|
|
s: service,
|
|
}
|
|
|
|
// Test getting address details using `GetAddressDetails` call, that always waits for the request to finish
|
|
details, err := api.GetAddressDetails(context.Background(), 1, address)
|
|
require.NoError(t, err)
|
|
require.Equal(t, true, details.HasActivity)
|
|
|
|
// empty params
|
|
details, err = api.AddressDetails(context.Background(), &requests.AddressDetails{})
|
|
require.Error(t, err)
|
|
require.ErrorIs(t, err, requests.ErrAddresInvalid)
|
|
require.Nil(t, details)
|
|
|
|
// no response longer than the set timeout
|
|
details, err = api.AddressDetails(context.Background(), &requests.AddressDetails{
|
|
Address: address,
|
|
TimeoutInMilliseconds: 500,
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, false, details.HasActivity)
|
|
|
|
// timeout longer than the response time
|
|
details, err = api.AddressDetails(context.Background(), &requests.AddressDetails{
|
|
Address: address,
|
|
TimeoutInMilliseconds: 1200,
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, true, details.HasActivity)
|
|
|
|
// specific chain and timeout longer than the response time
|
|
details, err = api.AddressDetails(context.Background(), &requests.AddressDetails{
|
|
Address: address,
|
|
ChainIDs: []uint64{chainID},
|
|
TimeoutInMilliseconds: 1200,
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, true, details.HasActivity)
|
|
}
|
|
|
|
// TestAPI_FetchOrGetCachedWalletBalances
|
|
func TestAPI_FetchOrGetCachedWalletBalances(t *testing.T) {
|
|
appDB, err := helpers.SetupTestMemorySQLDB(appdatabase.DbInitializer{})
|
|
require.NoError(t, err)
|
|
defer appDB.Close()
|
|
|
|
mockCtrl := gomock.NewController(t)
|
|
defer mockCtrl.Finish()
|
|
|
|
mockReader := mock_reader.NewMockReaderInterface(mockCtrl)
|
|
providerConfig := params.ProviderConfig{
|
|
Enabled: true,
|
|
Name: rpc.ProviderStatusProxy,
|
|
User: "user1",
|
|
Password: "pass1",
|
|
}
|
|
providerConfigs := []params.ProviderConfig{providerConfig}
|
|
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
fmt.Fprintln(w, `{"result": "0x10"}`)
|
|
}))
|
|
defer server.Close()
|
|
|
|
chainID := uint64(1)
|
|
|
|
networks := []params.Network{
|
|
{
|
|
ChainID: chainID,
|
|
DefaultRPCURL: server.URL + "/nodefleet/",
|
|
},
|
|
}
|
|
config := rpc.ClientConfig{
|
|
Client: nil,
|
|
UpstreamChainID: chainID,
|
|
Networks: networks,
|
|
DB: appDB,
|
|
WalletFeed: nil,
|
|
ProviderConfigs: providerConfigs,
|
|
}
|
|
c, err := rpc.NewClient(config)
|
|
require.NoError(t, err)
|
|
|
|
testTokenAddress1 := common.Address{0x34}
|
|
testAccAddress1 := common.Address{0x12}
|
|
storageToken := token.StorageToken{
|
|
Token: token.Token{
|
|
Name: "USD Tether",
|
|
Symbol: "USDT",
|
|
Decimals: 18,
|
|
},
|
|
BalancesPerChain: map[uint64]token.ChainBalance{
|
|
1: {
|
|
RawBalance: "1000000000000000000",
|
|
Balance: nil,
|
|
Address: testAccAddress1,
|
|
ChainID: 1,
|
|
HasError: false,
|
|
},
|
|
},
|
|
}
|
|
expectedTokens := map[common.Address][]token.StorageToken{
|
|
testAccAddress1: {storageToken},
|
|
}
|
|
|
|
mockReader.EXPECT().FetchOrGetCachedWalletBalances(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(expectedTokens, nil)
|
|
|
|
service := &Service{
|
|
rpcClient: c,
|
|
}
|
|
|
|
api := &API{
|
|
s: service,
|
|
reader: mockReader,
|
|
}
|
|
|
|
forceRefresh := true
|
|
balances, err := api.FetchOrGetCachedWalletBalances(context.Background(), []common.Address{testTokenAddress1}, forceRefresh)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, balances)
|
|
}
|