package wallet import ( "encoding/json" "path" "path/filepath" "testing" "github.com/nspcc-dev/neo-go/pkg/encoding/address" "github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest" "github.com/nspcc-dev/neo-go/pkg/util" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) const ( walletTemplate = "testWallet" ) func TestNewWallet(t *testing.T) { wallet := checkWalletConstructor(t) require.NotNil(t, wallet) } func TestNewWalletFromFile_Negative_EmptyFile(t *testing.T) { _ = checkWalletConstructor(t) walletFromFile, err2 := NewWalletFromFile(walletTemplate) require.Errorf(t, err2, "EOF") require.Nil(t, walletFromFile) } func TestNewWalletFromFile_Negative_NoFile(t *testing.T) { _, err := NewWalletFromFile(walletTemplate) require.Errorf(t, err, "open testWallet: no such file or directory") } func TestCreateAccountAndClose(t *testing.T) { wallet := checkWalletConstructor(t) errAcc := wallet.CreateAccount("testName", "testPass") require.NoError(t, errAcc) accounts := wallet.Accounts require.Len(t, accounts, 1) require.True(t, wallet.Accounts[0].CanSign()) wallet.Close() require.False(t, wallet.Accounts[0].CanSign()) } func TestAddAccount(t *testing.T) { wallets := []*Wallet{ checkWalletConstructor(t), NewInMemoryWallet(), } for _, w := range wallets { w.AddAccount(&Account{ privateKey: nil, Address: "real", EncryptedWIF: "", Label: "", Contract: nil, Locked: false, Default: false, }) accounts := w.Accounts require.Len(t, accounts, 1) require.Error(t, w.RemoveAccount("abc")) require.Len(t, w.Accounts, 1) require.NoError(t, w.RemoveAccount("real")) require.Len(t, w.Accounts, 0) } } func TestPath(t *testing.T) { wallet := checkWalletConstructor(t) path := wallet.Path() require.NotEmpty(t, path) } func TestSave(t *testing.T) { inMemWallet := NewInMemoryWallet() tmpDir := t.TempDir() file := filepath.Join(tmpDir, walletTemplate) inMemWallet.SetPath(file) wallets := []*Wallet{ checkWalletConstructor(t), inMemWallet, } for _, w := range wallets { w.AddAccount(&Account{ privateKey: nil, Address: "", EncryptedWIF: "", Label: "", Contract: nil, Locked: false, Default: false, }) errForSave := w.Save() require.NoError(t, errForSave) openedWallet, err := NewWalletFromFile(w.path) require.NoError(t, err) require.Equal(t, w.Accounts, openedWallet.Accounts) t.Run("change and rewrite", func(t *testing.T) { err := openedWallet.CreateAccount("test", "pass") require.NoError(t, err) w2, err := NewWalletFromFile(openedWallet.path) require.NoError(t, err) require.Equal(t, 2, len(w2.Accounts)) require.NoError(t, w2.Accounts[1].Decrypt("pass", w2.Scrypt)) _ = w2.Accounts[1].ScriptHash() // openedWallet has it for acc 1. require.Equal(t, openedWallet.Accounts, w2.Accounts) }) } } func TestJSONMarshallUnmarshal(t *testing.T) { wallet := checkWalletConstructor(t) bytes, err := wallet.JSON() require.NoError(t, err) require.NotNil(t, bytes) unmarshalledWallet := &Wallet{} errUnmarshal := json.Unmarshal(bytes, unmarshalledWallet) require.NoError(t, errUnmarshal) require.Equal(t, wallet.Version, unmarshalledWallet.Version) require.Equal(t, wallet.Accounts, unmarshalledWallet.Accounts) require.Equal(t, wallet.Scrypt, unmarshalledWallet.Scrypt) } func checkWalletConstructor(t *testing.T) *Wallet { tmpDir := t.TempDir() file := filepath.Join(tmpDir, walletTemplate) wallet, err := NewWallet(file) require.NoError(t, err) return wallet } func TestWallet_AddToken(t *testing.T) { w := checkWalletConstructor(t) tok := NewToken(util.Uint160{1, 2, 3}, "Rubl", "RUB", 2, manifest.NEP17StandardName) require.Equal(t, 0, len(w.Extra.Tokens)) w.AddToken(tok) require.Equal(t, 1, len(w.Extra.Tokens)) require.Error(t, w.RemoveToken(util.Uint160{4, 5, 6})) require.Equal(t, 1, len(w.Extra.Tokens)) require.NoError(t, w.RemoveToken(tok.Hash)) require.Equal(t, 0, len(w.Extra.Tokens)) } func TestWallet_GetAccount(t *testing.T) { wallet := checkWalletConstructor(t) accounts := []*Account{ { Contract: &Contract{ Script: []byte{0, 1, 2, 3}, }, }, { Contract: &Contract{ Script: []byte{3, 2, 1, 0}, }, }, } for _, acc := range accounts { acc.Address = address.Uint160ToString(acc.Contract.ScriptHash()) wallet.AddAccount(acc) } for i, acc := range accounts { h := acc.Contract.ScriptHash() assert.Equal(t, acc, wallet.GetAccount(h), "can't get %d account", i) } } func TestWalletGetChangeAddress(t *testing.T) { w1, err := NewWalletFromFile("testdata/wallet1.json") require.NoError(t, err) sh := w1.GetChangeAddress() // No default address, the first one is used. require.Equal(t, "Nhfg3TbpwogLvDGVvAvqyThbsHgoSUKwtn", address.Uint160ToString(sh)) w2, err := NewWalletFromFile("testdata/wallet2.json") require.NoError(t, err) sh = w2.GetChangeAddress() // Default address. require.Equal(t, "NMUedC8TSV2rE17wGguSvPk9XcmHSaT275", address.Uint160ToString(sh)) } func TestWalletForExamples(t *testing.T) { const ( examplesDir = "../../examples" walletFile = "my_wallet.json" walletPass = "qwerty" accountLabel = "my_account" ) w, err := NewWalletFromFile(path.Join(examplesDir, walletFile)) require.NoError(t, err) require.Equal(t, 1, len(w.Accounts)) require.Equal(t, accountLabel, w.Accounts[0].Label) require.NoError(t, w.Accounts[0].Decrypt(walletPass, w.Scrypt)) // we need to keep the owner of the example contracts the same as the wallet account require.Equal(t, "NbrUYaZgyhSkNoRo9ugRyEMdUZxrhkNaWB", w.Accounts[0].Address, "need to change `owner` in the example contracts") } func TestFromBytes(t *testing.T) { wallet := checkWalletConstructor(t) bts, err := wallet.JSON() require.NoError(t, err) w, err := NewWalletFromBytes(bts) require.NoError(t, err) require.Len(t, w.path, 0) w.SetPath(wallet.path) require.Equal(t, wallet, w) }