mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-17 15:28:45 +00:00
6ed0539723
* Allow importing from any file * Fix log * Add test * Fix test
354 lines
11 KiB
Go
354 lines
11 KiB
Go
package v2
|
|
|
|
import (
|
|
"context"
|
|
"crypto/rand"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"math/big"
|
|
"os"
|
|
"path/filepath"
|
|
"sort"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/prysmaticlabs/prysm/shared/bls"
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
"github.com/prysmaticlabs/prysm/shared/roughtime"
|
|
"github.com/prysmaticlabs/prysm/shared/testutil"
|
|
"github.com/prysmaticlabs/prysm/shared/testutil/assert"
|
|
"github.com/prysmaticlabs/prysm/shared/testutil/require"
|
|
v2keymanager "github.com/prysmaticlabs/prysm/validator/keymanager/v2"
|
|
"github.com/prysmaticlabs/prysm/validator/keymanager/v2/direct"
|
|
keystorev4 "github.com/wealdtech/go-eth2-wallet-encryptor-keystorev4"
|
|
)
|
|
|
|
func TestImport_Noninteractive(t *testing.T) {
|
|
walletDir, passwordsDir, passwordFilePath := setupWalletAndPasswordsDir(t)
|
|
randPath, err := rand.Int(rand.Reader, big.NewInt(1000000))
|
|
require.NoError(t, err, "Could not generate random file path")
|
|
keysDir := filepath.Join(testutil.TempDir(), fmt.Sprintf("/%d", randPath), "keysDir")
|
|
require.NoError(t, os.MkdirAll(keysDir, os.ModePerm))
|
|
t.Cleanup(func() {
|
|
require.NoError(t, os.RemoveAll(keysDir), "Failed to remove directory")
|
|
})
|
|
|
|
cliCtx := setupWalletCtx(t, &testWalletConfig{
|
|
walletDir: walletDir,
|
|
passwordsDir: passwordsDir,
|
|
keysDir: keysDir,
|
|
keymanagerKind: v2keymanager.Direct,
|
|
walletPasswordFile: passwordFilePath,
|
|
accountPasswordFile: passwordFilePath,
|
|
})
|
|
wallet, err := NewWallet(cliCtx, v2keymanager.Direct)
|
|
require.NoError(t, err)
|
|
require.NoError(t, wallet.SaveWallet())
|
|
ctx := context.Background()
|
|
encodedCfg, err := direct.MarshalConfigFile(ctx, direct.DefaultConfig())
|
|
require.NoError(t, err)
|
|
require.NoError(t, wallet.WriteKeymanagerConfigToDisk(ctx, encodedCfg))
|
|
keymanager, err := direct.NewKeymanager(
|
|
cliCtx,
|
|
wallet,
|
|
direct.DefaultConfig(),
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
// Make sure there are no accounts at the start.
|
|
accounts, err := keymanager.ValidatingAccountNames()
|
|
require.NoError(t, err)
|
|
assert.Equal(t, len(accounts), 0)
|
|
|
|
// Create 2 keys.
|
|
createKeystore(t, keysDir)
|
|
time.Sleep(time.Second)
|
|
createKeystore(t, keysDir)
|
|
|
|
require.NoError(t, ImportAccounts(cliCtx))
|
|
|
|
wallet, err = OpenWallet(cliCtx)
|
|
require.NoError(t, err)
|
|
km, err := wallet.InitializeKeymanager(cliCtx, true)
|
|
require.NoError(t, err)
|
|
keys, err := km.FetchValidatingPublicKeys(ctx)
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, 2, len(keys))
|
|
}
|
|
|
|
func TestImport_Noninteractive_RandomName(t *testing.T) {
|
|
walletDir, passwordsDir, passwordFilePath := setupWalletAndPasswordsDir(t)
|
|
randPath, err := rand.Int(rand.Reader, big.NewInt(1000000))
|
|
require.NoError(t, err, "Could not generate random file path")
|
|
keysDir := filepath.Join(testutil.TempDir(), fmt.Sprintf("/%d", randPath), "keysDir")
|
|
require.NoError(t, os.MkdirAll(keysDir, os.ModePerm))
|
|
t.Cleanup(func() {
|
|
require.NoError(t, os.RemoveAll(keysDir), "Failed to remove directory")
|
|
})
|
|
|
|
cliCtx := setupWalletCtx(t, &testWalletConfig{
|
|
walletDir: walletDir,
|
|
passwordsDir: passwordsDir,
|
|
keysDir: keysDir,
|
|
keymanagerKind: v2keymanager.Direct,
|
|
walletPasswordFile: passwordFilePath,
|
|
accountPasswordFile: passwordFilePath,
|
|
})
|
|
wallet, err := NewWallet(cliCtx, v2keymanager.Direct)
|
|
require.NoError(t, err)
|
|
require.NoError(t, wallet.SaveWallet())
|
|
ctx := context.Background()
|
|
encodedCfg, err := direct.MarshalConfigFile(ctx, direct.DefaultConfig())
|
|
require.NoError(t, err)
|
|
require.NoError(t, wallet.WriteKeymanagerConfigToDisk(ctx, encodedCfg))
|
|
keymanager, err := direct.NewKeymanager(
|
|
cliCtx,
|
|
wallet,
|
|
direct.DefaultConfig(),
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
// Make sure there are no accounts at the start.
|
|
accounts, err := keymanager.ValidatingAccountNames()
|
|
require.NoError(t, err)
|
|
assert.Equal(t, len(accounts), 0)
|
|
|
|
// Create 2 keys.
|
|
createRandomNameKeystore(t, keysDir)
|
|
time.Sleep(time.Second)
|
|
createRandomNameKeystore(t, keysDir)
|
|
|
|
require.NoError(t, ImportAccounts(cliCtx))
|
|
|
|
wallet, err = OpenWallet(cliCtx)
|
|
require.NoError(t, err)
|
|
km, err := wallet.InitializeKeymanager(cliCtx, true)
|
|
require.NoError(t, err)
|
|
keys, err := km.FetchValidatingPublicKeys(ctx)
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, 2, len(keys))
|
|
}
|
|
|
|
func TestImport_Noninteractive_Filepath(t *testing.T) {
|
|
walletDir, passwordsDir, passwordFilePath := setupWalletAndPasswordsDir(t)
|
|
randPath, err := rand.Int(rand.Reader, big.NewInt(1000000))
|
|
require.NoError(t, err, "Could not generate random file path")
|
|
keysDir := filepath.Join(testutil.TempDir(), fmt.Sprintf("/%d", randPath), "keysDir")
|
|
require.NoError(t, os.MkdirAll(keysDir, os.ModePerm))
|
|
t.Cleanup(func() {
|
|
require.NoError(t, os.RemoveAll(keysDir), "Failed to remove directory")
|
|
})
|
|
|
|
_, keystorePath := createKeystore(t, keysDir)
|
|
cliCtx := setupWalletCtx(t, &testWalletConfig{
|
|
walletDir: walletDir,
|
|
passwordsDir: passwordsDir,
|
|
keysDir: keystorePath,
|
|
keymanagerKind: v2keymanager.Direct,
|
|
walletPasswordFile: passwordFilePath,
|
|
accountPasswordFile: passwordFilePath,
|
|
})
|
|
wallet, err := NewWallet(cliCtx, v2keymanager.Direct)
|
|
require.NoError(t, err)
|
|
require.NoError(t, wallet.SaveWallet())
|
|
ctx := context.Background()
|
|
keymanagerCfg := direct.DefaultConfig()
|
|
encodedCfg, err := direct.MarshalConfigFile(ctx, keymanagerCfg)
|
|
require.NoError(t, err)
|
|
require.NoError(t, wallet.WriteKeymanagerConfigToDisk(ctx, encodedCfg))
|
|
keymanager, err := direct.NewKeymanager(
|
|
cliCtx,
|
|
wallet,
|
|
keymanagerCfg,
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
// Make sure there are no accounts at the start.
|
|
accounts, err := keymanager.ValidatingAccountNames()
|
|
require.NoError(t, err)
|
|
assert.Equal(t, len(accounts), 0)
|
|
|
|
require.NoError(t, ImportAccounts(cliCtx))
|
|
|
|
wallet, err = OpenWallet(cliCtx)
|
|
require.NoError(t, err)
|
|
km, err := wallet.InitializeKeymanager(cliCtx, true)
|
|
require.NoError(t, err)
|
|
keys, err := km.FetchValidatingPublicKeys(ctx)
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, 1, len(keys))
|
|
}
|
|
|
|
func TestImport_SortByDerivationPath(t *testing.T) {
|
|
type test struct {
|
|
name string
|
|
input []string
|
|
want []string
|
|
}
|
|
tests := []test{
|
|
{
|
|
name: "Basic sort",
|
|
input: []string{
|
|
"keystore_m_12381_3600_2_0_0.json",
|
|
"keystore_m_12381_3600_1_0_0.json",
|
|
"keystore_m_12381_3600_0_0_0.json",
|
|
},
|
|
want: []string{
|
|
"keystore_m_12381_3600_0_0_0.json",
|
|
"keystore_m_12381_3600_1_0_0.json",
|
|
"keystore_m_12381_3600_2_0_0.json",
|
|
},
|
|
},
|
|
{
|
|
name: "Large digit accounts",
|
|
input: []string{
|
|
"keystore_m_12381_3600_30020330_0_0.json",
|
|
"keystore_m_12381_3600_430490934_0_0.json",
|
|
"keystore_m_12381_3600_0_0_0.json",
|
|
"keystore_m_12381_3600_333_0_0.json",
|
|
},
|
|
want: []string{
|
|
"keystore_m_12381_3600_0_0_0.json",
|
|
"keystore_m_12381_3600_333_0_0.json",
|
|
"keystore_m_12381_3600_30020330_0_0.json",
|
|
"keystore_m_12381_3600_430490934_0_0.json",
|
|
},
|
|
},
|
|
{
|
|
name: "Some filenames with derivation path, others without",
|
|
input: []string{
|
|
"keystore_m_12381_3600_4_0_0.json",
|
|
"keystore.json",
|
|
"keystore-2309023.json",
|
|
"keystore_m_12381_3600_1_0_0.json",
|
|
"keystore_m_12381_3600_3_0_0.json",
|
|
},
|
|
want: []string{
|
|
"keystore_m_12381_3600_1_0_0.json",
|
|
"keystore_m_12381_3600_3_0_0.json",
|
|
"keystore_m_12381_3600_4_0_0.json",
|
|
"keystore.json",
|
|
"keystore-2309023.json",
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
sort.Sort(byDerivationPath(tt.input))
|
|
assert.DeepEqual(t, tt.want, tt.input)
|
|
})
|
|
}
|
|
}
|
|
|
|
func Test_importPrivateKeyAsAccount(t *testing.T) {
|
|
walletDir, _, passwordFilePath := setupWalletAndPasswordsDir(t)
|
|
randPath, err := rand.Int(rand.Reader, big.NewInt(1000000))
|
|
require.NoError(t, err, "Could not generate random file path")
|
|
privKeyDir := filepath.Join(testutil.TempDir(), fmt.Sprintf("/%d", randPath), "privKeys")
|
|
require.NoError(t, os.MkdirAll(privKeyDir, os.ModePerm))
|
|
t.Cleanup(func() {
|
|
require.NoError(t, os.RemoveAll(privKeyDir), "Failed to remove directory")
|
|
})
|
|
privKeyFileName := filepath.Join(privKeyDir, "privatekey.txt")
|
|
|
|
// We create a new private key and save it to a file on disk.
|
|
privKey := bls.RandKey()
|
|
privKeyHex := fmt.Sprintf("%x", privKey.Marshal())
|
|
require.NoError(
|
|
t,
|
|
ioutil.WriteFile(privKeyFileName, []byte(privKeyHex), params.BeaconIoConfig().ReadWritePermissions),
|
|
)
|
|
|
|
// We instantiate a new wallet from a cli context.
|
|
cliCtx := setupWalletCtx(t, &testWalletConfig{
|
|
walletDir: walletDir,
|
|
keymanagerKind: v2keymanager.Direct,
|
|
walletPasswordFile: passwordFilePath,
|
|
privateKeyFile: privKeyFileName,
|
|
})
|
|
wallet, err := NewWallet(cliCtx, v2keymanager.Direct)
|
|
require.NoError(t, err)
|
|
require.NoError(t, wallet.SaveWallet())
|
|
|
|
// We create a new direct keymanager for the wallet.
|
|
ctx := context.Background()
|
|
keymanagerCfg := direct.DefaultConfig()
|
|
encodedCfg, err := direct.MarshalConfigFile(ctx, keymanagerCfg)
|
|
require.NoError(t, err)
|
|
require.NoError(t, wallet.WriteKeymanagerConfigToDisk(ctx, encodedCfg))
|
|
keymanager, err := direct.NewKeymanager(
|
|
cliCtx,
|
|
wallet,
|
|
keymanagerCfg,
|
|
)
|
|
require.NoError(t, err)
|
|
assert.NoError(t, importPrivateKeyAsAccount(cliCtx, wallet, keymanager))
|
|
|
|
// We re-instantiate the keymanager and check we now have 1 public key.
|
|
keymanager, err = direct.NewKeymanager(
|
|
cliCtx,
|
|
wallet,
|
|
keymanagerCfg,
|
|
)
|
|
require.NoError(t, err)
|
|
pubKeys, err := keymanager.FetchValidatingPublicKeys(ctx)
|
|
require.NoError(t, err)
|
|
require.Equal(t, 1, len(pubKeys))
|
|
assert.DeepEqual(t, pubKeys[0], bytesutil.ToBytes48(privKey.PublicKey().Marshal()))
|
|
}
|
|
|
|
// Returns the fullPath to the newly created keystore file.
|
|
func createKeystore(t *testing.T, path string) (*v2keymanager.Keystore, string) {
|
|
validatingKey := bls.RandKey()
|
|
encryptor := keystorev4.New()
|
|
cryptoFields, err := encryptor.Encrypt(validatingKey.Marshal(), password)
|
|
require.NoError(t, err)
|
|
id, err := uuid.NewRandom()
|
|
require.NoError(t, err)
|
|
keystoreFile := &v2keymanager.Keystore{
|
|
Crypto: cryptoFields,
|
|
ID: id.String(),
|
|
Pubkey: fmt.Sprintf("%x", validatingKey.PublicKey().Marshal()),
|
|
Version: encryptor.Version(),
|
|
Name: encryptor.Name(),
|
|
}
|
|
encoded, err := json.MarshalIndent(keystoreFile, "", "\t")
|
|
require.NoError(t, err)
|
|
// Write the encoded keystore to disk with the timestamp appended
|
|
createdAt := roughtime.Now().Unix()
|
|
fullPath := filepath.Join(path, fmt.Sprintf(direct.KeystoreFileNameFormat, createdAt))
|
|
require.NoError(t, ioutil.WriteFile(fullPath, encoded, os.ModePerm))
|
|
return keystoreFile, fullPath
|
|
}
|
|
|
|
// Returns the fullPath to the newly created keystore file.
|
|
func createRandomNameKeystore(t *testing.T, path string) (*v2keymanager.Keystore, string) {
|
|
validatingKey := bls.RandKey()
|
|
encryptor := keystorev4.New()
|
|
cryptoFields, err := encryptor.Encrypt(validatingKey.Marshal(), password)
|
|
require.NoError(t, err)
|
|
id, err := uuid.NewRandom()
|
|
require.NoError(t, err)
|
|
keystoreFile := &v2keymanager.Keystore{
|
|
Crypto: cryptoFields,
|
|
ID: id.String(),
|
|
Pubkey: fmt.Sprintf("%x", validatingKey.PublicKey().Marshal()),
|
|
Version: encryptor.Version(),
|
|
Name: encryptor.Name(),
|
|
}
|
|
encoded, err := json.MarshalIndent(keystoreFile, "", "\t")
|
|
require.NoError(t, err)
|
|
// Write the encoded keystore to disk with the timestamp appended
|
|
random, err := rand.Int(rand.Reader, big.NewInt(1000000))
|
|
require.NoError(t, err)
|
|
fullPath := filepath.Join(path, fmt.Sprintf("test-%d-keystore", random.Int64()))
|
|
require.NoError(t, ioutil.WriteFile(fullPath, encoded, os.ModePerm))
|
|
return keystoreFile, fullPath
|
|
}
|