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 }