2017-12-19 18:45:30 +03:00
package account
2017-05-16 15:09:52 +03:00
import (
"errors"
"fmt"
"io/ioutil"
"os"
"path/filepath"
"reflect"
"testing"
2019-12-11 14:59:37 +01:00
"github.com/status-im/status-go/eth-node/crypto"
"github.com/status-im/status-go/eth-node/types"
2019-10-04 17:21:24 +02:00
"github.com/status-im/status-go/t/utils"
2017-10-11 16:20:51 +02:00
"github.com/stretchr/testify/require"
2017-12-19 16:07:31 +03:00
"github.com/stretchr/testify/suite"
2017-05-16 15:09:52 +03:00
)
2017-10-11 16:20:51 +02:00
func TestVerifyAccountPassword ( t * testing . T ) {
2019-08-20 18:38:40 +03:00
accManager := NewManager ( )
2017-05-16 15:09:52 +03:00
keyStoreDir , err := ioutil . TempDir ( os . TempDir ( ) , "accounts" )
2017-10-11 16:20:51 +02:00
require . NoError ( t , err )
2017-10-20 12:06:22 +03:00
defer os . RemoveAll ( keyStoreDir ) //nolint: errcheck
2017-05-16 15:09:52 +03:00
2017-10-11 16:20:51 +02:00
emptyKeyStoreDir , err := ioutil . TempDir ( os . TempDir ( ) , "accounts_empty" )
require . NoError ( t , err )
2017-10-20 12:06:22 +03:00
defer os . RemoveAll ( emptyKeyStoreDir ) //nolint: errcheck
2017-05-16 15:09:52 +03:00
// import account keys
2019-10-04 17:21:24 +02:00
utils . Init ( )
require . NoError ( t , utils . ImportTestAccount ( keyStoreDir , utils . GetAccount1PKFile ( ) ) )
require . NoError ( t , utils . ImportTestAccount ( keyStoreDir , utils . GetAccount2PKFile ( ) ) )
2017-05-16 15:09:52 +03:00
2019-12-11 14:59:37 +01:00
account1Address := types . BytesToAddress ( types . FromHex ( utils . TestConfig . Account1 . WalletAddress ) )
2017-05-16 15:09:52 +03:00
testCases := [ ] struct {
name string
keyPath string
address string
password string
expectedError error
} {
{
"correct address, correct password (decrypt should succeed)" ,
keyStoreDir ,
2019-10-04 17:21:24 +02:00
utils . TestConfig . Account1 . WalletAddress ,
utils . TestConfig . Account1 . Password ,
2017-05-16 15:09:52 +03:00
nil ,
} ,
{
"correct address, correct password, non-existent key store" ,
filepath . Join ( keyStoreDir , "non-existent-folder" ) ,
2019-10-04 17:21:24 +02:00
utils . TestConfig . Account1 . WalletAddress ,
utils . TestConfig . Account1 . Password ,
2017-05-16 15:09:52 +03:00
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 ,
2019-10-04 17:21:24 +02:00
utils . TestConfig . Account1 . WalletAddress ,
utils . TestConfig . Account1 . Password ,
2017-10-10 11:38:49 +02:00
fmt . Errorf ( "cannot locate account for address: %s" , account1Address . Hex ( ) ) ,
2017-05-16 15:09:52 +03:00
} ,
{
"wrong address, correct password" ,
keyStoreDir ,
"0x79791d3e8f2daa1f7fec29649d152c0ada3cc535" ,
2019-10-04 17:21:24 +02:00
utils . TestConfig . Account1 . Password ,
2017-10-10 11:38:49 +02:00
fmt . Errorf ( "cannot locate account for address: %s" , "0x79791d3E8F2dAa1F7FeC29649d152c0aDA3cc535" ) ,
2017-05-16 15:09:52 +03:00
} ,
{
"correct address, wrong password" ,
keyStoreDir ,
2019-10-04 17:21:24 +02:00
utils . TestConfig . Account1 . WalletAddress ,
2017-05-16 15:09:52 +03:00
"wrong password" , // wrong password
2019-10-04 17:21:24 +02:00
errors . New ( "could not decrypt key with given password" ) ,
2017-05-16 15:09:52 +03:00
} ,
}
for _ , testCase := range testCases {
2017-12-19 18:45:30 +03:00
accountKey , err := accManager . VerifyAccountPassword ( testCase . keyPath , testCase . address , testCase . password )
2017-05-16 15:09:52 +03:00
if ! reflect . DeepEqual ( err , testCase . expectedError ) {
2017-10-11 16:20:51 +02:00
require . FailNow ( t , fmt . Sprintf ( "unexpected error: expected \n'%v', got \n'%v'" , testCase . expectedError , err ) )
2017-05-16 15:09:52 +03:00
}
if err == nil {
if accountKey == nil {
2017-10-11 16:20:51 +02:00
require . Fail ( t , "no error reported, but account key is missing" )
2017-05-16 15:09:52 +03:00
}
2019-12-11 14:59:37 +01:00
accountAddress := types . BytesToAddress ( types . FromHex ( testCase . address ) )
2019-12-19 19:27:27 +01:00
if accountKey . Address != accountAddress {
2017-10-11 16:20:51 +02:00
require . Fail ( t , "account mismatch: have %s, want %s" , accountKey . Address . Hex ( ) , accountAddress . Hex ( ) )
2017-05-16 15:09:52 +03:00
}
}
}
}
2017-10-10 11:38:49 +02:00
// TestVerifyAccountPasswordWithAccountBeforeEIP55 verifies if VerifyAccountPassword
// can handle accounts before introduction of EIP55.
2017-10-11 16:20:51 +02:00
func TestVerifyAccountPasswordWithAccountBeforeEIP55 ( t * testing . T ) {
2017-10-10 11:38:49 +02:00
keyStoreDir , err := ioutil . TempDir ( "" , "status-accounts-test" )
2017-10-11 16:20:51 +02:00
require . NoError ( t , err )
2017-10-20 12:06:22 +03:00
defer os . RemoveAll ( keyStoreDir ) //nolint: errcheck
2017-10-10 11:38:49 +02:00
// Import keys and make sure one was created before EIP55 introduction.
2019-10-04 17:21:24 +02:00
utils . Init ( )
err = utils . ImportTestAccount ( keyStoreDir , "test-account3-before-eip55.pk" )
2017-10-11 16:20:51 +02:00
require . NoError ( t , err )
2017-10-10 11:38:49 +02:00
2019-08-20 18:38:40 +03:00
accManager := NewManager ( )
2017-10-10 11:38:49 +02:00
2019-12-11 14:59:37 +01:00
address := types . HexToAddress ( utils . TestConfig . Account3 . WalletAddress )
2019-10-04 17:21:24 +02:00
_ , err = accManager . VerifyAccountPassword ( keyStoreDir , address . Hex ( ) , utils . TestConfig . Account3 . Password )
2017-10-11 16:20:51 +02:00
require . NoError ( t , err )
2017-10-10 11:38:49 +02:00
}
2017-12-14 05:03:55 +03:00
2017-12-19 16:07:31 +03:00
func TestManagerTestSuite ( t * testing . T ) {
2019-08-20 18:38:40 +03:00
suite . Run ( t , new ( ManagerTestSuite ) )
2017-12-19 18:45:30 +03:00
}
2017-12-19 16:07:31 +03:00
type ManagerTestSuite struct {
suite . Suite
2017-12-21 20:07:08 +03:00
testAccount
2019-12-19 19:27:27 +01:00
accManager * GethManager
2019-08-20 18:38:40 +03:00
keydir string
2017-12-19 16:07:31 +03:00
}
2017-12-14 05:03:55 +03:00
2017-12-21 20:07:08 +03:00
type testAccount struct {
2019-01-18 10:01:14 +01:00
password string
walletAddress string
walletPubKey string
chatAddress string
chatPubKey string
mnemonic string
2017-12-21 20:07:08 +03:00
}
2018-01-03 18:51:27 +03:00
// SetupTest is used here for reinitializing the mock before every
// test function to avoid faulty execution.
func ( s * ManagerTestSuite ) SetupTest ( ) {
2019-08-20 18:38:40 +03:00
s . accManager = NewManager ( )
2019-07-31 10:50:40 +03:00
2019-08-20 18:38:40 +03:00
keyStoreDir , err := ioutil . TempDir ( os . TempDir ( ) , "accounts" )
s . Require ( ) . NoError ( err )
s . Require ( ) . NoError ( s . accManager . InitKeystore ( keyStoreDir ) )
s . keydir = keyStoreDir
testPassword := "test-password"
2019-08-01 11:27:06 +02:00
2019-08-20 18:38:40 +03:00
// Initial test - create test account
accountInfo , mnemonic , err := s . accManager . CreateAccount ( testPassword )
s . Require ( ) . NoError ( err )
s . Require ( ) . NotEmpty ( accountInfo . WalletAddress )
s . Require ( ) . NotEmpty ( accountInfo . WalletPubKey )
s . Require ( ) . NotEmpty ( accountInfo . ChatAddress )
s . Require ( ) . NotEmpty ( accountInfo . ChatPubKey )
s . Require ( ) . NotEmpty ( mnemonic )
// Before the complete decoupling of the keys, wallet and chat keys are the same
s . Equal ( accountInfo . WalletAddress , accountInfo . ChatAddress )
s . Equal ( accountInfo . WalletPubKey , accountInfo . ChatPubKey )
s . testAccount = testAccount {
testPassword ,
accountInfo . WalletAddress ,
accountInfo . WalletPubKey ,
accountInfo . ChatAddress ,
accountInfo . ChatPubKey ,
mnemonic ,
}
}
func ( s * ManagerTestSuite ) TearDownTest ( ) {
s . Require ( ) . NoError ( os . RemoveAll ( s . keydir ) )
2017-12-28 16:40:56 +03:00
}
func ( s * ManagerTestSuite ) TestRecoverAccount ( ) {
2019-01-18 10:01:14 +01:00
accountInfo , err := s . accManager . RecoverAccount ( s . password , s . mnemonic )
2017-12-19 16:07:31 +03:00
s . NoError ( err )
2019-01-18 10:01:14 +01:00
s . Equal ( s . walletAddress , accountInfo . WalletAddress )
s . Equal ( s . walletPubKey , accountInfo . WalletPubKey )
s . Equal ( s . chatAddress , accountInfo . ChatAddress )
s . Equal ( s . chatPubKey , accountInfo . ChatPubKey )
2017-12-14 05:03:55 +03:00
}
2017-12-18 17:08:31 +03:00
2019-06-27 00:28:16 +02:00
func ( s * ManagerTestSuite ) TestOnboarding ( ) {
// try to choose an account before starting onboarding
_ , _ , err := s . accManager . ImportOnboardingAccount ( "test-id" , "test-password" )
s . Equal ( ErrOnboardingNotStarted , err )
// generates 5 random accounts
count := 5
accounts , err := s . accManager . StartOnboarding ( count , 24 )
s . Require ( ) . NoError ( err )
s . Equal ( count , len ( accounts ) )
// try to choose an account with an undefined id
_ , _ , err = s . accManager . ImportOnboardingAccount ( "test-id" , "test-password" )
s . Equal ( ErrOnboardingAccountNotFound , err )
// choose one account and encrypt it with password
password := "test-onboarding-account"
account := accounts [ 0 ]
info , mnemonic , err := s . accManager . ImportOnboardingAccount ( account . ID , password )
s . Require ( ) . NoError ( err )
s . Equal ( account . Info , info )
s . Equal ( account . mnemonic , mnemonic )
s . Nil ( s . accManager . onboarding )
// try to decrypt it with password to check if it's been imported correctly
decAccount , _ , err := s . accManager . AddressToDecryptedAccount ( info . WalletAddress , password )
s . Require ( ) . NoError ( err )
s . Equal ( info . WalletAddress , decAccount . Address . Hex ( ) )
// try resetting onboarding
_ , err = s . accManager . StartOnboarding ( count , 24 )
s . Require ( ) . NoError ( err )
s . NotNil ( s . accManager . onboarding )
s . accManager . RemoveOnboarding ( )
s . Nil ( s . accManager . onboarding )
}
2019-08-20 18:38:40 +03:00
func ( s * ManagerTestSuite ) TestSelectAccountSuccess ( ) {
2019-12-11 14:59:37 +01:00
s . testSelectAccount ( types . HexToAddress ( s . testAccount . chatAddress ) , types . HexToAddress ( s . testAccount . walletAddress ) , s . testAccount . password , nil )
2019-08-20 18:38:40 +03:00
}
2019-07-31 10:50:40 +03:00
2019-08-20 18:38:40 +03:00
func ( s * ManagerTestSuite ) TestSelectAccountWrongAddress ( ) {
2019-12-11 14:59:37 +01:00
s . testSelectAccount ( types . HexToAddress ( "0x0000000000000000000000000000000000000001" ) , types . HexToAddress ( s . testAccount . walletAddress ) , s . testAccount . password , errors . New ( "cannot retrieve a valid key for a given account: no key for given address or file" ) )
2019-08-20 18:38:40 +03:00
}
func ( s * ManagerTestSuite ) TestSelectAccountWrongPassword ( ) {
2019-12-11 14:59:37 +01:00
s . testSelectAccount ( types . HexToAddress ( s . testAccount . chatAddress ) , types . HexToAddress ( s . testAccount . walletAddress ) , "wrong" , errors . New ( "cannot retrieve a valid key for a given account: could not decrypt key with given password" ) )
2019-08-20 18:38:40 +03:00
}
2019-01-18 10:01:14 +01:00
2019-12-11 14:59:37 +01:00
func ( s * ManagerTestSuite ) testSelectAccount ( chat , wallet types . Address , password string , expErr error ) {
2019-08-20 18:38:40 +03:00
loginParams := LoginParams {
ChatAddress : chat ,
MainAccount : wallet ,
Password : password ,
2017-12-19 18:45:30 +03:00
}
2019-08-20 18:38:40 +03:00
err := s . accManager . SelectAccount ( loginParams )
s . Require ( ) . Equal ( expErr , err )
selectedMainAccountAddress , walletErr := s . accManager . MainAccountAddress ( )
selectedChatAccount , chatErr := s . accManager . SelectedChatAccount ( )
if expErr == nil {
s . Require ( ) . NoError ( walletErr )
s . Require ( ) . NoError ( chatErr )
s . Equal ( wallet , selectedMainAccountAddress )
s . Equal ( chat , crypto . PubkeyToAddress ( selectedChatAccount . AccountKey . PrivateKey . PublicKey ) )
} else {
2019-12-11 14:59:37 +01:00
s . Equal ( types . Address { } , selectedMainAccountAddress )
2019-08-20 18:38:40 +03:00
s . Nil ( selectedChatAccount )
s . Equal ( walletErr , ErrNoAccountSelected )
s . Equal ( chatErr , ErrNoAccountSelected )
}
s . accManager . Logout ( )
2017-12-19 18:45:30 +03:00
}
2019-01-24 16:44:46 +01:00
func ( s * ManagerTestSuite ) TestSetChatAccount ( ) {
s . accManager . Logout ( )
privKey , err := crypto . GenerateKey ( )
s . Require ( ) . NoError ( err )
address := crypto . PubkeyToAddress ( privKey . PublicKey )
s . accManager . SetChatAccount ( privKey )
selectedChatAccount , err := s . accManager . SelectedChatAccount ( )
s . Require ( ) . NoError ( err )
s . Require ( ) . NotNil ( selectedChatAccount )
s . Equal ( privKey , selectedChatAccount . AccountKey . PrivateKey )
s . Equal ( address , selectedChatAccount . Address )
2019-07-26 16:45:10 +02:00
selectedMainAccountAddress , err := s . accManager . MainAccountAddress ( )
2019-01-24 16:44:46 +01:00
s . Error ( err )
2019-12-11 14:59:37 +01:00
s . Equal ( types . Address { } , selectedMainAccountAddress )
2017-12-18 17:08:31 +03:00
}
2017-12-19 19:11:46 +03:00
2017-12-19 19:59:43 +03:00
func ( s * ManagerTestSuite ) TestLogout ( ) {
2018-04-20 17:39:53 +02:00
s . accManager . Logout ( )
2019-12-11 14:59:37 +01:00
s . Equal ( types . Address { } , s . accManager . mainAccountAddress )
2019-07-26 16:45:10 +02:00
s . Nil ( s . accManager . selectedChatAccount )
s . Len ( s . accManager . watchAddresses , 0 )
2017-12-19 19:59:43 +03:00
}
2017-12-19 20:20:53 +03:00
2018-01-03 22:57:25 +03:00
// TestAccounts tests cases for (*Manager).Accounts.
2017-12-19 20:20:53 +03:00
func ( s * ManagerTestSuite ) TestAccounts ( ) {
2017-12-21 20:07:08 +03:00
// Select the test account
2019-07-26 16:45:10 +02:00
loginParams := LoginParams {
2019-12-11 14:59:37 +01:00
MainAccount : types . HexToAddress ( s . walletAddress ) ,
ChatAddress : types . HexToAddress ( s . chatAddress ) ,
2019-07-26 16:45:10 +02:00
Password : s . password ,
}
err := s . accManager . SelectAccount ( loginParams )
2017-12-19 20:20:53 +03:00
s . NoError ( err )
// Success
accs , err := s . accManager . Accounts ( )
s . NoError ( err )
s . NotNil ( accs )
2019-07-26 16:45:10 +02:00
// Selected main account address is zero address but doesn't fail
2019-12-11 14:59:37 +01:00
s . accManager . mainAccountAddress = types . Address { }
2017-12-19 20:20:53 +03:00
accs , err = s . accManager . Accounts ( )
s . NoError ( err )
s . NotNil ( accs )
}
2017-12-19 20:32:10 +03:00
2019-08-20 18:38:40 +03:00
func ( s * ManagerTestSuite ) TestAddressToDecryptedAccountSuccess ( ) {
s . testAddressToDecryptedAccount ( s . walletAddress , s . password , nil )
}
2019-07-31 10:50:40 +03:00
2019-08-20 18:38:40 +03:00
func ( s * ManagerTestSuite ) TestAddressToDecryptedAccountWrongAddress ( ) {
s . testAddressToDecryptedAccount ( "0x0001" , s . password , ErrAddressToAccountMappingFailure )
}
func ( s * ManagerTestSuite ) TestAddressToDecryptedAccountWrongPassword ( ) {
2019-10-04 17:21:24 +02:00
s . testAddressToDecryptedAccount ( s . walletAddress , "wrong" , errors . New ( "cannot retrieve a valid key for a given account: could not decrypt key with given password" ) )
2019-08-20 18:38:40 +03:00
}
func ( s * ManagerTestSuite ) testAddressToDecryptedAccount ( wallet , password string , expErr error ) {
acc , key , err := s . accManager . AddressToDecryptedAccount ( wallet , password )
if expErr != nil {
s . Equal ( expErr , err )
} else {
s . Require ( ) . NoError ( err )
s . Require ( ) . NotNil ( acc )
s . Require ( ) . NotNil ( key )
s . Equal ( acc . Address , key . Address )
2017-12-19 20:32:10 +03:00
}
}