mirror of
https://github.com/libp2p/go-libp2p-peerstore.git
synced 2025-01-19 03:30:08 +08:00
135 lines
3.6 KiB
Go
135 lines
3.6 KiB
Go
package pstoreds
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
|
|
base32 "github.com/multiformats/go-base32"
|
|
|
|
ds "github.com/ipfs/go-datastore"
|
|
query "github.com/ipfs/go-datastore/query"
|
|
|
|
ic "github.com/libp2p/go-libp2p-crypto"
|
|
peer "github.com/libp2p/go-libp2p-peer"
|
|
pstore "github.com/libp2p/go-libp2p-peerstore"
|
|
)
|
|
|
|
// Public and private keys are stored under the following db key pattern:
|
|
// /peers/keys/<b32 peer id no padding>/{pub, priv}
|
|
var (
|
|
kbBase = ds.NewKey("/peers/keys")
|
|
pubSuffix = ds.NewKey("/pub")
|
|
privSuffix = ds.NewKey("/priv")
|
|
)
|
|
|
|
type dsKeyBook struct {
|
|
ds ds.Datastore
|
|
}
|
|
|
|
var _ pstore.KeyBook = (*dsKeyBook)(nil)
|
|
|
|
func NewKeyBook(_ context.Context, store ds.Datastore, _ Options) (pstore.KeyBook, error) {
|
|
return &dsKeyBook{store}, nil
|
|
}
|
|
|
|
func (kb *dsKeyBook) PubKey(p peer.ID) ic.PubKey {
|
|
key := kbBase.ChildString(base32.RawStdEncoding.EncodeToString([]byte(p))).Child(pubSuffix)
|
|
|
|
var pk ic.PubKey
|
|
if value, err := kb.ds.Get(key); err == nil {
|
|
pk, err = ic.UnmarshalPublicKey(value)
|
|
if err != nil {
|
|
log.Errorf("error when unmarshalling pubkey from datastore for peer %s: %s\n", p.Pretty(), err)
|
|
}
|
|
} else if err == ds.ErrNotFound {
|
|
pk, err = p.ExtractPublicKey()
|
|
switch err {
|
|
case nil:
|
|
case peer.ErrNoPublicKey:
|
|
return nil
|
|
default:
|
|
log.Errorf("error when extracting pubkey from peer ID for peer %s: %s\n", p.Pretty(), err)
|
|
return nil
|
|
}
|
|
pkb, err := pk.Bytes()
|
|
if err != nil {
|
|
log.Errorf("error when turning extracted pubkey into bytes for peer %s: %s\n", p.Pretty(), err)
|
|
return nil
|
|
}
|
|
err = kb.ds.Put(key, pkb)
|
|
if err != nil {
|
|
log.Errorf("error when adding extracted pubkey to peerstore for peer %s: %s\n", p.Pretty(), err)
|
|
return nil
|
|
}
|
|
} else {
|
|
log.Errorf("error when fetching pubkey from datastore for peer %s: %s\n", p.Pretty(), err)
|
|
}
|
|
|
|
return pk
|
|
}
|
|
|
|
func (kb *dsKeyBook) AddPubKey(p peer.ID, pk ic.PubKey) error {
|
|
// check it's correct.
|
|
if !p.MatchesPublicKey(pk) {
|
|
return errors.New("peer ID does not match public key")
|
|
}
|
|
|
|
key := kbBase.ChildString(base32.RawStdEncoding.EncodeToString([]byte(p))).Child(pubSuffix)
|
|
val, err := pk.Bytes()
|
|
if err != nil {
|
|
log.Errorf("error while converting pubkey byte string for peer %s: %s\n", p.Pretty(), err)
|
|
return err
|
|
}
|
|
err = kb.ds.Put(key, val)
|
|
if err != nil {
|
|
log.Errorf("error while updating pubkey in datastore for peer %s: %s\n", p.Pretty(), err)
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (kb *dsKeyBook) PrivKey(p peer.ID) ic.PrivKey {
|
|
key := kbBase.ChildString(base32.RawStdEncoding.EncodeToString([]byte(p))).Child(privSuffix)
|
|
value, err := kb.ds.Get(key)
|
|
if err != nil {
|
|
log.Errorf("error while fetching privkey from datastore for peer %s: %s\n", p.Pretty(), err)
|
|
return nil
|
|
}
|
|
sk, err := ic.UnmarshalPrivateKey(value)
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
return sk
|
|
}
|
|
|
|
func (kb *dsKeyBook) AddPrivKey(p peer.ID, sk ic.PrivKey) error {
|
|
if sk == nil {
|
|
return errors.New("private key is nil")
|
|
}
|
|
// check it's correct.
|
|
if !p.MatchesPrivateKey(sk) {
|
|
return errors.New("peer ID does not match private key")
|
|
}
|
|
|
|
key := kbBase.ChildString(base32.RawStdEncoding.EncodeToString([]byte(p))).Child(privSuffix)
|
|
val, err := sk.Bytes()
|
|
if err != nil {
|
|
log.Errorf("error while converting privkey byte string for peer %s: %s\n", p.Pretty(), err)
|
|
return err
|
|
}
|
|
err = kb.ds.Put(key, val)
|
|
if err != nil {
|
|
log.Errorf("error while updating privkey in datastore for peer %s: %s\n", p.Pretty(), err)
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (kb *dsKeyBook) PeersWithKeys() peer.IDSlice {
|
|
ids, err := uniquePeerIds(kb.ds, kbBase, func(result query.Result) string {
|
|
return ds.RawKey(result.Key).Parent().Name()
|
|
})
|
|
if err != nil {
|
|
log.Errorf("error while retrieving peers with keys: %v", err)
|
|
}
|
|
return ids
|
|
}
|