feat: metadata protocol

This commit is contained in:
Richard Ramos 2023-10-15 15:16:40 -04:00 committed by richΛrd
parent a16d00624e
commit 19ba25ffcb
13 changed files with 679 additions and 17 deletions

View File

@ -121,6 +121,13 @@ var (
Destination: &options.KeyPasswd, Destination: &options.KeyPasswd,
EnvVars: []string{"WAKUNODE2_KEY_PASSWORD"}, EnvVars: []string{"WAKUNODE2_KEY_PASSWORD"},
}) })
ClusterID = altsrc.NewUintFlag(&cli.UintFlag{
Name: "cluster-id",
Value: 0,
Usage: "Cluster id that the node is running in. Node in a different cluster id is disconnected.",
Destination: &options.ClusterID,
EnvVars: []string{"WAKUNODE2_CLUSTER_ID"},
})
StaticNode = cliutils.NewGenericFlagMultiValue(&cli.GenericFlag{ StaticNode = cliutils.NewGenericFlagMultiValue(&cli.GenericFlag{
Name: "staticnode", Name: "staticnode",
Usage: "Multiaddr of peer to directly connect with. Option may be repeated", Usage: "Multiaddr of peer to directly connect with. Option may be repeated",

View File

@ -36,6 +36,7 @@ func main() {
NodeKey, NodeKey,
KeyFile, KeyFile,
KeyPassword, KeyPassword,
ClusterID,
StaticNode, StaticNode,
KeepAlive, KeepAlive,
PersistPeers, PersistPeers,

View File

@ -141,6 +141,7 @@ func Execute(options NodeOptions) error {
node.WithMaxPeerConnections(options.MaxPeerConnections), node.WithMaxPeerConnections(options.MaxPeerConnections),
node.WithPrometheusRegisterer(prometheus.DefaultRegisterer), node.WithPrometheusRegisterer(prometheus.DefaultRegisterer),
node.WithPeerStoreCapacity(options.PeerStoreCapacity), node.WithPeerStoreCapacity(options.PeerStoreCapacity),
node.WithClusterID(uint16(options.ClusterID)),
} }
if len(options.AdvertiseAddresses) != 0 { if len(options.AdvertiseAddresses) != 0 {
nodeOpts = append(nodeOpts, node.WithAdvertiseAddresses(options.AdvertiseAddresses...)) nodeOpts = append(nodeOpts, node.WithAdvertiseAddresses(options.AdvertiseAddresses...))

View File

@ -148,6 +148,7 @@ type RendezvousOptions struct {
type NodeOptions struct { type NodeOptions struct {
Port int Port int
Address string Address string
ClusterID uint
DNS4DomainName string DNS4DomainName string
NodeKey *ecdsa.PrivateKey NodeKey *ecdsa.PrivateKey
KeyFile string KeyFile string

View File

@ -397,10 +397,8 @@ func (d *DiscoveryV5) peerLoop(ctx context.Context) error {
} }
if nodeRS == nil { if nodeRS == nil {
// TODO: Node has no shard registered. // Node has no shards registered.
// Since for now, status-go uses both mixed static and named shards, we assume the node is valid return false
// Once status-go uses only static shards, we can't return true anymore.
return true
} }
if nodeRS.Cluster != localRS.Cluster { if nodeRS.Cluster != localRS.Cluster {

View File

@ -37,6 +37,7 @@ import (
"github.com/waku-org/go-waku/waku/v2/protocol/filter" "github.com/waku-org/go-waku/waku/v2/protocol/filter"
"github.com/waku-org/go-waku/waku/v2/protocol/legacy_filter" "github.com/waku-org/go-waku/waku/v2/protocol/legacy_filter"
"github.com/waku-org/go-waku/waku/v2/protocol/lightpush" "github.com/waku-org/go-waku/waku/v2/protocol/lightpush"
"github.com/waku-org/go-waku/waku/v2/protocol/metadata"
"github.com/waku-org/go-waku/waku/v2/protocol/pb" "github.com/waku-org/go-waku/waku/v2/protocol/pb"
"github.com/waku-org/go-waku/waku/v2/protocol/peer_exchange" "github.com/waku-org/go-waku/waku/v2/protocol/peer_exchange"
"github.com/waku-org/go-waku/waku/v2/protocol/relay" "github.com/waku-org/go-waku/waku/v2/protocol/relay"
@ -94,6 +95,7 @@ type WakuNode struct {
discoveryV5 Service discoveryV5 Service
peerExchange Service peerExchange Service
rendezvous Service rendezvous Service
metadata Service
legacyFilter ReceptorService legacyFilter ReceptorService
filterFullNode ReceptorService filterFullNode ReceptorService
filterLightNode Service filterLightNode Service
@ -253,6 +255,8 @@ func New(opts ...WakuNodeOption) (*WakuNode, error) {
w.log.Error("creating localnode", zap.Error(err)) w.log.Error("creating localnode", zap.Error(err))
} }
w.metadata = metadata.NewWakuMetadata(w.opts.clusterID, w.localNode, w.log)
//Initialize peer manager. //Initialize peer manager.
w.peermanager = peermanager.NewPeerManager(w.opts.maxPeerConnections, w.opts.peerStoreCapacity, w.log) w.peermanager = peermanager.NewPeerManager(w.opts.maxPeerConnections, w.opts.peerStoreCapacity, w.log)
@ -388,6 +392,12 @@ func (w *WakuNode) Start(ctx context.Context) error {
go w.startKeepAlive(ctx, w.opts.keepAliveInterval) go w.startKeepAlive(ctx, w.opts.keepAliveInterval)
} }
w.metadata.SetHost(host)
err = w.metadata.Start(ctx)
if err != nil {
return err
}
w.peerConnector.SetHost(host) w.peerConnector.SetHost(host)
w.peermanager.SetHost(host) w.peermanager.SetHost(host)
err = w.peerConnector.Start(ctx) err = w.peerConnector.Start(ctx)
@ -508,6 +518,8 @@ func (w *WakuNode) Stop() {
defer w.identificationEventSub.Close() defer w.identificationEventSub.Close()
defer w.addressChangesSub.Close() defer w.addressChangesSub.Close()
w.host.Network().StopNotify(w.connectionNotif)
w.relay.Stop() w.relay.Stop()
w.lightPush.Stop() w.lightPush.Stop()
w.store.Stop() w.store.Stop()

View File

@ -45,6 +45,7 @@ const defaultMinRelayPeersToPublish = 0
type WakuNodeParameters struct { type WakuNodeParameters struct {
hostAddr *net.TCPAddr hostAddr *net.TCPAddr
clusterID uint16
dns4Domain string dns4Domain string
advertiseAddrs []multiaddr.Multiaddr advertiseAddrs []multiaddr.Multiaddr
multiAddr []multiaddr.Multiaddr multiAddr []multiaddr.Multiaddr
@ -294,6 +295,14 @@ func WithPrivateKey(privKey *ecdsa.PrivateKey) WakuNodeOption {
} }
} }
// WithClusterID is used to set the node's ClusterID
func WithClusterID(clusterID uint16) WakuNodeOption {
return func(params *WakuNodeParameters) error {
params.clusterID = clusterID
return nil
}
}
// WithNTP is used to use ntp for any operation that requires obtaining time // WithNTP is used to use ntp for any operation that requires obtaining time
// A list of ntp servers can be passed but if none is specified, some defaults // A list of ntp servers can be passed but if none is specified, some defaults
// will be used // will be used

View File

@ -16,10 +16,8 @@ import (
// the number of connections per IP address // the number of connections per IP address
type ConnectionGater struct { type ConnectionGater struct {
sync.Mutex sync.Mutex
logger *zap.Logger logger *zap.Logger
limiter map[string]int limiter map[string]int
inbound int
outbound int
} }
const maxConnsPerIP = 10 const maxConnsPerIP = 10
@ -27,10 +25,8 @@ const maxConnsPerIP = 10
// NewConnectionGater creates a new instance of ConnectionGater // NewConnectionGater creates a new instance of ConnectionGater
func NewConnectionGater(logger *zap.Logger) *ConnectionGater { func NewConnectionGater(logger *zap.Logger) *ConnectionGater {
c := &ConnectionGater{ c := &ConnectionGater{
logger: logger.Named("connection-gater"), logger: logger.Named("connection-gater"),
limiter: make(map[string]int), limiter: make(map[string]int),
inbound: 0,
outbound: 0,
} }
return c return c
@ -61,11 +57,6 @@ func (c *ConnectionGater) InterceptAccept(n network.ConnMultiaddrs) (allow bool)
return false return false
} }
if false { // inbound > someLimit
c.logger.Info("connection not accepted. Max inbound connections reached", zap.String("multiaddr", n.RemoteMultiaddr().String()))
return false
}
return true return true
} }

View File

@ -0,0 +1,3 @@
package pb
//go:generate protoc -I. --go_opt=paths=source_relative --go_opt=Mwaku_metadata.proto=github.com/waku-org/go-waku/waku/v2/protocol/metadata/pb --go_out=. ./waku_metadata.proto

View File

@ -0,0 +1,229 @@
// Code generated by protoc-gen-go. DO NOT EDIT.
// versions:
// protoc-gen-go v1.31.0
// protoc v3.21.12
// source: waku_metadata.proto
package pb
import (
protoreflect "google.golang.org/protobuf/reflect/protoreflect"
protoimpl "google.golang.org/protobuf/runtime/protoimpl"
reflect "reflect"
sync "sync"
)
const (
// Verify that this generated code is sufficiently up-to-date.
_ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
// Verify that runtime/protoimpl is sufficiently up-to-date.
_ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
)
type WakuMetadataRequest struct {
state protoimpl.MessageState
sizeCache protoimpl.SizeCache
unknownFields protoimpl.UnknownFields
ClusterId *uint32 `protobuf:"varint,1,opt,name=cluster_id,json=clusterId,proto3,oneof" json:"cluster_id,omitempty"`
Shards []uint32 `protobuf:"varint,2,rep,packed,name=shards,proto3" json:"shards,omitempty"`
}
func (x *WakuMetadataRequest) Reset() {
*x = WakuMetadataRequest{}
if protoimpl.UnsafeEnabled {
mi := &file_waku_metadata_proto_msgTypes[0]
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
ms.StoreMessageInfo(mi)
}
}
func (x *WakuMetadataRequest) String() string {
return protoimpl.X.MessageStringOf(x)
}
func (*WakuMetadataRequest) ProtoMessage() {}
func (x *WakuMetadataRequest) ProtoReflect() protoreflect.Message {
mi := &file_waku_metadata_proto_msgTypes[0]
if protoimpl.UnsafeEnabled && x != nil {
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
if ms.LoadMessageInfo() == nil {
ms.StoreMessageInfo(mi)
}
return ms
}
return mi.MessageOf(x)
}
// Deprecated: Use WakuMetadataRequest.ProtoReflect.Descriptor instead.
func (*WakuMetadataRequest) Descriptor() ([]byte, []int) {
return file_waku_metadata_proto_rawDescGZIP(), []int{0}
}
func (x *WakuMetadataRequest) GetClusterId() uint32 {
if x != nil && x.ClusterId != nil {
return *x.ClusterId
}
return 0
}
func (x *WakuMetadataRequest) GetShards() []uint32 {
if x != nil {
return x.Shards
}
return nil
}
type WakuMetadataResponse struct {
state protoimpl.MessageState
sizeCache protoimpl.SizeCache
unknownFields protoimpl.UnknownFields
ClusterId *uint32 `protobuf:"varint,1,opt,name=cluster_id,json=clusterId,proto3,oneof" json:"cluster_id,omitempty"`
Shards []uint32 `protobuf:"varint,2,rep,packed,name=shards,proto3" json:"shards,omitempty"`
}
func (x *WakuMetadataResponse) Reset() {
*x = WakuMetadataResponse{}
if protoimpl.UnsafeEnabled {
mi := &file_waku_metadata_proto_msgTypes[1]
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
ms.StoreMessageInfo(mi)
}
}
func (x *WakuMetadataResponse) String() string {
return protoimpl.X.MessageStringOf(x)
}
func (*WakuMetadataResponse) ProtoMessage() {}
func (x *WakuMetadataResponse) ProtoReflect() protoreflect.Message {
mi := &file_waku_metadata_proto_msgTypes[1]
if protoimpl.UnsafeEnabled && x != nil {
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
if ms.LoadMessageInfo() == nil {
ms.StoreMessageInfo(mi)
}
return ms
}
return mi.MessageOf(x)
}
// Deprecated: Use WakuMetadataResponse.ProtoReflect.Descriptor instead.
func (*WakuMetadataResponse) Descriptor() ([]byte, []int) {
return file_waku_metadata_proto_rawDescGZIP(), []int{1}
}
func (x *WakuMetadataResponse) GetClusterId() uint32 {
if x != nil && x.ClusterId != nil {
return *x.ClusterId
}
return 0
}
func (x *WakuMetadataResponse) GetShards() []uint32 {
if x != nil {
return x.Shards
}
return nil
}
var File_waku_metadata_proto protoreflect.FileDescriptor
var file_waku_metadata_proto_rawDesc = []byte{
0x0a, 0x13, 0x77, 0x61, 0x6b, 0x75, 0x5f, 0x6d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x2e,
0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x02, 0x70, 0x62, 0x22, 0x60, 0x0a, 0x13, 0x57, 0x61, 0x6b,
0x75, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74,
0x12, 0x22, 0x0a, 0x0a, 0x63, 0x6c, 0x75, 0x73, 0x74, 0x65, 0x72, 0x5f, 0x69, 0x64, 0x18, 0x01,
0x20, 0x01, 0x28, 0x0d, 0x48, 0x00, 0x52, 0x09, 0x63, 0x6c, 0x75, 0x73, 0x74, 0x65, 0x72, 0x49,
0x64, 0x88, 0x01, 0x01, 0x12, 0x16, 0x0a, 0x06, 0x73, 0x68, 0x61, 0x72, 0x64, 0x73, 0x18, 0x02,
0x20, 0x03, 0x28, 0x0d, 0x52, 0x06, 0x73, 0x68, 0x61, 0x72, 0x64, 0x73, 0x42, 0x0d, 0x0a, 0x0b,
0x5f, 0x63, 0x6c, 0x75, 0x73, 0x74, 0x65, 0x72, 0x5f, 0x69, 0x64, 0x22, 0x61, 0x0a, 0x14, 0x57,
0x61, 0x6b, 0x75, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x52, 0x65, 0x73, 0x70, 0x6f,
0x6e, 0x73, 0x65, 0x12, 0x22, 0x0a, 0x0a, 0x63, 0x6c, 0x75, 0x73, 0x74, 0x65, 0x72, 0x5f, 0x69,
0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0d, 0x48, 0x00, 0x52, 0x09, 0x63, 0x6c, 0x75, 0x73, 0x74,
0x65, 0x72, 0x49, 0x64, 0x88, 0x01, 0x01, 0x12, 0x16, 0x0a, 0x06, 0x73, 0x68, 0x61, 0x72, 0x64,
0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x0d, 0x52, 0x06, 0x73, 0x68, 0x61, 0x72, 0x64, 0x73, 0x42,
0x0d, 0x0a, 0x0b, 0x5f, 0x63, 0x6c, 0x75, 0x73, 0x74, 0x65, 0x72, 0x5f, 0x69, 0x64, 0x62, 0x06,
0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33,
}
var (
file_waku_metadata_proto_rawDescOnce sync.Once
file_waku_metadata_proto_rawDescData = file_waku_metadata_proto_rawDesc
)
func file_waku_metadata_proto_rawDescGZIP() []byte {
file_waku_metadata_proto_rawDescOnce.Do(func() {
file_waku_metadata_proto_rawDescData = protoimpl.X.CompressGZIP(file_waku_metadata_proto_rawDescData)
})
return file_waku_metadata_proto_rawDescData
}
var file_waku_metadata_proto_msgTypes = make([]protoimpl.MessageInfo, 2)
var file_waku_metadata_proto_goTypes = []interface{}{
(*WakuMetadataRequest)(nil), // 0: pb.WakuMetadataRequest
(*WakuMetadataResponse)(nil), // 1: pb.WakuMetadataResponse
}
var file_waku_metadata_proto_depIdxs = []int32{
0, // [0:0] is the sub-list for method output_type
0, // [0:0] is the sub-list for method input_type
0, // [0:0] is the sub-list for extension type_name
0, // [0:0] is the sub-list for extension extendee
0, // [0:0] is the sub-list for field type_name
}
func init() { file_waku_metadata_proto_init() }
func file_waku_metadata_proto_init() {
if File_waku_metadata_proto != nil {
return
}
if !protoimpl.UnsafeEnabled {
file_waku_metadata_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} {
switch v := v.(*WakuMetadataRequest); i {
case 0:
return &v.state
case 1:
return &v.sizeCache
case 2:
return &v.unknownFields
default:
return nil
}
}
file_waku_metadata_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} {
switch v := v.(*WakuMetadataResponse); i {
case 0:
return &v.state
case 1:
return &v.sizeCache
case 2:
return &v.unknownFields
default:
return nil
}
}
}
file_waku_metadata_proto_msgTypes[0].OneofWrappers = []interface{}{}
file_waku_metadata_proto_msgTypes[1].OneofWrappers = []interface{}{}
type x struct{}
out := protoimpl.TypeBuilder{
File: protoimpl.DescBuilder{
GoPackagePath: reflect.TypeOf(x{}).PkgPath(),
RawDescriptor: file_waku_metadata_proto_rawDesc,
NumEnums: 0,
NumMessages: 2,
NumExtensions: 0,
NumServices: 0,
},
GoTypes: file_waku_metadata_proto_goTypes,
DependencyIndexes: file_waku_metadata_proto_depIdxs,
MessageInfos: file_waku_metadata_proto_msgTypes,
}.Build()
File_waku_metadata_proto = out.File
file_waku_metadata_proto_rawDesc = nil
file_waku_metadata_proto_goTypes = nil
file_waku_metadata_proto_depIdxs = nil
}

View File

@ -0,0 +1,13 @@
syntax = "proto3";
package pb;
message WakuMetadataRequest {
optional uint32 cluster_id = 1;
repeated uint32 shards = 2;
}
message WakuMetadataResponse {
optional uint32 cluster_id = 1;
repeated uint32 shards = 2;
}

View File

@ -0,0 +1,243 @@
package metadata
import (
"context"
"errors"
"math"
"strings"
"github.com/ethereum/go-ethereum/p2p/enode"
"github.com/libp2p/go-libp2p/core/host"
"github.com/libp2p/go-libp2p/core/network"
"github.com/libp2p/go-libp2p/core/peer"
libp2pProtocol "github.com/libp2p/go-libp2p/core/protocol"
"github.com/libp2p/go-msgio/pbio"
"github.com/multiformats/go-multiaddr"
"github.com/waku-org/go-waku/logging"
"github.com/waku-org/go-waku/waku/v2/protocol"
"github.com/waku-org/go-waku/waku/v2/protocol/enr"
"github.com/waku-org/go-waku/waku/v2/protocol/metadata/pb"
"go.uber.org/zap"
)
// MetadataID_v1 is the current Waku Metadata protocol identifier
const MetadataID_v1 = libp2pProtocol.ID("/vac/waku/metadata/1.0.0")
// WakuMetadata is the implementation of the Waku Metadata protocol
type WakuMetadata struct {
network.Notifiee
h host.Host
ctx context.Context
cancel context.CancelFunc
clusterID uint16
localnode *enode.LocalNode
log *zap.Logger
}
// NewWakuMetadata returns a new instance of Waku Metadata struct
// Takes an optional peermanager if WakuLightPush is being created along with WakuNode.
// If using libp2p host, then pass peermanager as nil
func NewWakuMetadata(clusterID uint16, localnode *enode.LocalNode, log *zap.Logger) *WakuMetadata {
m := new(WakuMetadata)
m.log = log.Named("metadata")
m.clusterID = clusterID
m.localnode = localnode
return m
}
// Sets the host to be able to mount or consume a protocol
func (wakuM *WakuMetadata) SetHost(h host.Host) {
wakuM.h = h
}
// Start inits the metadata protocol
func (wakuM *WakuMetadata) Start(ctx context.Context) error {
if wakuM.clusterID == 0 {
wakuM.log.Warn("no clusterID is specified. Protocol will not be initialized")
}
ctx, cancel := context.WithCancel(ctx)
wakuM.ctx = ctx
wakuM.cancel = cancel
wakuM.h.Network().Notify(wakuM)
wakuM.h.SetStreamHandlerMatch(MetadataID_v1, protocol.PrefixTextMatch(string(MetadataID_v1)), wakuM.onRequest(ctx))
wakuM.log.Info("metadata protocol started")
return nil
}
func (wakuM *WakuMetadata) getClusterAndShards() (*uint32, []uint32, error) {
shard, err := enr.RelaySharding(wakuM.localnode.Node().Record())
if err != nil {
return nil, nil, err
}
var shards []uint32
if shard != nil && shard.Cluster == uint16(wakuM.clusterID) {
for _, idx := range shard.Indices {
shards = append(shards, uint32(idx))
}
}
u32ClusterID := uint32(wakuM.clusterID)
return &u32ClusterID, shards, nil
}
func (wakuM *WakuMetadata) Request(ctx context.Context, peerID peer.ID) (*protocol.RelayShards, error) {
logger := wakuM.log.With(logging.HostID("peer", peerID))
connOpt, err := wakuM.h.NewStream(ctx, peerID, MetadataID_v1)
if err != nil {
logger.Error("creating stream to peer", zap.Error(err))
return nil, err
}
defer connOpt.Close()
defer func() {
err := connOpt.Reset()
if err != nil {
logger.Error("resetting connection", zap.Error(err))
}
}()
clusterID, shards, err := wakuM.getClusterAndShards()
if err != nil {
return nil, err
}
request := &pb.WakuMetadataRequest{}
request.ClusterId = clusterID
request.Shards = shards
writer := pbio.NewDelimitedWriter(connOpt)
reader := pbio.NewDelimitedReader(connOpt, math.MaxInt32)
err = writer.WriteMsg(request)
if err != nil {
logger.Error("writing request", zap.Error(err))
return nil, err
}
response := &pb.WakuMetadataResponse{}
err = reader.ReadMsg(response)
if err != nil {
logger.Error("reading response", zap.Error(err))
return nil, err
}
if response.ClusterId == nil {
return nil, nil // Node is not using sharding
}
result := &protocol.RelayShards{}
result.Cluster = uint16(*response.ClusterId)
for _, i := range response.Shards {
result.Indices = append(result.Indices, uint16(i))
}
return result, nil
}
func (wakuM *WakuMetadata) onRequest(ctx context.Context) func(s network.Stream) {
return func(s network.Stream) {
defer s.Close()
logger := wakuM.log.With(logging.HostID("peer", s.Conn().RemotePeer()))
request := &pb.WakuMetadataRequest{}
writer := pbio.NewDelimitedWriter(s)
reader := pbio.NewDelimitedReader(s, math.MaxInt32)
err := reader.ReadMsg(request)
if err != nil {
logger.Error("reading request", zap.Error(err))
return
}
response := new(pb.WakuMetadataResponse)
clusterID, shards, err := wakuM.getClusterAndShards()
if err != nil {
logger.Error("obtaining shard info", zap.Error(err))
} else {
response.ClusterId = clusterID
response.Shards = shards
}
err = writer.WriteMsg(response)
if err != nil {
logger.Error("writing response", zap.Error(err))
_ = s.Reset()
}
}
}
// Stop unmounts the metadata protocol
func (wakuM *WakuMetadata) Stop() {
if wakuM.cancel == nil {
return
}
wakuM.h.Network().StopNotify(wakuM)
wakuM.cancel()
wakuM.h.RemoveStreamHandler(MetadataID_v1)
}
// Listen is called when network starts listening on an addr
func (wakuM *WakuMetadata) Listen(n network.Network, m multiaddr.Multiaddr) {
// Do nothing
}
// ListenClose is called when network stops listening on an address
func (wakuM *WakuMetadata) ListenClose(n network.Network, m multiaddr.Multiaddr) {
// Do nothing
}
// Connected is called when a connection is opened
func (wakuM *WakuMetadata) Connected(n network.Network, cc network.Conn) {
go func() {
// Metadata verification is done only if a clusterID is specified
if wakuM.clusterID == 0 {
return
}
peerID := cc.RemotePeer()
logger := wakuM.log.With(logging.HostID("peerID", peerID))
shouldDisconnect := true
shard, err := wakuM.Request(wakuM.ctx, peerID)
if err == nil {
if shard == nil {
err = errors.New("no shard reported")
} else if shard.Cluster != wakuM.clusterID {
err = errors.New("different clusterID reported")
}
} else {
// Only disconnect from peers if they support the protocol
// TODO: open a PR in go-libp2p to create a var with this error to not have to compare strings but use errors.Is instead
if strings.Contains(err.Error(), "protocols not supported") {
shouldDisconnect = false
}
}
if shouldDisconnect && err != nil {
logger.Error("disconnecting from peer", zap.Error(err))
wakuM.h.Peerstore().RemovePeer(peerID)
if err := wakuM.h.Network().ClosePeer(peerID); err != nil {
logger.Error("could not disconnect from peer", zap.Error(err))
}
}
}()
}
// Disconnected is called when a connection closed
func (wakuM *WakuMetadata) Disconnected(n network.Network, cc network.Conn) {
// Do nothing
}

View File

@ -0,0 +1,154 @@
package metadata
import (
"context"
"crypto/rand"
"testing"
"time"
gcrypto "github.com/ethereum/go-ethereum/crypto"
"github.com/libp2p/go-libp2p/core/peer"
"github.com/libp2p/go-libp2p/core/peerstore"
"github.com/stretchr/testify/require"
"github.com/waku-org/go-waku/tests"
"github.com/waku-org/go-waku/waku/v2/protocol"
"github.com/waku-org/go-waku/waku/v2/protocol/enr"
"github.com/waku-org/go-waku/waku/v2/utils"
)
func createWakuMetadata(t *testing.T, rs *protocol.RelayShards) *WakuMetadata {
port, err := tests.FindFreePort(t, "", 5)
require.NoError(t, err)
host, err := tests.MakeHost(context.Background(), port, rand.Reader)
require.NoError(t, err)
key, _ := gcrypto.GenerateKey()
localNode, err := enr.NewLocalnode(key)
require.NoError(t, err)
cluster := uint16(0)
if rs != nil {
err = enr.WithWakuRelaySharding(*rs)(localNode)
require.NoError(t, err)
cluster = rs.Cluster
}
m1 := NewWakuMetadata(cluster, localNode, utils.Logger())
m1.SetHost(host)
err = m1.Start(context.TODO())
require.NoError(t, err)
return m1
}
func TestWakuMetadataRequest(t *testing.T) {
testShard16 := uint16(16)
rs16_1, err := protocol.NewRelayShards(testShard16, 1)
require.NoError(t, err)
rs16_2, err := protocol.NewRelayShards(testShard16, 2)
require.NoError(t, err)
m16_1 := createWakuMetadata(t, &rs16_1)
m16_2 := createWakuMetadata(t, &rs16_2)
m_noRS := createWakuMetadata(t, nil)
m16_1.h.Peerstore().AddAddrs(m16_2.h.ID(), m16_2.h.Network().ListenAddresses(), peerstore.PermanentAddrTTL)
m16_1.h.Peerstore().AddAddrs(m_noRS.h.ID(), m_noRS.h.Network().ListenAddresses(), peerstore.PermanentAddrTTL)
// Query a peer that is subscribed to a shard
result, err := m16_1.Request(context.Background(), m16_2.h.ID())
require.NoError(t, err)
require.Equal(t, testShard16, result.Cluster)
require.Equal(t, rs16_2.Indices, result.Indices)
// Updating the peer shards
rs16_2.Indices = append(rs16_2.Indices, 3, 4)
err = enr.WithWakuRelaySharding(rs16_2)(m16_2.localnode)
require.NoError(t, err)
// Query same peer, after that peer subscribes to more shards
result, err = m16_1.Request(context.Background(), m16_2.h.ID())
require.NoError(t, err)
require.Equal(t, testShard16, result.Cluster)
require.ElementsMatch(t, rs16_2.Indices, result.Indices)
// Query a peer not subscribed to a shard
result, err = m16_1.Request(context.Background(), m_noRS.h.ID())
require.NoError(t, err)
require.Equal(t, uint16(0), result.Cluster)
require.Len(t, result.Indices, 0)
}
func TestNoNetwork(t *testing.T) {
cluster1 := uint16(1)
rs1, err := protocol.NewRelayShards(cluster1, 1)
require.NoError(t, err)
m1 := createWakuMetadata(t, &rs1)
// host2 does not support metadata protocol
port, err := tests.FindFreePort(t, "", 5)
require.NoError(t, err)
host2, err := tests.MakeHost(context.Background(), port, rand.Reader)
require.NoError(t, err)
m1.h.Peerstore().AddAddrs(host2.ID(), host2.Network().ListenAddresses(), peerstore.PermanentAddrTTL)
_, err = m1.h.Network().DialPeer(context.TODO(), host2.ID())
require.NoError(t, err)
time.Sleep(2 * time.Second)
// Verifying peer connections
require.Len(t, m1.h.Network().Peers(), 1)
require.Len(t, host2.Network().Peers(), 1)
}
// go test -timeout 300s -run TestDropConnectionOnDiffNetworks github.com/waku-org/go-waku/waku/v2/protocol/metadata -count 1 -v
func TestDropConnectionOnDiffNetworks(t *testing.T) {
cluster1 := uint16(1)
cluster2 := uint16(2)
// Initializing metadata and peer managers
rs1, err := protocol.NewRelayShards(cluster1, 1)
require.NoError(t, err)
m1 := createWakuMetadata(t, &rs1)
rs2, err := protocol.NewRelayShards(cluster2, 1)
require.NoError(t, err)
m2 := createWakuMetadata(t, &rs2)
rs3, err := protocol.NewRelayShards(cluster2, 1)
require.NoError(t, err)
m3 := createWakuMetadata(t, &rs3)
// Creating connection between peers
// 1->2 (fails)
m1.h.Peerstore().AddAddrs(m2.h.ID(), m2.h.Network().ListenAddresses(), peerstore.PermanentAddrTTL)
_, err = m1.h.Network().DialPeer(context.TODO(), m2.h.ID())
require.NoError(t, err)
// 1->3 (fails)
m1.h.Peerstore().AddAddrs(m3.h.ID(), m3.h.Network().ListenAddresses(), peerstore.PermanentAddrTTL)
_, err = m1.h.Network().DialPeer(context.TODO(), m3.h.ID())
require.NoError(t, err)
// 2->3 (succeeds)
m2.h.Peerstore().AddAddrs(m3.h.ID(), m3.h.Network().ListenAddresses(), peerstore.PermanentAddrTTL)
_, err = m2.h.Network().DialPeer(context.TODO(), m3.h.ID())
require.NoError(t, err)
time.Sleep(2 * time.Second)
// Verifying peer connections
require.Len(t, m1.h.Network().Peers(), 0)
require.Len(t, m2.h.Network().Peers(), 1)
require.Len(t, m3.h.Network().Peers(), 1)
require.Equal(t, []peer.ID{m3.h.ID()}, m2.h.Network().Peers())
require.Equal(t, []peer.ID{m2.h.ID()}, m3.h.Network().Peers())
}