keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
package keychain
|
|
|
|
|
2018-12-10 03:30:59 +01:00
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
2022-02-23 14:48:00 +01:00
|
|
|
"github.com/btcsuite/btcd/btcec/v2"
|
|
|
|
"github.com/btcsuite/btcd/btcec/v2/ecdsa"
|
2018-12-10 03:30:59 +01:00
|
|
|
)
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
|
|
|
|
const (
|
2022-05-12 12:47:13 +02:00
|
|
|
// KeyDerivationVersionLegacy is the previous version of the key
|
|
|
|
// derivation schema defined below. We use a version as this means that
|
|
|
|
// we'll be able to accept new seed in the future and be able to discern
|
|
|
|
// if the software is compatible with the version of the seed.
|
|
|
|
KeyDerivationVersionLegacy = 0
|
|
|
|
|
|
|
|
// KeyDerivationVersionTaproot is the most recent version of the key
|
|
|
|
// derivation scheme that marks the introduction of the Taproot
|
|
|
|
// derivation with BIP0086 support.
|
|
|
|
KeyDerivationVersionTaproot = 1
|
|
|
|
|
|
|
|
// CurrentKeyDerivationVersion is the current default key derivation
|
|
|
|
// version that is used for new seeds.
|
|
|
|
CurrentKeyDerivationVersion = KeyDerivationVersionTaproot
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
|
|
|
|
// BIP0043Purpose is the "purpose" value that we'll use for the first
|
|
|
|
// version or our key derivation scheme. All keys are expected to be
|
|
|
|
// derived from this purpose, then the particular coin type of the
|
|
|
|
// chain where the keys are to be used. Slightly adhering to BIP0043
|
|
|
|
// allows us to not deviate too far from a widely used standard, and
|
|
|
|
// also fits into existing implementations of the BIP's template.
|
|
|
|
//
|
|
|
|
// NOTE: BRICK SQUUUUUAD.
|
|
|
|
BIP0043Purpose = 1017
|
|
|
|
)
|
|
|
|
|
2022-05-12 12:47:13 +02:00
|
|
|
// IsKnownVersion returns true if the given version is one of the known
|
|
|
|
// derivation scheme versions as defined by this package.
|
|
|
|
func IsKnownVersion(internalVersion uint8) bool {
|
|
|
|
return internalVersion == KeyDerivationVersionLegacy ||
|
|
|
|
internalVersion == KeyDerivationVersionTaproot
|
|
|
|
}
|
|
|
|
|
2018-12-10 03:30:59 +01:00
|
|
|
var (
|
|
|
|
// MaxKeyRangeScan is the maximum number of keys that we'll attempt to
|
|
|
|
// scan with if a caller knows the public key, but not the KeyLocator
|
|
|
|
// and wishes to derive a private key.
|
|
|
|
MaxKeyRangeScan = 100000
|
|
|
|
|
|
|
|
// ErrCannotDerivePrivKey is returned when DerivePrivKey is unable to
|
|
|
|
// derive a private key given only the public key and target key
|
|
|
|
// family.
|
|
|
|
ErrCannotDerivePrivKey = fmt.Errorf("unable to derive private key")
|
|
|
|
)
|
|
|
|
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
// KeyFamily represents a "family" of keys that will be used within various
|
|
|
|
// contracts created by lnd. These families are meant to be distinct branches
|
|
|
|
// within the HD key chain of the backing wallet. Usage of key families within
|
|
|
|
// the interface below are strict in order to promote integrability and the
|
|
|
|
// ability to restore all keys given a user master seed backup.
|
|
|
|
//
|
|
|
|
// The key derivation in this file follows the following hierarchy based on
|
|
|
|
// BIP43:
|
|
|
|
//
|
2019-07-22 21:52:36 +02:00
|
|
|
// * m/1017'/coinType'/keyFamily'/0/index
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
type KeyFamily uint32
|
|
|
|
|
|
|
|
const (
|
|
|
|
// KeyFamilyMultiSig are keys to be used within multi-sig scripts.
|
|
|
|
KeyFamilyMultiSig KeyFamily = 0
|
|
|
|
|
|
|
|
// KeyFamilyRevocationBase are keys that are used within channels to
|
|
|
|
// create revocation basepoints that the remote party will use to
|
|
|
|
// create revocation keys for us.
|
2020-07-04 03:39:22 +02:00
|
|
|
KeyFamilyRevocationBase KeyFamily = 1
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
|
|
|
|
// KeyFamilyHtlcBase are keys used within channels that will be
|
|
|
|
// combined with per-state randomness to produce public keys that will
|
|
|
|
// be used in HTLC scripts.
|
|
|
|
KeyFamilyHtlcBase KeyFamily = 2
|
|
|
|
|
|
|
|
// KeyFamilyPaymentBase are keys used within channels that will be
|
|
|
|
// combined with per-state randomness to produce public keys that will
|
|
|
|
// be used in scripts that pay directly to us without any delay.
|
|
|
|
KeyFamilyPaymentBase KeyFamily = 3
|
|
|
|
|
|
|
|
// KeyFamilyDelayBase are keys used within channels that will be
|
|
|
|
// combined with per-state randomness to produce public keys that will
|
|
|
|
// be used in scripts that pay to us, but require a CSV delay before we
|
|
|
|
// can sweep the funds.
|
|
|
|
KeyFamilyDelayBase KeyFamily = 4
|
|
|
|
|
|
|
|
// KeyFamilyRevocationRoot is a family of keys which will be used to
|
|
|
|
// derive the root of a revocation tree for a particular channel.
|
|
|
|
KeyFamilyRevocationRoot KeyFamily = 5
|
|
|
|
|
|
|
|
// KeyFamilyNodeKey is a family of keys that will be used to derive
|
|
|
|
// keys that will be advertised on the network to represent our current
|
|
|
|
// "identity" within the network. Peers will need our latest node key
|
|
|
|
// in order to establish a transport session with us on the Lightning
|
|
|
|
// p2p level (BOLT-0008).
|
|
|
|
KeyFamilyNodeKey KeyFamily = 6
|
2018-12-10 04:38:45 +01:00
|
|
|
|
|
|
|
// KeyFamilyStaticBackup is the family of keys that will be used to
|
|
|
|
// derive keys that we use to encrypt and decrypt our set of static
|
|
|
|
// backups. These backups may either be stored within watch towers for
|
|
|
|
// a payment, or self stored on disk in a single file containing all
|
|
|
|
// the static channel backups.
|
|
|
|
KeyFamilyStaticBackup KeyFamily = 7
|
2019-04-24 05:03:48 +02:00
|
|
|
|
|
|
|
// KeyFamilyTowerSession is the family of keys that will be used to
|
|
|
|
// derive session keys when negotiating sessions with watchtowers. The
|
|
|
|
// session keys are limited to the lifetime of the session and are used
|
|
|
|
// to increase privacy in the watchtower protocol.
|
|
|
|
KeyFamilyTowerSession KeyFamily = 8
|
2019-06-21 01:55:32 +02:00
|
|
|
|
|
|
|
// KeyFamilyTowerID is the family of keys used to derive the public key
|
|
|
|
// of a watchtower. This made distinct from the node key to offer a form
|
|
|
|
// of rudimentary whitelisting, i.e. via knowledge of the pubkey,
|
|
|
|
// preventing others from having full access to the tower just as a
|
|
|
|
// result of knowing the node key.
|
|
|
|
KeyFamilyTowerID KeyFamily = 9
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
)
|
|
|
|
|
2021-10-14 15:42:56 +02:00
|
|
|
// VersionZeroKeyFamilies is a slice of all the known key families for first
|
|
|
|
// version of the key derivation schema defined in this package.
|
|
|
|
var VersionZeroKeyFamilies = []KeyFamily{
|
|
|
|
KeyFamilyMultiSig,
|
|
|
|
KeyFamilyRevocationBase,
|
|
|
|
KeyFamilyHtlcBase,
|
|
|
|
KeyFamilyPaymentBase,
|
|
|
|
KeyFamilyDelayBase,
|
|
|
|
KeyFamilyRevocationRoot,
|
|
|
|
KeyFamilyNodeKey,
|
|
|
|
KeyFamilyStaticBackup,
|
|
|
|
KeyFamilyTowerSession,
|
|
|
|
KeyFamilyTowerID,
|
|
|
|
}
|
|
|
|
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
// KeyLocator is a two-tuple that can be used to derive *any* key that has ever
|
|
|
|
// been used under the key derivation mechanisms described in this file.
|
|
|
|
// Version 0 of our key derivation schema uses the following BIP43-like
|
|
|
|
// derivation:
|
|
|
|
//
|
2019-07-22 21:52:36 +02:00
|
|
|
// * m/1017'/coinType'/keyFamily'/0/index
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
//
|
2019-07-22 21:52:36 +02:00
|
|
|
// Our purpose is 1017 (chosen arbitrary for now), and the coin type will vary
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
// based on which coin/chain the channels are being created on. The key family
|
|
|
|
// are actually just individual "accounts" in the nomenclature of BIP43. By
|
|
|
|
// default we assume a branch of 0 (external). Finally, the key index (which
|
|
|
|
// will vary per channel and use case) is the final element which allows us to
|
|
|
|
// deterministically derive keys.
|
|
|
|
type KeyLocator struct {
|
2018-08-14 04:19:58 +02:00
|
|
|
// TODO(roasbeef): add the key scope as well??
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
|
|
|
|
// Family is the family of key being identified.
|
|
|
|
Family KeyFamily
|
|
|
|
|
|
|
|
// Index is the precise index of the key being identified.
|
|
|
|
Index uint32
|
|
|
|
}
|
|
|
|
|
2018-08-14 04:19:58 +02:00
|
|
|
// IsEmpty returns true if a KeyLocator is "empty". This may be the case where
|
|
|
|
// we learn of a key from a remote party for a contract, but don't know the
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
// precise details of its derivation (as we don't know the private key!).
|
|
|
|
func (k KeyLocator) IsEmpty() bool {
|
|
|
|
return k.Family == 0 && k.Index == 0
|
|
|
|
}
|
|
|
|
|
|
|
|
// KeyDescriptor wraps a KeyLocator and also optionally includes a public key.
|
|
|
|
// Either the KeyLocator must be non-empty, or the public key pointer be
|
|
|
|
// non-nil. This will be used by the KeyRing interface to lookup arbitrary
|
|
|
|
// private keys, and also within the SignDescriptor struct to locate precisely
|
|
|
|
// which keys should be used for signing.
|
|
|
|
type KeyDescriptor struct {
|
|
|
|
// KeyLocator is the internal KeyLocator of the descriptor.
|
|
|
|
KeyLocator
|
|
|
|
|
|
|
|
// PubKey is an optional public key that fully describes a target key.
|
|
|
|
// If this is nil, the KeyLocator MUST NOT be empty.
|
|
|
|
PubKey *btcec.PublicKey
|
|
|
|
}
|
|
|
|
|
|
|
|
// KeyRing is the primary interface that will be used to perform public
|
|
|
|
// derivation of various keys used within the peer-to-peer network, and also
|
|
|
|
// within any created contracts. All derivation required by the KeyRing is
|
|
|
|
// based off of public derivation, so a system with only an extended public key
|
|
|
|
// (for the particular purpose+family) can derive this set of keys.
|
|
|
|
type KeyRing interface {
|
|
|
|
// DeriveNextKey attempts to derive the *next* key within the key
|
|
|
|
// family (account in BIP43) specified. This method should return the
|
|
|
|
// next external child within this branch.
|
|
|
|
DeriveNextKey(keyFam KeyFamily) (KeyDescriptor, error)
|
|
|
|
|
|
|
|
// DeriveKey attempts to derive an arbitrary key specified by the
|
|
|
|
// passed KeyLocator. This may be used in several recovery scenarios,
|
|
|
|
// or when manually rotating something like our current default node
|
|
|
|
// key.
|
|
|
|
DeriveKey(keyLoc KeyLocator) (KeyDescriptor, error)
|
|
|
|
}
|
|
|
|
|
2020-04-28 10:06:17 +02:00
|
|
|
// SecretKeyRing is a ring similar to the regular KeyRing interface, but it is
|
|
|
|
// also able to derive *private keys*. As this is a super-set of the regular
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
// KeyRing, we also expect the SecretKeyRing to implement the fully KeyRing
|
|
|
|
// interface. The methods in this struct may be used to extract the node key in
|
|
|
|
// order to accept inbound network connections, or to do manual signing for
|
|
|
|
// recovery purposes.
|
|
|
|
type SecretKeyRing interface {
|
|
|
|
KeyRing
|
|
|
|
|
2020-04-28 10:06:19 +02:00
|
|
|
ECDHRing
|
|
|
|
|
2021-09-23 16:54:27 +02:00
|
|
|
MessageSignerRing
|
2020-04-28 10:06:19 +02:00
|
|
|
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
// DerivePrivKey attempts to derive the private key that corresponds to
|
2018-12-10 03:30:59 +01:00
|
|
|
// the passed key descriptor. If the public key is set, then this
|
|
|
|
// method will perform an in-order scan over the key set, with a max of
|
|
|
|
// MaxKeyRangeScan keys. In order for this to work, the caller MUST set
|
|
|
|
// the KeyFamily within the partially populated KeyLocator.
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
DerivePrivKey(keyDesc KeyDescriptor) (*btcec.PrivateKey, error)
|
|
|
|
}
|
|
|
|
|
2021-09-23 16:54:27 +02:00
|
|
|
// MessageSignerRing is an interface that abstracts away basic low-level ECDSA
|
2020-04-28 10:06:17 +02:00
|
|
|
// signing on keys within a key ring.
|
2021-09-23 16:54:27 +02:00
|
|
|
type MessageSignerRing interface {
|
2021-09-23 16:54:26 +02:00
|
|
|
// SignMessage signs the given message, single or double SHA256 hashing
|
2021-09-23 16:54:29 +02:00
|
|
|
// it first, with the private key described in the key locator.
|
|
|
|
SignMessage(keyLoc KeyLocator, msg []byte,
|
2022-02-23 14:48:00 +01:00
|
|
|
doubleHash bool) (*ecdsa.Signature, error)
|
2020-04-28 10:06:17 +02:00
|
|
|
|
2021-09-23 16:54:27 +02:00
|
|
|
// SignMessageCompact signs the given message, single or double SHA256
|
2021-09-23 16:54:29 +02:00
|
|
|
// hashing it first, with the private key described in the key locator
|
|
|
|
// and returns the signature in the compact, public key recoverable
|
|
|
|
// format.
|
|
|
|
SignMessageCompact(keyLoc KeyLocator, msg []byte,
|
2021-09-23 16:54:27 +02:00
|
|
|
doubleHash bool) ([]byte, error)
|
2020-04-28 10:06:17 +02:00
|
|
|
}
|
|
|
|
|
2021-09-23 16:54:27 +02:00
|
|
|
// SingleKeyMessageSigner is an abstraction interface that hides the
|
2020-04-28 10:06:17 +02:00
|
|
|
// implementation of the low-level ECDSA signing operations by wrapping a
|
|
|
|
// single, specific private key.
|
2021-09-23 16:54:27 +02:00
|
|
|
type SingleKeyMessageSigner interface {
|
2020-04-28 10:06:17 +02:00
|
|
|
// PubKey returns the public key of the wrapped private key.
|
|
|
|
PubKey() *btcec.PublicKey
|
|
|
|
|
2021-09-23 16:54:30 +02:00
|
|
|
// KeyLocator returns the locator that describes the wrapped private
|
|
|
|
// key.
|
|
|
|
KeyLocator() KeyLocator
|
|
|
|
|
2021-09-23 16:54:26 +02:00
|
|
|
// SignMessage signs the given message, single or double SHA256 hashing
|
|
|
|
// it first, with the wrapped private key.
|
2022-02-23 14:48:00 +01:00
|
|
|
SignMessage(message []byte, doubleHash bool) (*ecdsa.Signature, error)
|
2020-04-28 10:06:17 +02:00
|
|
|
|
2021-09-23 16:54:27 +02:00
|
|
|
// SignMessageCompact signs the given message, single or double SHA256
|
|
|
|
// hashing it first, with the wrapped private key and returns the
|
|
|
|
// signature in the compact, public key recoverable format.
|
|
|
|
SignMessageCompact(message []byte, doubleHash bool) ([]byte, error)
|
2020-04-28 10:06:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// ECDHRing is an interface that abstracts away basic low-level ECDH shared key
|
|
|
|
// generation on keys within a key ring.
|
|
|
|
type ECDHRing interface {
|
|
|
|
// ECDH performs a scalar multiplication (ECDH-like operation) between
|
|
|
|
// the target key descriptor and remote public key. The output
|
|
|
|
// returned will be the sha256 of the resulting shared point serialized
|
|
|
|
// in compressed format. If k is our private key, and P is the public
|
|
|
|
// key, we perform the following operation:
|
|
|
|
//
|
|
|
|
// sx := k*P
|
|
|
|
// s := sha256(sx.SerializeCompressed())
|
|
|
|
ECDH(keyDesc KeyDescriptor, pubKey *btcec.PublicKey) ([32]byte, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SingleKeyECDH is an abstraction interface that hides the implementation of an
|
|
|
|
// ECDH operation by wrapping a single, specific private key.
|
|
|
|
type SingleKeyECDH interface {
|
|
|
|
// PubKey returns the public key of the wrapped private key.
|
|
|
|
PubKey() *btcec.PublicKey
|
|
|
|
|
|
|
|
// ECDH performs a scalar multiplication (ECDH-like operation) between
|
|
|
|
// the wrapped private key and remote public key. The output returned
|
|
|
|
// will be the sha256 of the resulting shared point serialized in
|
|
|
|
// compressed format.
|
|
|
|
ECDH(pubKey *btcec.PublicKey) ([32]byte, error)
|
|
|
|
}
|
|
|
|
|
keychain: introduce new package for deterministic key derivation + seed
In this commit, we add a new package to lnd: the keychain package. The
role of this package is to house all the interfaces that lnd will use
to generate the various keys it needs to create contracts and operate
within the network. Additionally, we also use this package to define a
deterministic key derivation scheme, that can be implemented by any
software/hardware that partially understands BIP43-like derivation.
The first version (v0) of the keychain schema is very simple. We re-use
BIP43, with a slight twist. Re-using BIP43 lets us leverage all the
existing libraries out there as far as compatibility. The purpose used
is 1017, and the coin type, the particular coin type of the chain where
the keys are meant to be used. Within our schema, we define multiple
“key families”. Each key family is actually just manifested as a new
“account” within the BIP44/43 family. With this schema, if we have
static description of the keys used within a channel, given the seed,
and the “KeyLocator”, we can re-derive all keys necessary be able to
re-sign for the channel.
2018-02-17 23:55:05 +01:00
|
|
|
// TODO(roasbeef): extend to actually support scalar mult of key?
|
|
|
|
// * would allow to push in initial handshake auth into interface as well
|