2022-09-13 07:10:59 +00:00
|
|
|
package token
|
2021-09-09 14:28:54 +00:00
|
|
|
|
|
|
|
import (
|
2024-03-08 12:52:39 +00:00
|
|
|
"math/big"
|
2021-09-09 14:28:54 +00:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
|
2023-02-06 17:05:58 +00:00
|
|
|
"github.com/status-im/status-go/params"
|
2024-03-08 12:52:39 +00:00
|
|
|
"github.com/status-im/status-go/services/wallet/bigint"
|
2023-12-22 09:43:19 +00:00
|
|
|
"github.com/status-im/status-go/services/wallet/community"
|
2023-08-11 11:28:45 +00:00
|
|
|
"github.com/status-im/status-go/t/helpers"
|
|
|
|
"github.com/status-im/status-go/walletdatabase"
|
2021-09-09 14:28:54 +00:00
|
|
|
)
|
|
|
|
|
2022-10-25 14:50:32 +00:00
|
|
|
func setupTestTokenDB(t *testing.T) (*Manager, func()) {
|
2023-08-11 11:28:45 +00:00
|
|
|
db, err := helpers.SetupTestMemorySQLDB(walletdatabase.DbInitializer{})
|
2021-09-09 14:28:54 +00:00
|
|
|
require.NoError(t, err)
|
2023-09-11 14:44:43 +00:00
|
|
|
|
|
|
|
return &Manager{
|
2023-12-12 07:37:57 +00:00
|
|
|
db: db,
|
|
|
|
RPCClient: nil,
|
2024-01-19 15:57:04 +00:00
|
|
|
ContractMaker: nil,
|
2023-12-12 07:37:57 +00:00
|
|
|
networkManager: nil,
|
|
|
|
stores: nil,
|
|
|
|
communityTokensDB: nil,
|
2023-12-14 16:50:46 +00:00
|
|
|
communityManager: nil,
|
2023-09-11 14:44:43 +00:00
|
|
|
}, func() {
|
|
|
|
require.NoError(t, db.Close())
|
|
|
|
}
|
2021-09-09 14:28:54 +00:00
|
|
|
}
|
|
|
|
|
2023-12-12 07:37:57 +00:00
|
|
|
func upsertCommunityToken(t *testing.T, token *Token, manager *Manager) {
|
|
|
|
require.NotNil(t, token.CommunityData)
|
|
|
|
|
|
|
|
err := manager.UpsertCustom(*token)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Community ID is only discovered by calling contract, so must be updated manually
|
|
|
|
_, err = manager.db.Exec("UPDATE tokens SET community_id = ? WHERE address = ?", token.CommunityData.ID, token.Address)
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
2021-09-09 14:28:54 +00:00
|
|
|
func TestCustoms(t *testing.T) {
|
|
|
|
manager, stop := setupTestTokenDB(t)
|
|
|
|
defer stop()
|
|
|
|
|
2023-10-17 15:05:05 +00:00
|
|
|
rst, err := manager.GetCustoms(false)
|
2021-09-09 14:28:54 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Nil(t, rst)
|
|
|
|
|
|
|
|
token := Token{
|
|
|
|
Address: common.Address{1},
|
|
|
|
Name: "Zilliqa",
|
|
|
|
Symbol: "ZIL",
|
|
|
|
Decimals: 12,
|
|
|
|
ChainID: 777,
|
|
|
|
}
|
|
|
|
|
2022-10-25 14:25:08 +00:00
|
|
|
err = manager.UpsertCustom(token)
|
2021-09-09 14:28:54 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2023-10-17 15:05:05 +00:00
|
|
|
rst, err = manager.GetCustoms(false)
|
2021-09-09 14:28:54 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 1, len(rst))
|
|
|
|
require.Equal(t, token, *rst[0])
|
|
|
|
|
2022-10-25 14:25:08 +00:00
|
|
|
err = manager.DeleteCustom(777, token.Address)
|
2021-09-09 14:28:54 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2023-10-17 15:05:05 +00:00
|
|
|
rst, err = manager.GetCustoms(false)
|
2021-09-09 14:28:54 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 0, len(rst))
|
|
|
|
}
|
2023-02-06 17:05:58 +00:00
|
|
|
|
2023-12-12 07:37:57 +00:00
|
|
|
func TestCommunityTokens(t *testing.T) {
|
|
|
|
manager, stop := setupTestTokenDB(t)
|
|
|
|
defer stop()
|
|
|
|
|
|
|
|
rst, err := manager.GetCustoms(true)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Nil(t, rst)
|
|
|
|
|
|
|
|
token := Token{
|
|
|
|
Address: common.Address{1},
|
|
|
|
Name: "Zilliqa",
|
|
|
|
Symbol: "ZIL",
|
|
|
|
Decimals: 12,
|
|
|
|
ChainID: 777,
|
|
|
|
}
|
|
|
|
|
|
|
|
err = manager.UpsertCustom(token)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
communityToken := Token{
|
|
|
|
Address: common.Address{2},
|
|
|
|
Name: "Communitia",
|
|
|
|
Symbol: "COM",
|
|
|
|
Decimals: 12,
|
|
|
|
ChainID: 777,
|
2023-12-22 09:43:19 +00:00
|
|
|
CommunityData: &community.Data{
|
2023-12-12 07:37:57 +00:00
|
|
|
ID: "random_community_id",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
upsertCommunityToken(t, &communityToken, manager)
|
|
|
|
|
|
|
|
rst, err = manager.GetCustoms(false)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 2, len(rst))
|
|
|
|
require.Equal(t, token, *rst[0])
|
|
|
|
require.Equal(t, communityToken, *rst[1])
|
|
|
|
|
|
|
|
rst, err = manager.GetCustoms(true)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 1, len(rst))
|
|
|
|
require.Equal(t, communityToken, *rst[0])
|
|
|
|
}
|
|
|
|
|
2023-10-17 15:05:05 +00:00
|
|
|
func toTokenMap(tokens []*Token) storeMap {
|
|
|
|
tokenMap := storeMap{}
|
|
|
|
|
|
|
|
for _, token := range tokens {
|
|
|
|
addTokMap := tokenMap[token.ChainID]
|
|
|
|
if addTokMap == nil {
|
|
|
|
addTokMap = make(addressTokenMap)
|
|
|
|
}
|
|
|
|
|
|
|
|
addTokMap[token.Address] = token
|
|
|
|
tokenMap[token.ChainID] = addTokMap
|
|
|
|
}
|
|
|
|
|
|
|
|
return tokenMap
|
|
|
|
}
|
|
|
|
|
2023-02-06 17:05:58 +00:00
|
|
|
func TestTokenOverride(t *testing.T) {
|
|
|
|
networks := []params.Network{
|
|
|
|
{
|
|
|
|
ChainID: 1,
|
|
|
|
ChainName: "TestChain1",
|
|
|
|
TokenOverrides: []params.TokenOverride{
|
|
|
|
{
|
|
|
|
Symbol: "SNT",
|
|
|
|
Address: common.Address{11},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}, {
|
|
|
|
ChainID: 2,
|
|
|
|
ChainName: "TestChain2",
|
|
|
|
TokenOverrides: []params.TokenOverride{
|
|
|
|
{
|
|
|
|
Symbol: "STT",
|
|
|
|
Address: common.Address{33},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2023-03-14 17:33:05 +00:00
|
|
|
|
|
|
|
tokenList := []*Token{
|
|
|
|
&Token{
|
|
|
|
Address: common.Address{1},
|
|
|
|
Symbol: "SNT",
|
|
|
|
ChainID: 1,
|
2023-02-06 17:05:58 +00:00
|
|
|
},
|
2023-03-14 17:33:05 +00:00
|
|
|
&Token{
|
|
|
|
Address: common.Address{2},
|
|
|
|
Symbol: "TNT",
|
|
|
|
ChainID: 1,
|
|
|
|
},
|
|
|
|
&Token{
|
|
|
|
Address: common.Address{3},
|
|
|
|
Symbol: "STT",
|
|
|
|
ChainID: 2,
|
|
|
|
},
|
|
|
|
&Token{
|
|
|
|
Address: common.Address{4},
|
|
|
|
Symbol: "TTT",
|
|
|
|
ChainID: 2,
|
2023-02-06 17:05:58 +00:00
|
|
|
},
|
|
|
|
}
|
2023-03-14 17:33:05 +00:00
|
|
|
testStore := &DefaultStore{
|
|
|
|
tokenList,
|
|
|
|
}
|
|
|
|
|
2023-10-17 15:05:05 +00:00
|
|
|
overrideTokensInPlace(networks, tokenList)
|
2023-06-14 13:16:15 +00:00
|
|
|
tokens := testStore.GetTokens()
|
2023-03-14 17:33:05 +00:00
|
|
|
tokenMap := toTokenMap(tokens)
|
|
|
|
_, found := tokenMap[1][common.Address{1}]
|
2023-02-06 17:05:58 +00:00
|
|
|
require.False(t, found)
|
2023-03-14 17:33:05 +00:00
|
|
|
require.Equal(t, common.Address{11}, tokenMap[1][common.Address{11}].Address)
|
|
|
|
require.Equal(t, common.Address{2}, tokenMap[1][common.Address{2}].Address)
|
|
|
|
_, found = tokenMap[2][common.Address{3}]
|
2023-02-06 17:05:58 +00:00
|
|
|
require.False(t, found)
|
2023-03-14 17:33:05 +00:00
|
|
|
require.Equal(t, common.Address{33}, tokenMap[2][common.Address{33}].Address)
|
|
|
|
require.Equal(t, common.Address{4}, tokenMap[2][common.Address{4}].Address)
|
|
|
|
}
|
2023-12-21 14:12:50 +00:00
|
|
|
|
|
|
|
func TestMarkAsPreviouslyOwnedToken(t *testing.T) {
|
|
|
|
manager, stop := setupTestTokenDB(t)
|
|
|
|
defer stop()
|
|
|
|
|
|
|
|
owner := common.HexToAddress("0x1234567890abcdef")
|
|
|
|
token := &Token{
|
|
|
|
Address: common.HexToAddress("0xabcdef1234567890"),
|
|
|
|
Name: "TestToken",
|
|
|
|
Symbol: "TT",
|
|
|
|
Decimals: 18,
|
|
|
|
ChainID: 1,
|
|
|
|
}
|
|
|
|
|
2024-02-19 13:55:38 +00:00
|
|
|
isFirst, err := manager.MarkAsPreviouslyOwnedToken(nil, owner)
|
2023-12-21 14:12:50 +00:00
|
|
|
require.Error(t, err)
|
2024-02-19 13:55:38 +00:00
|
|
|
require.False(t, isFirst)
|
2023-12-21 14:12:50 +00:00
|
|
|
|
2024-02-19 13:55:38 +00:00
|
|
|
isFirst, err = manager.MarkAsPreviouslyOwnedToken(token, common.Address{})
|
2023-12-21 14:12:50 +00:00
|
|
|
require.Error(t, err)
|
2024-02-19 13:55:38 +00:00
|
|
|
require.False(t, isFirst)
|
2023-12-21 14:12:50 +00:00
|
|
|
|
2024-02-19 13:55:38 +00:00
|
|
|
isFirst, err = manager.MarkAsPreviouslyOwnedToken(token, owner)
|
2023-12-21 14:12:50 +00:00
|
|
|
require.NoError(t, err)
|
2024-02-19 13:55:38 +00:00
|
|
|
require.True(t, isFirst)
|
2023-12-21 14:12:50 +00:00
|
|
|
|
|
|
|
// Verify that the token balance was inserted correctly
|
|
|
|
var count int
|
|
|
|
err = manager.db.QueryRow(`SELECT count(*) FROM token_balances`).Scan(&count)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 1, count)
|
|
|
|
|
|
|
|
token.Name = "123"
|
|
|
|
|
2024-02-19 13:55:38 +00:00
|
|
|
isFirst, err = manager.MarkAsPreviouslyOwnedToken(token, owner)
|
2023-12-21 14:12:50 +00:00
|
|
|
require.NoError(t, err)
|
2024-02-19 13:55:38 +00:00
|
|
|
require.False(t, isFirst)
|
2023-12-21 14:12:50 +00:00
|
|
|
|
|
|
|
// Not updated because already exists
|
|
|
|
err = manager.db.QueryRow(`SELECT count(*) FROM token_balances`).Scan(&count)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 1, count)
|
|
|
|
|
|
|
|
token.ChainID = 2
|
|
|
|
|
2024-02-19 13:55:38 +00:00
|
|
|
isFirst, err = manager.MarkAsPreviouslyOwnedToken(token, owner)
|
2023-12-21 14:12:50 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Same token on different chains counts as different token
|
|
|
|
err = manager.db.QueryRow(`SELECT count(*) FROM token_balances`).Scan(&count)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 2, count)
|
2024-02-19 13:55:38 +00:00
|
|
|
require.True(t, isFirst)
|
2023-12-21 14:12:50 +00:00
|
|
|
}
|
2024-03-08 12:52:39 +00:00
|
|
|
|
|
|
|
func TestGetTokenHistoricalBalance(t *testing.T) {
|
|
|
|
manager, stop := setupTestTokenDB(t)
|
|
|
|
defer stop()
|
|
|
|
|
|
|
|
account := common.HexToAddress("0x1234567890abcdef")
|
|
|
|
chainID := uint64(1)
|
|
|
|
testSymbol := "TEST"
|
|
|
|
block := int64(1)
|
|
|
|
timestamp := int64(1629878400) // Replace with desired timestamp
|
|
|
|
historyBalance := big.NewInt(0)
|
|
|
|
|
|
|
|
// Test case when no rows are returned
|
|
|
|
balance, err := manager.GetTokenHistoricalBalance(account, chainID, testSymbol, timestamp)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Nil(t, balance)
|
|
|
|
|
|
|
|
// Test case when a row is returned
|
|
|
|
historyBalance.SetInt64(int64(100))
|
|
|
|
_, err = manager.db.Exec("INSERT INTO balance_history (currency, chain_id, address, timestamp, balance, block) VALUES (?, ?, ?, ?, ?, ?)", testSymbol, chainID, account, timestamp-100, (*bigint.SQLBigIntBytes)(historyBalance), block)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
expectedBalance := big.NewInt(100)
|
|
|
|
balance, err = manager.GetTokenHistoricalBalance(account, chainID, testSymbol, timestamp)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, expectedBalance, balance)
|
|
|
|
|
|
|
|
// Test multiple values. Must return the most recent one
|
|
|
|
historyBalance.SetInt64(int64(100))
|
|
|
|
_, err = manager.db.Exec("INSERT INTO balance_history (currency, chain_id, address, timestamp, balance, block) VALUES (?, ?, ?, ?, ?, ?)", testSymbol, chainID, account, timestamp-200, (*bigint.SQLBigIntBytes)(historyBalance), block)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
historyBalance.SetInt64(int64(50))
|
|
|
|
symbol := "TEST2"
|
|
|
|
_, err = manager.db.Exec("INSERT INTO balance_history (currency, chain_id, address, timestamp, balance, block) VALUES (?, ?, ?, ?, ?, ?)", symbol, chainID, account, timestamp-1, (*bigint.SQLBigIntBytes)(historyBalance), block)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
historyBalance.SetInt64(int64(50))
|
|
|
|
chainID = uint64(2)
|
|
|
|
_, err = manager.db.Exec("INSERT INTO balance_history (currency, chain_id, address, timestamp, balance, block) VALUES (?, ?, ?, ?, ?, ?)", testSymbol, chainID, account, timestamp-1, (*bigint.SQLBigIntBytes)(historyBalance), block)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
chainID = uint64(1)
|
|
|
|
balance, err = manager.GetTokenHistoricalBalance(account, chainID, testSymbol, timestamp)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, expectedBalance, balance)
|
|
|
|
}
|