136 lines
3.9 KiB
Go
136 lines
3.9 KiB
Go
|
// TODO: Make independent version for Nimbus
|
||
|
|
||
|
package account
|
||
|
|
||
|
import (
|
||
|
"crypto/ecdsa"
|
||
|
"errors"
|
||
|
"io/ioutil"
|
||
|
"os"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/ethereum/go-ethereum/accounts"
|
||
|
"github.com/ethereum/go-ethereum/accounts/keystore"
|
||
|
"github.com/ethereum/go-ethereum/common"
|
||
|
"github.com/status-im/status-go/eth-node/types"
|
||
|
"github.com/status-im/status-go/extkeys"
|
||
|
)
|
||
|
|
||
|
// makeAccountManager creates ethereum accounts.Manager with single disk backend and lightweight kdf.
|
||
|
// If keydir is empty new temporary directory with go-ethereum-keystore will be intialized.
|
||
|
func makeAccountManager(keydir string) (manager *accounts.Manager, err error) {
|
||
|
if keydir == "" {
|
||
|
// There is no datadir.
|
||
|
keydir, err = ioutil.TempDir("", "go-ethereum-keystore")
|
||
|
}
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := os.MkdirAll(keydir, 0700); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
config := accounts.Config{InsecureUnlockAllowed: false}
|
||
|
return accounts.NewManager(&config, keystore.NewKeyStore(keydir, keystore.LightScryptN, keystore.LightScryptP)), nil
|
||
|
}
|
||
|
|
||
|
func makeKeyStore(manager *accounts.Manager) (types.KeyStore, error) {
|
||
|
backends := manager.Backends(keystore.KeyStoreType)
|
||
|
if len(backends) == 0 {
|
||
|
return nil, ErrAccountKeyStoreMissing
|
||
|
}
|
||
|
keyStore, ok := backends[0].(*keystore.KeyStore)
|
||
|
if !ok {
|
||
|
return nil, ErrAccountKeyStoreMissing
|
||
|
}
|
||
|
|
||
|
return adaptGethKeyStore(keyStore), nil
|
||
|
}
|
||
|
|
||
|
type gethKeyStoreAdapter struct {
|
||
|
keystore *keystore.KeyStore
|
||
|
}
|
||
|
|
||
|
func adaptGethKeyStore(keystore *keystore.KeyStore) types.KeyStore {
|
||
|
return &gethKeyStoreAdapter{keystore: keystore}
|
||
|
}
|
||
|
|
||
|
func (k *gethKeyStoreAdapter) ImportECDSA(priv *ecdsa.PrivateKey, passphrase string) (types.Account, error) {
|
||
|
gethAccount, err := k.keystore.ImportECDSA(priv, passphrase)
|
||
|
return accountFrom(gethAccount), err
|
||
|
}
|
||
|
|
||
|
func (k *gethKeyStoreAdapter) ImportSingleExtendedKey(extKey *extkeys.ExtendedKey, passphrase string) (types.Account, error) {
|
||
|
gethAccount, err := k.keystore.ImportSingleExtendedKey(extKey, passphrase)
|
||
|
return accountFrom(gethAccount), err
|
||
|
}
|
||
|
|
||
|
func (k *gethKeyStoreAdapter) ImportExtendedKeyForPurpose(keyPurpose extkeys.KeyPurpose, extKey *extkeys.ExtendedKey, passphrase string) (types.Account, error) {
|
||
|
gethAccount, err := k.keystore.ImportExtendedKeyForPurpose(keyPurpose, extKey, passphrase)
|
||
|
return accountFrom(gethAccount), err
|
||
|
}
|
||
|
|
||
|
func (k *gethKeyStoreAdapter) AccountDecryptedKey(a types.Account, auth string) (types.Account, *types.Key, error) {
|
||
|
gethAccount, err := gethAccountFrom(a)
|
||
|
if err != nil {
|
||
|
return types.Account{}, nil, err
|
||
|
}
|
||
|
|
||
|
var gethKey *keystore.Key
|
||
|
gethAccount, gethKey, err = k.keystore.AccountDecryptedKey(gethAccount, auth)
|
||
|
return accountFrom(gethAccount), keyFrom(gethKey), err
|
||
|
}
|
||
|
|
||
|
func (k *gethKeyStoreAdapter) Delete(a types.Account, auth string) error {
|
||
|
gethAccount, err := gethAccountFrom(a)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
return k.keystore.Delete(gethAccount, auth)
|
||
|
}
|
||
|
|
||
|
// parseGethURL converts a user supplied URL into the accounts specific structure.
|
||
|
func parseGethURL(url string) (accounts.URL, error) {
|
||
|
parts := strings.Split(url, "://")
|
||
|
if len(parts) != 2 || parts[0] == "" {
|
||
|
return accounts.URL{}, errors.New("protocol scheme missing")
|
||
|
}
|
||
|
return accounts.URL{
|
||
|
Scheme: parts[0],
|
||
|
Path: parts[1],
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
func gethAccountFrom(account types.Account) (accounts.Account, error) {
|
||
|
var (
|
||
|
gethAccount accounts.Account
|
||
|
err error
|
||
|
)
|
||
|
gethAccount.Address = common.Address(account.Address)
|
||
|
if account.URL != "" {
|
||
|
gethAccount.URL, err = parseGethURL(account.URL)
|
||
|
}
|
||
|
return gethAccount, err
|
||
|
}
|
||
|
|
||
|
func accountFrom(gethAccount accounts.Account) types.Account {
|
||
|
return types.Account{
|
||
|
Address: types.Address(gethAccount.Address),
|
||
|
URL: gethAccount.URL.String(),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func keyFrom(k *keystore.Key) *types.Key {
|
||
|
if k == nil {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
return &types.Key{
|
||
|
Id: k.Id,
|
||
|
Address: types.Address(k.Address),
|
||
|
PrivateKey: k.PrivateKey,
|
||
|
ExtendedKey: k.ExtendedKey,
|
||
|
SubAccountIndex: k.SubAccountIndex,
|
||
|
}
|
||
|
}
|