// Copyright 2016 The go-ethereum Authors // This file is part of go-ethereum. // // go-ethereum is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // go-ethereum is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with go-ethereum. If not, see . package main import ( "os" "path/filepath" "runtime" "testing" "github.com/cespare/cp" ) // These tests are 'smoke tests' for the account related // subcommands and flags. // // For most tests, the test files from package accounts // are copied into a temporary keystore directory. func tmpDatadirWithKeystore(t *testing.T) string { datadir := t.TempDir() keystore := filepath.Join(datadir, "keystore") source := filepath.Join("..", "..", "accounts", "keystore", "testdata", "keystore") if err := cp.CopyAll(keystore, source); err != nil { t.Fatal(err) } return datadir } func TestAccountListEmpty(t *testing.T) { t.Parallel() geth := runGeth(t, "account", "list") geth.ExpectExit() } func TestAccountList(t *testing.T) { t.Parallel() datadir := tmpDatadirWithKeystore(t) var want = ` Account #0: {7ef5a6135f1fd6a02593eedc869c6d41d934aef8} keystore://{{.Datadir}}/keystore/UTC--2016-03-22T12-57-55.920751759Z--7ef5a6135f1fd6a02593eedc869c6d41d934aef8 Account #1: {f466859ead1932d743d622cb74fc058882e8648a} keystore://{{.Datadir}}/keystore/aaa Account #2: {289d485d9771714cce91d3393d764e1311907acc} keystore://{{.Datadir}}/keystore/zzz ` if runtime.GOOS == "windows" { want = ` Account #0: {7ef5a6135f1fd6a02593eedc869c6d41d934aef8} keystore://{{.Datadir}}\keystore\UTC--2016-03-22T12-57-55.920751759Z--7ef5a6135f1fd6a02593eedc869c6d41d934aef8 Account #1: {f466859ead1932d743d622cb74fc058882e8648a} keystore://{{.Datadir}}\keystore\aaa Account #2: {289d485d9771714cce91d3393d764e1311907acc} keystore://{{.Datadir}}\keystore\zzz ` } { geth := runGeth(t, "account", "list", "--datadir", datadir) geth.Expect(want) geth.ExpectExit() } { geth := runGeth(t, "--datadir", datadir, "account", "list") geth.Expect(want) geth.ExpectExit() } } func TestAccountNew(t *testing.T) { t.Parallel() geth := runGeth(t, "account", "new", "--lightkdf") defer geth.ExpectExit() geth.Expect(` Your new account is locked with a password. Please give a password. Do not forget this password. !! Unsupported terminal, password will be echoed. Password: {{.InputLine "foobar"}} Repeat password: {{.InputLine "foobar"}} Your new key was generated `) geth.ExpectRegexp(` Public address of the key: 0x[0-9a-fA-F]{40} Path of the secret key file: .*UTC--.+--[0-9a-f]{40} - You can share your public address with anyone. Others need it to interact with you. - You must NEVER share the secret key with anyone! The key controls access to your funds! - You must BACKUP your key file! Without the key, it's impossible to access account funds! - You must REMEMBER your password! Without the password, it's impossible to decrypt the key! `) } func TestAccountImport(t *testing.T) { t.Parallel() tests := []struct{ name, key, output string }{ { name: "correct account", key: "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef", output: "Address: {fcad0b19bb29d4674531d6f115237e16afce377c}\n", }, { name: "invalid character", key: "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef1", output: "Fatal: Failed to load the private key: invalid character '1' at end of key file\n", }, } for _, test := range tests { t.Run(test.name, func(t *testing.T) { t.Parallel() importAccountWithExpect(t, test.key, test.output) }) } } func TestAccountHelp(t *testing.T) { t.Parallel() geth := runGeth(t, "account", "-h") geth.WaitExit() if have, want := geth.ExitStatus(), 0; have != want { t.Errorf("exit error, have %d want %d", have, want) } geth = runGeth(t, "account", "import", "-h") geth.WaitExit() if have, want := geth.ExitStatus(), 0; have != want { t.Errorf("exit error, have %d want %d", have, want) } } func importAccountWithExpect(t *testing.T, key string, expected string) { dir := t.TempDir() keyfile := filepath.Join(dir, "key.prv") if err := os.WriteFile(keyfile, []byte(key), 0600); err != nil { t.Error(err) } passwordFile := filepath.Join(dir, "password.txt") if err := os.WriteFile(passwordFile, []byte("foobar"), 0600); err != nil { t.Error(err) } geth := runGeth(t, "--lightkdf", "account", "import", "-password", passwordFile, keyfile) defer geth.ExpectExit() geth.Expect(expected) } func TestAccountNewBadRepeat(t *testing.T) { t.Parallel() geth := runGeth(t, "account", "new", "--lightkdf") defer geth.ExpectExit() geth.Expect(` Your new account is locked with a password. Please give a password. Do not forget this password. !! Unsupported terminal, password will be echoed. Password: {{.InputLine "something"}} Repeat password: {{.InputLine "something else"}} Fatal: Passwords do not match `) } func TestAccountUpdate(t *testing.T) { t.Parallel() datadir := tmpDatadirWithKeystore(t) geth := runGeth(t, "account", "update", "--datadir", datadir, "--lightkdf", "f466859ead1932d743d622cb74fc058882e8648a") defer geth.ExpectExit() geth.Expect(` Please give a NEW password. Do not forget this password. !! Unsupported terminal, password will be echoed. Password: {{.InputLine "foobar2"}} Repeat password: {{.InputLine "foobar2"}} Please provide the OLD password for account f466859ead1932d743d622cb74fc058882e8648a | Attempt 1/3 Password: {{.InputLine "foobar"}} `) } func TestWalletImport(t *testing.T) { t.Parallel() geth := runGeth(t, "wallet", "import", "--lightkdf", "testdata/guswallet.json") defer geth.ExpectExit() geth.Expect(` !! Unsupported terminal, password will be echoed. Password: {{.InputLine "foo"}} Address: {d4584b5f6229b7be90727b0fc8c6b91bb427821f} `) files, err := os.ReadDir(filepath.Join(geth.Datadir, "keystore")) if len(files) != 1 { t.Errorf("expected one key file in keystore directory, found %d files (error: %v)", len(files), err) } } func TestWalletImportBadPassword(t *testing.T) { t.Parallel() geth := runGeth(t, "wallet", "import", "--lightkdf", "testdata/guswallet.json") defer geth.ExpectExit() geth.Expect(` !! Unsupported terminal, password will be echoed. Password: {{.InputLine "wrong"}} Fatal: could not decrypt key with given password `) }