2019-02-08 12:45:23 -08:00
|
|
|
// +build ledger,test_ledger_mock
|
|
|
|
|
|
|
|
package crypto
|
|
|
|
|
|
|
|
import (
|
2019-02-25 03:34:39 -08:00
|
|
|
"fmt"
|
2019-05-06 09:50:05 -07:00
|
|
|
|
2019-02-08 12:45:23 -08:00
|
|
|
"github.com/btcsuite/btcd/btcec"
|
|
|
|
"github.com/pkg/errors"
|
2019-04-04 07:36:39 -07:00
|
|
|
|
2019-06-05 16:26:17 -07:00
|
|
|
secp256k1 "github.com/tendermint/btcd/btcec"
|
|
|
|
"github.com/tendermint/tendermint/crypto"
|
|
|
|
tmsecp256k1 "github.com/tendermint/tendermint/crypto/secp256k1"
|
|
|
|
|
2019-05-06 09:50:05 -07:00
|
|
|
bip39 "github.com/cosmos/go-bip39"
|
|
|
|
|
2020-04-08 02:38:28 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/crypto/hd"
|
2019-04-04 07:36:39 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/tests"
|
2019-06-05 16:26:17 -07:00
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
2019-02-08 12:45:23 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
// If ledger support (build tag) has been enabled, which implies a CGO dependency,
|
|
|
|
// set the discoverLedger function which is responsible for loading the Ledger
|
|
|
|
// device at runtime or returning an error.
|
|
|
|
func init() {
|
|
|
|
discoverLedger = func() (LedgerSECP256K1, error) {
|
|
|
|
return LedgerSECP256K1Mock{}, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type LedgerSECP256K1Mock struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mock LedgerSECP256K1Mock) Close() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-04-30 01:58:21 -07:00
|
|
|
// GetPublicKeySECP256K1 mocks a ledger device
|
|
|
|
// as per the original API, it returns an uncompressed key
|
2019-02-08 12:45:23 -08:00
|
|
|
func (mock LedgerSECP256K1Mock) GetPublicKeySECP256K1(derivationPath []uint32) ([]byte, error) {
|
|
|
|
if derivationPath[0] != 44 {
|
|
|
|
return nil, errors.New("Invalid derivation path")
|
|
|
|
}
|
2019-07-29 11:01:53 -07:00
|
|
|
|
|
|
|
if derivationPath[1] != sdk.GetConfig().GetCoinType() {
|
2019-02-08 12:45:23 -08:00
|
|
|
return nil, errors.New("Invalid derivation path")
|
|
|
|
}
|
|
|
|
|
|
|
|
seed, err := bip39.NewSeedWithErrorChecking(tests.TestMnemonic, "")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
path := hd.NewParams(derivationPath[0], derivationPath[1], derivationPath[2], derivationPath[3] != 0, derivationPath[4])
|
|
|
|
masterPriv, ch := hd.ComputeMastersFromSeed(seed)
|
|
|
|
derivedPriv, err := hd.DerivePrivateKeyForPath(masterPriv, ch, path.String())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
_, pubkeyObject := secp256k1.PrivKeyFromBytes(secp256k1.S256(), derivedPriv[:])
|
|
|
|
|
|
|
|
return pubkeyObject.SerializeUncompressed(), nil
|
|
|
|
}
|
|
|
|
|
2019-04-30 01:58:21 -07:00
|
|
|
// GetAddressPubKeySECP256K1 mocks a ledger device
|
|
|
|
// as per the original API, it returns a compressed key and a bech32 address
|
|
|
|
func (mock LedgerSECP256K1Mock) GetAddressPubKeySECP256K1(derivationPath []uint32, hrp string) ([]byte, string, error) {
|
|
|
|
pk, err := mock.GetPublicKeySECP256K1(derivationPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
// re-serialize in the 33-byte compressed format
|
|
|
|
cmp, err := btcec.ParsePubKey(pk[:], btcec.S256())
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", fmt.Errorf("error parsing public key: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var compressedPublicKey tmsecp256k1.PubKeySecp256k1
|
|
|
|
copy(compressedPublicKey[:], cmp.SerializeCompressed())
|
|
|
|
|
|
|
|
// Generate the bech32 addr using existing tmcrypto/etc.
|
2019-06-05 16:26:17 -07:00
|
|
|
addr := sdk.AccAddress(compressedPublicKey.Address()).String()
|
2019-04-30 01:58:21 -07:00
|
|
|
return pk, addr, err
|
|
|
|
}
|
|
|
|
|
2019-02-08 12:45:23 -08:00
|
|
|
func (mock LedgerSECP256K1Mock) SignSECP256K1(derivationPath []uint32, message []byte) ([]byte, error) {
|
|
|
|
path := hd.NewParams(derivationPath[0], derivationPath[1], derivationPath[2], derivationPath[3] != 0, derivationPath[4])
|
|
|
|
seed, err := bip39.NewSeedWithErrorChecking(tests.TestMnemonic, "")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
masterPriv, ch := hd.ComputeMastersFromSeed(seed)
|
|
|
|
derivedPriv, err := hd.DerivePrivateKeyForPath(masterPriv, ch, path.String())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
priv, _ := secp256k1.PrivKeyFromBytes(secp256k1.S256(), derivedPriv[:])
|
|
|
|
|
|
|
|
sig, err := priv.Sign(crypto.Sha256(message))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Need to return DER as the ledger does
|
|
|
|
sig2 := btcec.Signature{R: sig.R, S: sig.S}
|
|
|
|
return sig2.Serialize(), nil
|
|
|
|
}
|
2019-02-25 03:34:39 -08:00
|
|
|
|
|
|
|
// ShowAddressSECP256K1 shows the address for the corresponding bip32 derivation path
|
|
|
|
func (mock LedgerSECP256K1Mock) ShowAddressSECP256K1(bip32Path []uint32, hrp string) error {
|
|
|
|
fmt.Printf("Request to show address for %v at %v", hrp, bip32Path)
|
|
|
|
return nil
|
|
|
|
}
|