2018-04-20 15:39:53 +00:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
2019-12-05 08:00:57 +00:00
|
|
|
"crypto/sha256"
|
2019-01-24 15:44:46 +00:00
|
|
|
"encoding/hex"
|
2019-03-28 14:56:21 +00:00
|
|
|
"encoding/json"
|
2018-04-20 15:39:53 +00:00
|
|
|
"fmt"
|
2020-12-10 13:38:58 +00:00
|
|
|
"github.com/status-im/status-go/account/generator"
|
2019-08-29 08:06:22 +00:00
|
|
|
"io/ioutil"
|
2018-04-20 15:39:53 +00:00
|
|
|
"math/rand"
|
2019-08-29 08:06:22 +00:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2018-04-20 15:39:53 +00:00
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
|
2020-01-02 09:10:19 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
2019-12-11 13:59:37 +00:00
|
|
|
gethcrypto "github.com/ethereum/go-ethereum/crypto"
|
2020-01-02 09:10:19 +00:00
|
|
|
|
2019-01-24 15:44:46 +00:00
|
|
|
"github.com/status-im/status-go/account"
|
2019-12-11 13:59:37 +00:00
|
|
|
"github.com/status-im/status-go/eth-node/crypto"
|
2019-11-23 17:57:05 +00:00
|
|
|
"github.com/status-im/status-go/eth-node/types"
|
2019-08-29 08:06:22 +00:00
|
|
|
"github.com/status-im/status-go/multiaccounts"
|
|
|
|
"github.com/status-im/status-go/multiaccounts/accounts"
|
2018-06-08 11:29:50 +00:00
|
|
|
"github.com/status-im/status-go/node"
|
|
|
|
"github.com/status-im/status-go/params"
|
2018-08-02 07:07:55 +00:00
|
|
|
"github.com/status-im/status-go/rpc"
|
2019-03-28 14:56:21 +00:00
|
|
|
"github.com/status-im/status-go/services/typeddata"
|
2018-09-13 16:31:29 +00:00
|
|
|
"github.com/status-im/status-go/t/utils"
|
2019-08-29 08:06:22 +00:00
|
|
|
"github.com/status-im/status-go/transactions"
|
2018-04-20 15:39:53 +00:00
|
|
|
)
|
|
|
|
|
2019-12-27 09:58:25 +00:00
|
|
|
var (
|
|
|
|
networks = json.RawMessage("{}")
|
|
|
|
settings = accounts.Settings{
|
|
|
|
Address: types.HexToAddress("0xeC540f3745Ff2964AFC1171a5A0DD726d1F6B472"),
|
|
|
|
CurrentNetwork: "mainnet_rpc",
|
|
|
|
DappsAddress: types.HexToAddress("0xe1300f99fDF7346986CbC766903245087394ecd0"),
|
|
|
|
EIP1581Address: types.HexToAddress("0xe1DDDE9235a541d1344550d969715CF43982de9f"),
|
|
|
|
InstallationID: "d3efcff6-cffa-560e-a547-21d3858cbc51",
|
|
|
|
KeyUID: "0x4e8129f3edfc004875be17bf468a784098a9f69b53c095be1f52deff286935ab",
|
|
|
|
LatestDerivedPath: 0,
|
|
|
|
Name: "Jittery Cornflowerblue Kingbird",
|
|
|
|
Networks: &networks,
|
|
|
|
PhotoPath: "",
|
|
|
|
PreviewPrivacy: false,
|
|
|
|
PublicKey: "0x04211fe0f69772ecf7eb0b5bfc7678672508a9fb01f2d699096f0d59ef7fe1a0cb1e648a80190db1c0f5f088872444d846f2956d0bd84069f3f9f69335af852ac0",
|
|
|
|
SigningPhrase: "yurt joey vibe",
|
|
|
|
WalletRootAddress: types.HexToAddress("0xeB591fd819F86D0A6a2EF2Bcb94f77807a7De1a6")}
|
|
|
|
)
|
|
|
|
|
2018-04-20 15:39:53 +00:00
|
|
|
func TestBackendStartNodeConcurrently(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2018-09-13 16:31:29 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
2018-04-20 15:39:53 +00:00
|
|
|
count := 2
|
|
|
|
resultCh := make(chan error)
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(count)
|
|
|
|
|
|
|
|
for i := 0; i < count; i++ {
|
|
|
|
go func() {
|
2018-09-13 16:31:29 +00:00
|
|
|
resultCh <- backend.StartNode(config)
|
2018-04-20 15:39:53 +00:00
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
// close channel as otherwise for loop never finishes
|
|
|
|
go func() { wg.Wait(); close(resultCh) }()
|
|
|
|
|
|
|
|
var results []error
|
|
|
|
for err := range resultCh {
|
|
|
|
results = append(results, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
require.Contains(t, results, nil)
|
|
|
|
require.Contains(t, results, node.ErrNodeRunning)
|
|
|
|
|
2018-09-13 16:31:29 +00:00
|
|
|
err = backend.StopNode()
|
2018-04-20 15:39:53 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestBackendRestartNodeConcurrently(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2018-09-13 16:31:29 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2018-04-20 15:39:53 +00:00
|
|
|
count := 3
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
|
|
|
require.NoError(t, backend.StartNode(config))
|
2018-04-20 15:39:53 +00:00
|
|
|
defer func() {
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
}()
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(count)
|
|
|
|
|
|
|
|
for i := 0; i < count; i++ {
|
|
|
|
go func(idx int) {
|
|
|
|
assert.NoError(t, backend.RestartNode())
|
|
|
|
wg.Done()
|
|
|
|
}(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(adam): add concurrent tests for ResetChainData()
|
|
|
|
|
|
|
|
func TestBackendGettersConcurrently(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2018-09-13 16:31:29 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
2018-09-13 16:31:29 +00:00
|
|
|
err = backend.StartNode(config)
|
2018-04-20 15:39:53 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
defer func() {
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
}()
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
assert.NotNil(t, backend.StatusNode())
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
assert.NotNil(t, backend.AccountManager())
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
2018-06-27 08:11:45 +00:00
|
|
|
assert.NotNil(t, backend.personalAPI)
|
2018-04-20 15:39:53 +00:00
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
assert.NotNil(t, backend.Transactor())
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
assert.True(t, backend.IsNodeRunning())
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
assert.True(t, backend.IsNodeRunning())
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestBackendAccountsConcurrently(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2018-09-13 16:31:29 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
2018-09-13 16:31:29 +00:00
|
|
|
err = backend.StartNode(config)
|
2018-04-20 15:39:53 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
defer func() {
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
}()
|
|
|
|
|
|
|
|
var wgCreateAccounts sync.WaitGroup
|
|
|
|
count := 3
|
2020-12-10 13:38:58 +00:00
|
|
|
type AccountData struct {
|
|
|
|
MasterAccount generator.GeneratedAccountInfo
|
|
|
|
AccountInfo account.Info
|
|
|
|
Password string
|
|
|
|
}
|
|
|
|
addressCh := make(chan AccountData, count) // use buffered channel to avoid blocking
|
2018-04-20 15:39:53 +00:00
|
|
|
|
|
|
|
// create new accounts concurrently
|
|
|
|
for i := 0; i < count; i++ {
|
|
|
|
wgCreateAccounts.Add(1)
|
|
|
|
go func(pass string) {
|
2020-12-10 13:38:58 +00:00
|
|
|
MKInfo, accountInfo, _, err := backend.AccountManager().CreateAccount(pass)
|
2018-04-20 15:39:53 +00:00
|
|
|
assert.NoError(t, err)
|
2020-12-10 13:38:58 +00:00
|
|
|
addressCh <- AccountData{MKInfo, accountInfo, pass}
|
2018-04-20 15:39:53 +00:00
|
|
|
wgCreateAccounts.Done()
|
2020-09-25 07:31:55 +00:00
|
|
|
}("password-00" + fmt.Sprint(i))
|
2018-04-20 15:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// close addressCh as otherwise for loop never finishes
|
|
|
|
go func() { wgCreateAccounts.Wait(); close(addressCh) }()
|
|
|
|
|
|
|
|
// select, reselect or logout concurrently
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
2020-12-10 13:38:58 +00:00
|
|
|
for accountData := range addressCh {
|
2018-04-20 15:39:53 +00:00
|
|
|
wg.Add(1)
|
2020-12-10 13:38:58 +00:00
|
|
|
go func(accountData AccountData) {
|
2019-07-26 14:45:10 +00:00
|
|
|
loginParams := account.LoginParams{
|
2020-12-10 13:38:58 +00:00
|
|
|
MainAccount: types.HexToAddress(accountData.AccountInfo.WalletAddress),
|
|
|
|
ChatAddress: types.HexToAddress(accountData.AccountInfo.ChatAddress),
|
|
|
|
Password: accountData.Password,
|
|
|
|
MultiAccount: accountData.MasterAccount.ToMultiAccount(),
|
2019-07-26 14:45:10 +00:00
|
|
|
}
|
|
|
|
assert.NoError(t, backend.SelectAccount(loginParams))
|
2018-04-20 15:39:53 +00:00
|
|
|
wg.Done()
|
2020-12-10 13:38:58 +00:00
|
|
|
}(accountData)
|
2018-04-20 15:39:53 +00:00
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
assert.NoError(t, backend.Logout())
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
}
|
|
|
|
|
2019-01-24 15:44:46 +00:00
|
|
|
func TestBackendInjectChatAccount(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2019-01-24 15:44:46 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
2019-01-24 15:44:46 +00:00
|
|
|
err = backend.StartNode(config)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer func() {
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
}()
|
|
|
|
|
2020-12-10 13:38:58 +00:00
|
|
|
backend.account = &multiaccounts.Account{KeyUID: "0xdeadbeef"}
|
|
|
|
|
2019-12-11 13:59:37 +00:00
|
|
|
chatPrivKey, err := gethcrypto.GenerateKey()
|
2019-01-24 15:44:46 +00:00
|
|
|
require.NoError(t, err)
|
2019-12-11 13:59:37 +00:00
|
|
|
encryptionPrivKey, err := gethcrypto.GenerateKey()
|
2019-01-24 15:44:46 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2019-12-11 13:59:37 +00:00
|
|
|
chatPrivKeyHex := hex.EncodeToString(gethcrypto.FromECDSA(chatPrivKey))
|
|
|
|
chatPubKeyHex := types.EncodeHex(gethcrypto.FromECDSAPub(&chatPrivKey.PublicKey))
|
|
|
|
encryptionPrivKeyHex := hex.EncodeToString(gethcrypto.FromECDSA(encryptionPrivKey))
|
2019-01-24 15:44:46 +00:00
|
|
|
|
|
|
|
whisperService, err := backend.StatusNode().WhisperService()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// public key should not be already in whisper
|
|
|
|
require.False(t, whisperService.HasKeyPair(chatPubKeyHex), "identity already present in whisper")
|
|
|
|
|
|
|
|
// call InjectChatAccount
|
|
|
|
require.NoError(t, backend.InjectChatAccount(chatPrivKeyHex, encryptionPrivKeyHex))
|
|
|
|
|
|
|
|
// public key should now be in whisper
|
|
|
|
require.True(t, whisperService.HasKeyPair(chatPubKeyHex), "identity not injected into whisper")
|
|
|
|
|
|
|
|
// wallet account should not be selected
|
2019-07-26 14:45:10 +00:00
|
|
|
mainAccountAddress, err := backend.AccountManager().MainAccountAddress()
|
2019-12-11 13:59:37 +00:00
|
|
|
require.Equal(t, types.Address{}, mainAccountAddress)
|
2019-01-24 15:44:46 +00:00
|
|
|
require.Equal(t, account.ErrNoAccountSelected, err)
|
|
|
|
|
|
|
|
// selected chat account should have the key injected previously
|
|
|
|
chatAcc, err := backend.AccountManager().SelectedChatAccount()
|
|
|
|
require.Nil(t, err)
|
|
|
|
require.Equal(t, chatPrivKey, chatAcc.AccountKey.PrivateKey)
|
|
|
|
}
|
|
|
|
|
2018-04-20 15:39:53 +00:00
|
|
|
func TestBackendConnectionChangesConcurrently(t *testing.T) {
|
2018-06-27 08:11:45 +00:00
|
|
|
connections := [...]string{wifi, cellular, unknown}
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2018-04-20 15:39:53 +00:00
|
|
|
count := 3
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
for i := 0; i < count; i++ {
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
connIdx := rand.Intn(len(connections))
|
2018-04-23 13:34:49 +00:00
|
|
|
backend.ConnectionChange(connections[connIdx], false)
|
2018-04-20 15:39:53 +00:00
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
}
|
|
|
|
|
2018-06-27 08:11:45 +00:00
|
|
|
func TestBackendConnectionChangesToOffline(t *testing.T) {
|
2019-11-23 17:57:05 +00:00
|
|
|
b := NewGethStatusBackend()
|
2018-06-27 08:11:45 +00:00
|
|
|
b.ConnectionChange(none, false)
|
|
|
|
assert.True(t, b.connectionState.Offline)
|
|
|
|
|
|
|
|
b.ConnectionChange(wifi, false)
|
|
|
|
assert.False(t, b.connectionState.Offline)
|
|
|
|
|
|
|
|
b.ConnectionChange("unknown-state", false)
|
|
|
|
assert.False(t, b.connectionState.Offline)
|
|
|
|
}
|
|
|
|
|
2018-04-20 15:39:53 +00:00
|
|
|
func TestBackendCallRPCConcurrently(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2018-09-13 16:31:29 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
2018-04-20 15:39:53 +00:00
|
|
|
count := 3
|
|
|
|
|
2018-09-13 16:31:29 +00:00
|
|
|
err = backend.StartNode(config)
|
2018-04-20 15:39:53 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
defer func() {
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
}()
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
for i := 0; i < count; i++ {
|
|
|
|
wg.Add(1)
|
|
|
|
go func(idx int) {
|
2018-12-20 08:31:17 +00:00
|
|
|
result, err := backend.CallRPC(fmt.Sprintf(
|
2018-04-20 15:39:53 +00:00
|
|
|
`{"jsonrpc":"2.0","method":"web3_clientVersion","params":[],"id":%d}`,
|
|
|
|
idx+1,
|
|
|
|
))
|
2018-12-20 08:31:17 +00:00
|
|
|
assert.NoError(t, err)
|
2018-04-20 15:39:53 +00:00
|
|
|
assert.NotContains(t, result, "error")
|
|
|
|
wg.Done()
|
|
|
|
}(i)
|
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
go func(idx int) {
|
2018-12-20 08:31:17 +00:00
|
|
|
result, err := backend.CallPrivateRPC(fmt.Sprintf(
|
2018-04-20 15:39:53 +00:00
|
|
|
`{"jsonrpc":"2.0","method":"web3_clientVersion","params":[],"id":%d}`,
|
|
|
|
idx+1,
|
|
|
|
))
|
2018-12-20 08:31:17 +00:00
|
|
|
assert.NoError(t, err)
|
2018-04-20 15:39:53 +00:00
|
|
|
assert.NotContains(t, result, "error")
|
|
|
|
wg.Done()
|
|
|
|
}(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
}
|
|
|
|
|
2018-06-27 08:11:45 +00:00
|
|
|
func TestAppStateChange(t *testing.T) {
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2018-06-27 08:11:45 +00:00
|
|
|
|
|
|
|
var testCases = []struct {
|
|
|
|
name string
|
|
|
|
fromState appState
|
|
|
|
toState appState
|
|
|
|
expectedState appState
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "success",
|
|
|
|
fromState: appStateInactive,
|
|
|
|
toState: appStateBackground,
|
|
|
|
expectedState: appStateBackground,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "invalid state",
|
|
|
|
fromState: appStateInactive,
|
|
|
|
toState: "unexisting",
|
|
|
|
expectedState: appStateInactive,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
|
|
backend.appState = tc.fromState
|
|
|
|
backend.AppStateChange(tc.toState.String())
|
|
|
|
assert.Equal(t, tc.expectedState.String(), backend.appState.String())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-02 07:07:55 +00:00
|
|
|
func TestBlockedRPCMethods(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2018-09-13 16:31:29 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
2018-09-13 16:31:29 +00:00
|
|
|
err = backend.StartNode(config)
|
2018-08-02 07:07:55 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
defer func() { require.NoError(t, backend.StopNode()) }()
|
|
|
|
|
|
|
|
for idx, m := range rpc.BlockedMethods() {
|
2018-12-20 08:31:17 +00:00
|
|
|
result, err := backend.CallRPC(fmt.Sprintf(
|
2018-08-02 07:07:55 +00:00
|
|
|
`{"jsonrpc":"2.0","method":"%s","params":[],"id":%d}`,
|
|
|
|
m,
|
|
|
|
idx+1,
|
|
|
|
))
|
2018-12-20 08:31:17 +00:00
|
|
|
assert.NoError(t, err)
|
2018-08-02 07:07:55 +00:00
|
|
|
assert.Contains(t, result, fmt.Sprintf(`{"code":-32700,"message":"%s"}`, rpc.ErrMethodNotFound))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-20 08:31:17 +00:00
|
|
|
func TestCallRPCWithStoppedNode(t *testing.T) {
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2018-12-20 08:31:17 +00:00
|
|
|
|
|
|
|
resp, err := backend.CallRPC(
|
|
|
|
`{"jsonrpc":"2.0","method":"web3_clientVersion","params":[],"id":1}`,
|
|
|
|
)
|
|
|
|
assert.Equal(t, ErrRPCClientUnavailable, err)
|
|
|
|
assert.Equal(t, "", resp)
|
|
|
|
|
|
|
|
resp, err = backend.CallPrivateRPC(
|
|
|
|
`{"jsonrpc":"2.0","method":"web3_clientVersion","params":[],"id":1}`,
|
|
|
|
)
|
|
|
|
assert.Equal(t, ErrRPCClientUnavailable, err)
|
|
|
|
assert.Equal(t, "", resp)
|
|
|
|
}
|
|
|
|
|
2018-08-16 11:37:53 +00:00
|
|
|
// TODO(adam): add concurrent tests for: SendTransaction
|
2019-01-17 11:02:45 +00:00
|
|
|
|
|
|
|
func TestStartStopMultipleTimes(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2019-01-17 11:02:45 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
2019-01-17 11:02:45 +00:00
|
|
|
config.NoDiscovery = false
|
|
|
|
// doesn't have to be running. just any valid enode to bypass validation.
|
|
|
|
config.ClusterConfig.BootNodes = []string{
|
|
|
|
"enode://e8a7c03b58911e98bbd66accb2a55d57683f35b23bf9dfca89e5e244eb5cc3f25018b4112db507faca34fb69ffb44b362f79eda97a669a8df29c72e654416784@0.0.0.0:30404",
|
|
|
|
}
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, backend.StartNode(config))
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
require.NoError(t, backend.StartNode(config))
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
}
|
2019-03-28 14:56:21 +00:00
|
|
|
|
2019-04-18 13:52:08 +00:00
|
|
|
func TestSignHash(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2019-04-18 13:52:08 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
2019-04-18 13:52:08 +00:00
|
|
|
|
|
|
|
require.NoError(t, backend.StartNode(config))
|
|
|
|
defer func() {
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
}()
|
|
|
|
|
2020-12-10 13:38:58 +00:00
|
|
|
backend.account = &multiaccounts.Account{KeyUID: "0xdeadbeef"}
|
|
|
|
|
2019-04-18 13:52:08 +00:00
|
|
|
var testCases = []struct {
|
|
|
|
name string
|
|
|
|
chatPrivKeyHex string
|
|
|
|
hashHex string
|
|
|
|
expectedSignatureHex string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "tc1",
|
|
|
|
chatPrivKeyHex: "facadefacadefacadefacadefacadefacadefacadefacadefacadefacadefaca",
|
|
|
|
hashHex: "0xa4735de5193362fe856416000105cdfa6ce56265607311cebae93b26e5adf438",
|
|
|
|
expectedSignatureHex: "0x176c971bae188c663614fc535ac9dbf62871dfeaadb38645809a510d28b3c4b0245415d5547c1b27f7cfea3341564f9c6981421144d3606b455346be69bd078c01",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
const dummyEncKey = "facadefacadefacadefacadefacadefacadefacadefacadefacadefacadefaca"
|
|
|
|
|
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
|
|
require.NoError(t, backend.InjectChatAccount(tc.chatPrivKeyHex, dummyEncKey))
|
|
|
|
|
|
|
|
signature, err := backend.SignHash(tc.hashHex)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, signature, tc.expectedSignatureHex)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-28 14:56:21 +00:00
|
|
|
func TestHashTypedData(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2019-03-28 14:56:21 +00:00
|
|
|
config, err := utils.MakeTestNodeConfig(params.StatusChainNetworkID)
|
|
|
|
require.NoError(t, err)
|
2019-08-20 15:38:40 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(config.KeyStoreDir))
|
2019-03-28 14:56:21 +00:00
|
|
|
err = backend.StartNode(config)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer func() {
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
}()
|
|
|
|
|
|
|
|
eip712Domain := "EIP712Domain"
|
|
|
|
mytypes := typeddata.Types{
|
|
|
|
eip712Domain: []typeddata.Field{
|
|
|
|
{Name: "name", Type: "string"},
|
|
|
|
{Name: "version", Type: "string"},
|
|
|
|
{Name: "chainId", Type: "uint256"},
|
|
|
|
{Name: "verifyingContract", Type: "address"},
|
|
|
|
},
|
|
|
|
"Text": []typeddata.Field{
|
|
|
|
{Name: "body", Type: "string"},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
domain := map[string]json.RawMessage{
|
|
|
|
"name": json.RawMessage(`"Ether Text"`),
|
|
|
|
"version": json.RawMessage(`"1"`),
|
|
|
|
"chainId": json.RawMessage(fmt.Sprintf("%d", params.StatusChainNetworkID)),
|
|
|
|
"verifyingContract": json.RawMessage(`"0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC"`),
|
|
|
|
}
|
|
|
|
msg := map[string]json.RawMessage{
|
|
|
|
"body": json.RawMessage(`"Hello, Bob!"`),
|
|
|
|
}
|
|
|
|
|
|
|
|
typed := typeddata.TypedData{
|
|
|
|
Types: mytypes,
|
|
|
|
PrimaryType: "Text",
|
|
|
|
Domain: domain,
|
|
|
|
Message: msg,
|
|
|
|
}
|
|
|
|
|
|
|
|
hash, err := backend.HashTypedData(typed)
|
|
|
|
require.NoError(t, err)
|
2019-11-23 17:57:05 +00:00
|
|
|
assert.NotEqual(t, types.Hash{}, hash)
|
2019-03-28 14:56:21 +00:00
|
|
|
}
|
2019-08-29 08:06:22 +00:00
|
|
|
|
|
|
|
func TestBackendGetVerifiedAccount(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-08-29 08:06:22 +00:00
|
|
|
password := "test"
|
|
|
|
tmpdir, err := ioutil.TempDir("", "verified-account-test-")
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer os.Remove(tmpdir)
|
2019-11-23 17:57:05 +00:00
|
|
|
backend := NewGethStatusBackend()
|
2019-08-29 08:06:22 +00:00
|
|
|
backend.UpdateRootDataDir(tmpdir)
|
2019-11-23 17:57:05 +00:00
|
|
|
require.NoError(t, backend.AccountManager().InitKeystore(filepath.Join(tmpdir, "keystore")))
|
2019-12-05 08:00:57 +00:00
|
|
|
require.NoError(t, backend.ensureAppDBOpened(multiaccounts.Account{KeyUID: "0x1"}, password))
|
2019-08-29 08:06:22 +00:00
|
|
|
config, err := params.NewNodeConfig(tmpdir, 178733)
|
|
|
|
require.NoError(t, err)
|
|
|
|
// this is for StatusNode().Config() call inside of the getVerifiedWalletAccount
|
|
|
|
require.NoError(t, backend.StartNode(config))
|
|
|
|
defer func() {
|
|
|
|
require.NoError(t, backend.StopNode())
|
|
|
|
}()
|
|
|
|
|
|
|
|
t.Run("AccountDoesntExist", func(t *testing.T) {
|
2019-12-11 13:59:37 +00:00
|
|
|
pkey, err := gethcrypto.GenerateKey()
|
2019-08-29 08:06:22 +00:00
|
|
|
require.NoError(t, err)
|
2019-12-11 13:59:37 +00:00
|
|
|
address := gethcrypto.PubkeyToAddress(pkey.PublicKey)
|
2019-08-29 08:06:22 +00:00
|
|
|
key, err := backend.getVerifiedWalletAccount(address.String(), password)
|
|
|
|
require.EqualError(t, err, transactions.ErrAccountDoesntExist.Error())
|
|
|
|
require.Nil(t, key)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("PasswordDoesntMatch", func(t *testing.T) {
|
|
|
|
pkey, err := crypto.GenerateKey()
|
|
|
|
require.NoError(t, err)
|
|
|
|
address := crypto.PubkeyToAddress(pkey.PublicKey)
|
|
|
|
db := accounts.NewDB(backend.appDB)
|
2019-11-23 17:57:05 +00:00
|
|
|
_, err = backend.AccountManager().ImportAccount(pkey, password)
|
2019-08-29 08:06:22 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, db.SaveAccounts([]accounts.Account{{Address: address}}))
|
|
|
|
key, err := backend.getVerifiedWalletAccount(address.String(), "wrong-password")
|
2019-10-04 15:21:24 +00:00
|
|
|
require.EqualError(t, err, "could not decrypt key with given password")
|
2019-08-29 08:06:22 +00:00
|
|
|
require.Nil(t, key)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Success", func(t *testing.T) {
|
|
|
|
pkey, err := crypto.GenerateKey()
|
|
|
|
require.NoError(t, err)
|
|
|
|
address := crypto.PubkeyToAddress(pkey.PublicKey)
|
|
|
|
db := accounts.NewDB(backend.appDB)
|
2019-11-23 17:57:05 +00:00
|
|
|
_, err = backend.AccountManager().ImportAccount(pkey, password)
|
2019-08-29 08:06:22 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, db.SaveAccounts([]accounts.Account{{Address: address}}))
|
|
|
|
key, err := backend.getVerifiedWalletAccount(address.String(), password)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, address, key.Address)
|
|
|
|
})
|
|
|
|
}
|
2019-09-02 19:03:15 +00:00
|
|
|
|
|
|
|
func TestLoginWithKey(t *testing.T) {
|
2019-10-04 15:21:24 +00:00
|
|
|
utils.Init()
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
b := NewGethStatusBackend()
|
2019-12-11 13:59:37 +00:00
|
|
|
chatKey, err := gethcrypto.GenerateKey()
|
2019-09-02 19:03:15 +00:00
|
|
|
require.NoError(t, err)
|
2019-12-11 13:59:37 +00:00
|
|
|
walletKey, err := gethcrypto.GenerateKey()
|
2019-12-18 15:09:04 +00:00
|
|
|
require.NoError(t, err)
|
2019-12-11 13:59:37 +00:00
|
|
|
keyUIDHex := sha256.Sum256(gethcrypto.FromECDSAPub(&chatKey.PublicKey))
|
2019-12-05 08:00:57 +00:00
|
|
|
keyUID := types.EncodeHex(keyUIDHex[:])
|
2019-09-02 19:03:15 +00:00
|
|
|
main := multiaccounts.Account{
|
2019-12-05 08:00:57 +00:00
|
|
|
KeyUID: keyUID,
|
2019-09-02 19:03:15 +00:00
|
|
|
}
|
|
|
|
tmpdir, err := ioutil.TempDir("", "login-with-key-test-")
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer os.Remove(tmpdir)
|
|
|
|
conf, err := params.NewNodeConfig(tmpdir, 1777)
|
|
|
|
require.NoError(t, err)
|
2019-12-11 13:59:37 +00:00
|
|
|
keyhex := hex.EncodeToString(gethcrypto.FromECDSA(chatKey))
|
2019-09-02 19:03:15 +00:00
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
require.NoError(t, b.AccountManager().InitKeystore(conf.KeyStoreDir))
|
2019-09-02 19:03:15 +00:00
|
|
|
b.UpdateRootDataDir(conf.DataDir)
|
|
|
|
require.NoError(t, b.OpenAccounts())
|
|
|
|
|
2019-12-18 15:09:04 +00:00
|
|
|
address := crypto.PubkeyToAddress(walletKey.PublicKey)
|
2019-12-27 09:58:25 +00:00
|
|
|
require.NoError(t, b.SaveAccountAndStartNodeWithKey(main, "test-pass", settings, conf, []accounts.Account{{Address: address, Wallet: true}}, keyhex))
|
2019-09-02 19:03:15 +00:00
|
|
|
require.NoError(t, b.Logout())
|
|
|
|
require.NoError(t, b.StopNode())
|
|
|
|
|
|
|
|
require.NoError(t, b.StartNodeWithKey(main, "test-pass", keyhex))
|
|
|
|
defer func() {
|
|
|
|
assert.NoError(t, b.Logout())
|
|
|
|
assert.NoError(t, b.StopNode())
|
|
|
|
}()
|
|
|
|
extkey, err := b.accountManager.SelectedChatAccount()
|
|
|
|
require.NoError(t, err)
|
2019-12-18 15:09:04 +00:00
|
|
|
require.Equal(t, crypto.PubkeyToAddress(chatKey.PublicKey), extkey.Address)
|
2019-09-02 19:03:15 +00:00
|
|
|
}
|
2020-07-13 10:45:36 +00:00
|
|
|
|
|
|
|
func TestDeleteMulticcount(t *testing.T) {
|
|
|
|
backend := NewGethStatusBackend()
|
|
|
|
|
|
|
|
rootDataDir, err := ioutil.TempDir("", "test-keystore-dir")
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer os.Remove(rootDataDir)
|
|
|
|
|
|
|
|
keyStoreDir := filepath.Join(rootDataDir, "keystore")
|
|
|
|
|
|
|
|
backend.rootDataDir = rootDataDir
|
|
|
|
|
|
|
|
err = backend.AccountManager().InitKeystore(keyStoreDir)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
backend.AccountManager()
|
|
|
|
accs, err := backend.AccountManager().
|
|
|
|
AccountsGenerator().
|
|
|
|
GenerateAndDeriveAddresses(12, 1, "", []string{"m/44'/60'/0'/0"})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
generateAccount := accs[0]
|
|
|
|
accountInfo, err := backend.AccountManager().
|
|
|
|
AccountsGenerator().
|
|
|
|
StoreAccount(generateAccount.ID, "123123")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
account := multiaccounts.Account{
|
|
|
|
Name: "foo",
|
|
|
|
Timestamp: 1,
|
|
|
|
KeycardPairing: "pairing",
|
|
|
|
KeyUID: generateAccount.KeyUID,
|
|
|
|
}
|
|
|
|
|
|
|
|
err = backend.ensureAppDBOpened(account, "123123")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
settings := accounts.Settings{
|
|
|
|
Address: types.HexToAddress(accountInfo.Address),
|
|
|
|
CurrentNetwork: "mainnet_rpc",
|
|
|
|
DappsAddress: types.HexToAddress(accountInfo.Address),
|
|
|
|
EIP1581Address: types.HexToAddress(accountInfo.Address),
|
|
|
|
InstallationID: "d3efcff6-cffa-560e-a547-21d3858cbc51",
|
|
|
|
KeyUID: account.KeyUID,
|
|
|
|
LatestDerivedPath: 0,
|
|
|
|
Name: "Jittery Cornflowerblue Kingbird",
|
|
|
|
Networks: &networks,
|
|
|
|
PhotoPath: "",
|
|
|
|
PreviewPrivacy: false,
|
|
|
|
PublicKey: accountInfo.PublicKey,
|
|
|
|
SigningPhrase: "yurt joey vibe",
|
|
|
|
WalletRootAddress: types.HexToAddress(accountInfo.Address)}
|
|
|
|
|
|
|
|
err = backend.saveAccountsAndSettings(
|
|
|
|
settings,
|
|
|
|
¶ms.NodeConfig{},
|
|
|
|
nil)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = backend.OpenAccounts()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = backend.SaveAccount(account)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
files, err := ioutil.ReadDir(rootDataDir)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotEqual(t, 3, len(files))
|
|
|
|
|
|
|
|
err = backend.DeleteMulticcount(account.KeyUID, keyStoreDir)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
files, err = ioutil.ReadDir(rootDataDir)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 3, len(files))
|
|
|
|
}
|