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)
}