2019-05-17 18:13:18 +08:00
|
|
|
package pstoreds
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"sync"
|
|
|
|
|
2019-05-31 21:51:16 +08:00
|
|
|
peer "github.com/libp2p/go-libp2p-core/peer"
|
2019-05-17 18:13:18 +08:00
|
|
|
|
2019-05-31 21:51:16 +08:00
|
|
|
pstore "github.com/libp2p/go-libp2p-core/peerstore"
|
2019-05-17 18:13:18 +08:00
|
|
|
)
|
|
|
|
|
2019-05-18 16:42:47 +08:00
|
|
|
type protoSegment struct {
|
|
|
|
sync.RWMutex
|
2019-05-17 18:13:18 +08:00
|
|
|
}
|
|
|
|
|
2019-05-18 16:42:47 +08:00
|
|
|
type protoSegments [256]*protoSegment
|
2019-05-17 18:13:18 +08:00
|
|
|
|
2019-05-18 16:42:47 +08:00
|
|
|
func (s *protoSegments) get(p peer.ID) *protoSegment {
|
|
|
|
return s[byte(p[len(p)-1])]
|
2019-05-17 18:13:18 +08:00
|
|
|
}
|
|
|
|
|
2019-05-18 16:42:47 +08:00
|
|
|
type dsProtoBook struct {
|
|
|
|
segments protoSegments
|
|
|
|
meta pstore.PeerMetadata
|
2019-05-17 18:13:18 +08:00
|
|
|
}
|
|
|
|
|
2019-05-18 16:42:47 +08:00
|
|
|
var _ pstore.ProtoBook = (*dsProtoBook)(nil)
|
2019-05-17 18:13:18 +08:00
|
|
|
|
2020-03-06 20:07:36 +08:00
|
|
|
func NewProtoBook(meta pstore.PeerMetadata) *dsProtoBook {
|
2019-05-18 16:42:47 +08:00
|
|
|
return &dsProtoBook{
|
|
|
|
meta: meta,
|
|
|
|
segments: func() (ret protoSegments) {
|
|
|
|
for i := range ret {
|
|
|
|
ret[i] = &protoSegment{}
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}(),
|
|
|
|
}
|
2019-05-17 18:13:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (pb *dsProtoBook) SetProtocols(p peer.ID, protos ...string) error {
|
2019-07-03 02:44:57 +08:00
|
|
|
if err := p.Validate(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-07-10 22:45:49 +08:00
|
|
|
s := pb.segments.get(p)
|
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
2019-05-17 18:13:18 +08:00
|
|
|
|
|
|
|
protomap := make(map[string]struct{}, len(protos))
|
|
|
|
for _, proto := range protos {
|
|
|
|
protomap[proto] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
return pb.meta.Put(p, "protocols", protomap)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pb *dsProtoBook) AddProtocols(p peer.ID, protos ...string) error {
|
2019-07-03 02:44:57 +08:00
|
|
|
if err := p.Validate(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-07-10 22:45:49 +08:00
|
|
|
s := pb.segments.get(p)
|
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
2019-05-17 18:13:18 +08:00
|
|
|
|
|
|
|
pmap, err := pb.getProtocolMap(p)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, proto := range protos {
|
|
|
|
pmap[proto] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
return pb.meta.Put(p, "protocols", pmap)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pb *dsProtoBook) GetProtocols(p peer.ID) ([]string, error) {
|
2019-07-03 02:44:57 +08:00
|
|
|
if err := p.Validate(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-07-10 22:45:49 +08:00
|
|
|
s := pb.segments.get(p)
|
|
|
|
s.RLock()
|
|
|
|
defer s.RUnlock()
|
2019-05-17 18:13:18 +08:00
|
|
|
|
|
|
|
pmap, err := pb.getProtocolMap(p)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
res := make([]string, 0, len(pmap))
|
|
|
|
for proto := range pmap {
|
|
|
|
res = append(res, proto)
|
|
|
|
}
|
|
|
|
|
|
|
|
return res, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pb *dsProtoBook) SupportsProtocols(p peer.ID, protos ...string) ([]string, error) {
|
2019-07-03 02:44:57 +08:00
|
|
|
if err := p.Validate(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-07-10 22:45:49 +08:00
|
|
|
s := pb.segments.get(p)
|
|
|
|
s.RLock()
|
|
|
|
defer s.RUnlock()
|
2019-05-17 18:13:18 +08:00
|
|
|
|
|
|
|
pmap, err := pb.getProtocolMap(p)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
res := make([]string, 0, len(protos))
|
|
|
|
for _, proto := range protos {
|
|
|
|
if _, ok := pmap[proto]; ok {
|
|
|
|
res = append(res, proto)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return res, nil
|
|
|
|
}
|
|
|
|
|
2019-06-20 06:53:07 +08:00
|
|
|
func (pb *dsProtoBook) RemoveProtocols(p peer.ID, protos ...string) error {
|
2019-07-03 02:44:57 +08:00
|
|
|
if err := p.Validate(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-06-20 06:53:07 +08:00
|
|
|
s := pb.segments.get(p)
|
2019-07-10 22:45:49 +08:00
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
2019-06-20 06:53:07 +08:00
|
|
|
|
|
|
|
pmap, err := pb.getProtocolMap(p)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(pmap) == 0 {
|
|
|
|
// nothing to do.
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, proto := range protos {
|
|
|
|
delete(pmap, proto)
|
|
|
|
}
|
|
|
|
|
|
|
|
return pb.meta.Put(p, "protocols", pmap)
|
|
|
|
}
|
|
|
|
|
2019-05-17 18:13:18 +08:00
|
|
|
func (pb *dsProtoBook) getProtocolMap(p peer.ID) (map[string]struct{}, error) {
|
|
|
|
iprotomap, err := pb.meta.Get(p, "protocols")
|
|
|
|
switch err {
|
|
|
|
default:
|
|
|
|
return nil, err
|
|
|
|
case pstore.ErrNotFound:
|
|
|
|
return make(map[string]struct{}), nil
|
|
|
|
case nil:
|
|
|
|
cast, ok := iprotomap.(map[string]struct{})
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("stored protocol set was not a map")
|
|
|
|
}
|
|
|
|
|
|
|
|
return cast, nil
|
|
|
|
}
|
|
|
|
}
|