493 lines
13 KiB
Go
493 lines
13 KiB
Go
package account
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"reflect"
|
|
"testing"
|
|
|
|
"github.com/ethereum/go-ethereum/accounts"
|
|
"github.com/ethereum/go-ethereum/accounts/keystore"
|
|
gethcommon "github.com/ethereum/go-ethereum/common"
|
|
whisper "github.com/ethereum/go-ethereum/whisper/whisperv5"
|
|
"github.com/golang/mock/gomock"
|
|
"github.com/status-im/status-go/geth/common"
|
|
. "github.com/status-im/status-go/testing"
|
|
"github.com/stretchr/testify/require"
|
|
"github.com/stretchr/testify/suite"
|
|
)
|
|
|
|
func TestVerifyAccountPassword(t *testing.T) {
|
|
accManager := NewManager(nil)
|
|
keyStoreDir, err := ioutil.TempDir(os.TempDir(), "accounts")
|
|
require.NoError(t, err)
|
|
defer os.RemoveAll(keyStoreDir) //nolint: errcheck
|
|
|
|
emptyKeyStoreDir, err := ioutil.TempDir(os.TempDir(), "accounts_empty")
|
|
require.NoError(t, err)
|
|
defer os.RemoveAll(emptyKeyStoreDir) //nolint: errcheck
|
|
|
|
// import account keys
|
|
require.NoError(t, common.ImportTestAccount(keyStoreDir, GetAccount1PKFile()))
|
|
require.NoError(t, common.ImportTestAccount(keyStoreDir, GetAccount2PKFile()))
|
|
|
|
account1Address := gethcommon.BytesToAddress(gethcommon.FromHex(TestConfig.Account1.Address))
|
|
|
|
testCases := []struct {
|
|
name string
|
|
keyPath string
|
|
address string
|
|
password string
|
|
expectedError error
|
|
}{
|
|
{
|
|
"correct address, correct password (decrypt should succeed)",
|
|
keyStoreDir,
|
|
TestConfig.Account1.Address,
|
|
TestConfig.Account1.Password,
|
|
nil,
|
|
},
|
|
{
|
|
"correct address, correct password, non-existent key store",
|
|
filepath.Join(keyStoreDir, "non-existent-folder"),
|
|
TestConfig.Account1.Address,
|
|
TestConfig.Account1.Password,
|
|
fmt.Errorf("cannot traverse key store folder: lstat %s/non-existent-folder: no such file or directory", keyStoreDir),
|
|
},
|
|
{
|
|
"correct address, correct password, empty key store (pk is not there)",
|
|
emptyKeyStoreDir,
|
|
TestConfig.Account1.Address,
|
|
TestConfig.Account1.Password,
|
|
fmt.Errorf("cannot locate account for address: %s", account1Address.Hex()),
|
|
},
|
|
{
|
|
"wrong address, correct password",
|
|
keyStoreDir,
|
|
"0x79791d3e8f2daa1f7fec29649d152c0ada3cc535",
|
|
TestConfig.Account1.Password,
|
|
fmt.Errorf("cannot locate account for address: %s", "0x79791d3E8F2dAa1F7FeC29649d152c0aDA3cc535"),
|
|
},
|
|
{
|
|
"correct address, wrong password",
|
|
keyStoreDir,
|
|
TestConfig.Account1.Address,
|
|
"wrong password", // wrong password
|
|
errors.New("could not decrypt key with given passphrase"),
|
|
},
|
|
}
|
|
for _, testCase := range testCases {
|
|
accountKey, err := accManager.VerifyAccountPassword(testCase.keyPath, testCase.address, testCase.password)
|
|
if !reflect.DeepEqual(err, testCase.expectedError) {
|
|
require.FailNow(t, fmt.Sprintf("unexpected error: expected \n'%v', got \n'%v'", testCase.expectedError, err))
|
|
}
|
|
if err == nil {
|
|
if accountKey == nil {
|
|
require.Fail(t, "no error reported, but account key is missing")
|
|
}
|
|
accountAddress := gethcommon.BytesToAddress(gethcommon.FromHex(testCase.address))
|
|
if accountKey.Address != accountAddress {
|
|
require.Fail(t, "account mismatch: have %s, want %s", accountKey.Address.Hex(), accountAddress.Hex())
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// TestVerifyAccountPasswordWithAccountBeforeEIP55 verifies if VerifyAccountPassword
|
|
// can handle accounts before introduction of EIP55.
|
|
func TestVerifyAccountPasswordWithAccountBeforeEIP55(t *testing.T) {
|
|
keyStoreDir, err := ioutil.TempDir("", "status-accounts-test")
|
|
require.NoError(t, err)
|
|
defer os.RemoveAll(keyStoreDir) //nolint: errcheck
|
|
|
|
// Import keys and make sure one was created before EIP55 introduction.
|
|
err = common.ImportTestAccount(keyStoreDir, "test-account3-before-eip55.pk")
|
|
require.NoError(t, err)
|
|
|
|
accManager := NewManager(nil)
|
|
|
|
address := gethcommon.HexToAddress(TestConfig.Account3.Address)
|
|
_, err = accManager.VerifyAccountPassword(keyStoreDir, address.Hex(), TestConfig.Account3.Password)
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
var (
|
|
testErrWhisper = errors.New("Can't return a whisper service")
|
|
testErrKeyStore = errors.New("Can't return a key store")
|
|
)
|
|
|
|
func TestManagerTestSuite(t *testing.T) {
|
|
nodeManager := newMockNodeManager(t)
|
|
accManager := NewManager(nodeManager)
|
|
|
|
keyStoreDir, err := ioutil.TempDir(os.TempDir(), "accounts")
|
|
require.NoError(t, err)
|
|
keyStore := keystore.NewKeyStore(keyStoreDir, keystore.LightScryptN, keystore.LightScryptP)
|
|
defer os.RemoveAll(keyStoreDir) //nolint: errcheck
|
|
|
|
testPassword := "test-password"
|
|
|
|
// Initial test - create test account
|
|
nodeManager.EXPECT().AccountKeyStore().Return(keyStore, nil)
|
|
addr, pubKey, mnemonic, err := accManager.CreateAccount(testPassword)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, addr)
|
|
require.NotEmpty(t, pubKey)
|
|
require.NotEmpty(t, mnemonic)
|
|
|
|
s := &ManagerTestSuite{
|
|
testAccount: testAccount{
|
|
"test-password",
|
|
addr,
|
|
pubKey,
|
|
mnemonic,
|
|
},
|
|
nodeManager: nodeManager,
|
|
accManager: accManager,
|
|
keyStore: keyStore,
|
|
shh: whisper.New(nil),
|
|
gethAccManager: accounts.NewManager(),
|
|
}
|
|
|
|
suite.Run(t, s)
|
|
}
|
|
|
|
func newMockNodeManager(t *testing.T) *common.MockNodeManager {
|
|
ctrl := gomock.NewController(t)
|
|
return common.NewMockNodeManager(ctrl)
|
|
}
|
|
|
|
type ManagerTestSuite struct {
|
|
suite.Suite
|
|
testAccount
|
|
nodeManager *common.MockNodeManager
|
|
accManager *Manager
|
|
keyStore *keystore.KeyStore
|
|
shh *whisper.Whisper
|
|
gethAccManager *accounts.Manager
|
|
}
|
|
|
|
type testAccount struct {
|
|
password string
|
|
address string
|
|
pubKey string
|
|
mnemonic string
|
|
}
|
|
|
|
// reinitMock is for reassigning a new mock node manager to account manager.
|
|
// Stating the amount of times for mock calls kills the flexibility for
|
|
// development so this is a good workaround to use with EXPECT().AnyTimes()
|
|
func (s *ManagerTestSuite) reinitMock() {
|
|
s.nodeManager = newMockNodeManager(s.T())
|
|
s.accManager.nodeManager = s.nodeManager
|
|
}
|
|
|
|
func (s *ManagerTestSuite) TestCreateAccount() {
|
|
s.reinitMock()
|
|
|
|
// Don't fail on empty password
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(s.keyStore, nil)
|
|
_, _, _, err := s.accManager.CreateAccount(s.password)
|
|
s.NoError(err)
|
|
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(nil, testErrKeyStore)
|
|
_, _, _, err = s.accManager.CreateAccount(s.password)
|
|
s.Equal(err, testErrKeyStore)
|
|
}
|
|
|
|
func (s *ManagerTestSuite) TestRecoverAccount() {
|
|
s.reinitMock()
|
|
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(s.keyStore, nil)
|
|
addr, pubKey, err := s.accManager.RecoverAccount(s.password, s.mnemonic)
|
|
s.NoError(err)
|
|
s.Equal(s.address, addr)
|
|
s.Equal(s.pubKey, pubKey)
|
|
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(nil, testErrKeyStore)
|
|
_, _, err = s.accManager.RecoverAccount(s.password, s.mnemonic)
|
|
s.Equal(err, testErrKeyStore)
|
|
}
|
|
|
|
func (s *ManagerTestSuite) TestSelectAccount() {
|
|
s.reinitMock()
|
|
|
|
testCases := []struct {
|
|
name string
|
|
accountKeyStoreReturn []interface{}
|
|
whisperServiceReturn []interface{}
|
|
address string
|
|
password string
|
|
fail bool
|
|
}{
|
|
{
|
|
"success",
|
|
[]interface{}{s.keyStore, nil},
|
|
[]interface{}{s.shh, nil},
|
|
s.address,
|
|
s.password,
|
|
false,
|
|
},
|
|
{
|
|
"fail_keyStore",
|
|
[]interface{}{nil, testErrKeyStore},
|
|
[]interface{}{s.shh, nil},
|
|
s.address,
|
|
s.password,
|
|
true,
|
|
},
|
|
{
|
|
"fail_whisperService",
|
|
[]interface{}{s.keyStore, nil},
|
|
[]interface{}{nil, testErrWhisper},
|
|
s.address,
|
|
s.password,
|
|
true,
|
|
},
|
|
{
|
|
"fail_wrongAddress",
|
|
[]interface{}{s.keyStore, nil},
|
|
[]interface{}{s.shh, nil},
|
|
"wrong-address",
|
|
s.password,
|
|
true,
|
|
},
|
|
{
|
|
"fail_wrongPassword",
|
|
[]interface{}{s.keyStore, nil},
|
|
[]interface{}{s.shh, nil},
|
|
s.address,
|
|
"wrong-password",
|
|
true,
|
|
},
|
|
}
|
|
|
|
for _, testCase := range testCases {
|
|
s.T().Run(testCase.name, func(t *testing.T) {
|
|
s.reinitMock()
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(testCase.accountKeyStoreReturn...).AnyTimes()
|
|
s.nodeManager.EXPECT().WhisperService().Return(testCase.whisperServiceReturn...).AnyTimes()
|
|
err := s.accManager.SelectAccount(testCase.address, testCase.password)
|
|
if testCase.fail {
|
|
s.Error(err)
|
|
} else {
|
|
s.NoError(err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func (s *ManagerTestSuite) TestCreateChildAccount() {
|
|
s.reinitMock()
|
|
|
|
// First, test the negative case where an account is not selected
|
|
// and an address is not provided.
|
|
s.accManager.selectedAccount = nil
|
|
s.T().Run("fail_noAccount", func(t *testing.T) {
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(s.keyStore, nil).AnyTimes()
|
|
_, _, err := s.accManager.CreateChildAccount("", s.password)
|
|
s.Error(err)
|
|
})
|
|
|
|
// Now, select the test account for rest of the test cases.
|
|
s.reinitMock()
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(s.keyStore, nil).AnyTimes()
|
|
s.nodeManager.EXPECT().WhisperService().Return(s.shh, nil).AnyTimes()
|
|
err := s.accManager.SelectAccount(s.address, s.password)
|
|
s.NoError(err)
|
|
|
|
testCases := []struct {
|
|
name string
|
|
address string
|
|
password string
|
|
accountKeyStoreReturn []interface{}
|
|
fail bool
|
|
}{
|
|
{
|
|
"success",
|
|
s.address,
|
|
s.password,
|
|
[]interface{}{s.keyStore, nil},
|
|
false,
|
|
},
|
|
{
|
|
"fail_keyStore",
|
|
s.address,
|
|
s.password,
|
|
[]interface{}{nil, testErrKeyStore},
|
|
true,
|
|
},
|
|
{
|
|
"fail_wrongAddress",
|
|
"wrong-address",
|
|
s.password,
|
|
[]interface{}{s.keyStore, nil},
|
|
true,
|
|
},
|
|
{
|
|
"fail_wrongPassword",
|
|
s.address,
|
|
"wrong-password",
|
|
[]interface{}{s.keyStore, nil},
|
|
true,
|
|
},
|
|
}
|
|
|
|
for _, testCase := range testCases {
|
|
s.T().Run(testCase.name, func(t *testing.T) {
|
|
s.reinitMock()
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(testCase.accountKeyStoreReturn...).AnyTimes()
|
|
childAddr, childPubKey, err := s.accManager.CreateChildAccount(testCase.address, testCase.password)
|
|
if testCase.fail {
|
|
s.Error(err)
|
|
} else {
|
|
s.NoError(err)
|
|
s.NotEmpty(childAddr)
|
|
s.NotEmpty(childPubKey)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func (s *ManagerTestSuite) TestSelectedAndReSelectAccount() {
|
|
s.reinitMock()
|
|
|
|
// Select the test account
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(s.keyStore, nil).AnyTimes()
|
|
s.nodeManager.EXPECT().WhisperService().Return(s.shh, nil).AnyTimes()
|
|
err := s.accManager.SelectAccount(s.address, s.password)
|
|
s.NoError(err)
|
|
|
|
s.T().Run("success", func(t *testing.T) {
|
|
acc, err := s.accManager.SelectedAccount()
|
|
s.NoError(err)
|
|
s.NotNil(acc)
|
|
|
|
err = s.accManager.ReSelectAccount()
|
|
s.NoError(err)
|
|
})
|
|
|
|
s.T().Run("ReSelect_fail_whisper", func(t *testing.T) {
|
|
s.reinitMock()
|
|
s.nodeManager.EXPECT().WhisperService().Return(nil, testErrWhisper).AnyTimes()
|
|
err = s.accManager.ReSelectAccount()
|
|
s.Error(err)
|
|
})
|
|
|
|
s.accManager.selectedAccount = nil
|
|
s.reinitMock()
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(s.keyStore, nil).AnyTimes()
|
|
s.nodeManager.EXPECT().WhisperService().Return(s.shh, nil).AnyTimes()
|
|
|
|
s.T().Run("Selected_fail_noAccount", func(t *testing.T) {
|
|
_, err := s.accManager.SelectedAccount()
|
|
s.Equal(ErrNoAccountSelected, err)
|
|
})
|
|
|
|
s.T().Run("ReSelect_success_noAccount", func(t *testing.T) {
|
|
err = s.accManager.ReSelectAccount()
|
|
s.NoError(err)
|
|
})
|
|
}
|
|
|
|
func (s *ManagerTestSuite) TestLogout() {
|
|
s.reinitMock()
|
|
|
|
s.nodeManager.EXPECT().WhisperService().Return(s.shh, nil)
|
|
err := s.accManager.Logout()
|
|
s.NoError(err)
|
|
|
|
s.nodeManager.EXPECT().WhisperService().Return(nil, testErrWhisper)
|
|
err = s.accManager.Logout()
|
|
s.Error(err)
|
|
}
|
|
|
|
func (s *ManagerTestSuite) TestAccounts() {
|
|
s.reinitMock()
|
|
|
|
// Select the test account
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(s.keyStore, nil).AnyTimes()
|
|
s.nodeManager.EXPECT().WhisperService().Return(s.shh, nil).AnyTimes()
|
|
err := s.accManager.SelectAccount(s.address, s.password)
|
|
s.NoError(err)
|
|
|
|
// Success
|
|
s.nodeManager.EXPECT().AccountManager().Return(s.gethAccManager, nil)
|
|
accs, err := s.accManager.Accounts()
|
|
s.NoError(err)
|
|
s.NotNil(accs)
|
|
|
|
// Can't get an account manager
|
|
s.nodeManager.EXPECT().AccountManager().Return(nil, errors.New("Can't return an account manager"))
|
|
_, err = s.accManager.Accounts()
|
|
s.Error(err)
|
|
|
|
// Selected account is nil but doesn't fail
|
|
s.accManager.selectedAccount = nil
|
|
s.nodeManager.EXPECT().AccountManager().Return(s.gethAccManager, nil)
|
|
accs, err = s.accManager.Accounts()
|
|
s.NoError(err)
|
|
s.NotNil(accs)
|
|
}
|
|
|
|
func (s *ManagerTestSuite) TestAddressToDecryptedAccount() {
|
|
s.reinitMock()
|
|
|
|
testCases := []struct {
|
|
name string
|
|
accountKeyStoreReturn []interface{}
|
|
address string
|
|
password string
|
|
fail bool
|
|
}{
|
|
{
|
|
"success",
|
|
[]interface{}{s.keyStore, nil},
|
|
s.address,
|
|
s.password,
|
|
false,
|
|
},
|
|
{
|
|
"fail_keyStore",
|
|
[]interface{}{nil, testErrKeyStore},
|
|
s.address,
|
|
s.password,
|
|
true,
|
|
},
|
|
{
|
|
"fail_wrongAddress",
|
|
[]interface{}{s.keyStore, nil},
|
|
"wrong-address",
|
|
s.password,
|
|
true,
|
|
},
|
|
{
|
|
"fail_wrongPassword",
|
|
[]interface{}{s.keyStore, nil},
|
|
s.address,
|
|
"wrong-password",
|
|
true,
|
|
},
|
|
}
|
|
|
|
for _, testCase := range testCases {
|
|
s.T().Run(testCase.name, func(t *testing.T) {
|
|
s.reinitMock()
|
|
s.nodeManager.EXPECT().AccountKeyStore().Return(testCase.accountKeyStoreReturn...).AnyTimes()
|
|
acc, key, err := s.accManager.AddressToDecryptedAccount(testCase.address, testCase.password)
|
|
if testCase.fail {
|
|
s.Error(err)
|
|
} else {
|
|
s.NoError(err)
|
|
s.NotNil(acc)
|
|
s.NotNil(key)
|
|
s.Equal(acc.Address, key.Address)
|
|
s.keyStore.Find(acc)
|
|
}
|
|
})
|
|
}
|
|
}
|