2023-09-29 10:43:25 +05:30
|
|
|
package subscription
|
2023-02-08 19:33:06 -04:00
|
|
|
|
|
|
|
import (
|
2023-06-22 14:55:51 -04:00
|
|
|
"encoding/json"
|
2023-09-29 10:43:25 +05:30
|
|
|
"errors"
|
2023-02-08 19:33:06 -04:00
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/google/uuid"
|
|
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol"
|
2023-05-08 17:33:10 -04:00
|
|
|
"go.uber.org/zap"
|
2023-09-19 15:52:11 +03:00
|
|
|
"golang.org/x/exp/maps"
|
2023-02-08 19:33:06 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
type SubscriptionDetails struct {
|
|
|
|
sync.RWMutex
|
|
|
|
|
2023-05-08 17:33:10 -04:00
|
|
|
ID string
|
2023-02-08 19:33:06 -04:00
|
|
|
mapRef *SubscriptionsMap
|
2023-05-08 17:33:10 -04:00
|
|
|
Closed bool
|
2023-02-08 19:33:06 -04:00
|
|
|
once sync.Once
|
|
|
|
|
2023-05-08 17:33:10 -04:00
|
|
|
PeerID peer.ID
|
2023-09-29 10:43:25 +05:30
|
|
|
ContentFilter protocol.ContentFilter
|
2023-02-08 19:33:06 -04:00
|
|
|
C chan *protocol.Envelope
|
|
|
|
}
|
|
|
|
|
2023-09-19 15:52:11 +03:00
|
|
|
// Map of SubscriptionDetails.ID to subscriptions
|
2023-02-08 19:33:06 -04:00
|
|
|
type SubscriptionSet map[string]*SubscriptionDetails
|
|
|
|
|
|
|
|
type PeerSubscription struct {
|
2023-09-29 10:43:25 +05:30
|
|
|
PeerID peer.ID
|
|
|
|
SubsPerPubsubTopic map[string]SubscriptionSet
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
type SubscriptionsMap struct {
|
|
|
|
sync.RWMutex
|
2023-05-08 17:33:10 -04:00
|
|
|
logger *zap.Logger
|
2023-09-29 10:43:25 +05:30
|
|
|
Items map[peer.ID]*PeerSubscription
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
var ErrNotFound = errors.New("not found")
|
|
|
|
|
2023-05-08 17:33:10 -04:00
|
|
|
func NewSubscriptionMap(logger *zap.Logger) *SubscriptionsMap {
|
2023-02-08 19:33:06 -04:00
|
|
|
return &SubscriptionsMap{
|
2023-05-08 17:33:10 -04:00
|
|
|
logger: logger.Named("subscription-map"),
|
2023-09-29 10:43:25 +05:30
|
|
|
Items: make(map[peer.ID]*PeerSubscription),
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
func (sub *SubscriptionsMap) NewSubscription(peerID peer.ID, cf protocol.ContentFilter) *SubscriptionDetails {
|
2023-02-08 19:33:06 -04:00
|
|
|
sub.Lock()
|
|
|
|
defer sub.Unlock()
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
peerSubscription, ok := sub.Items[peerID]
|
2023-02-08 19:33:06 -04:00
|
|
|
if !ok {
|
|
|
|
peerSubscription = &PeerSubscription{
|
2023-09-29 10:43:25 +05:30
|
|
|
PeerID: peerID,
|
|
|
|
SubsPerPubsubTopic: make(map[string]SubscriptionSet),
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
2023-09-29 10:43:25 +05:30
|
|
|
sub.Items[peerID] = peerSubscription
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
_, ok = peerSubscription.SubsPerPubsubTopic[cf.PubsubTopic]
|
2023-02-08 19:33:06 -04:00
|
|
|
if !ok {
|
2023-09-29 10:43:25 +05:30
|
|
|
peerSubscription.SubsPerPubsubTopic[cf.PubsubTopic] = make(SubscriptionSet)
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
details := &SubscriptionDetails{
|
2023-05-08 17:33:10 -04:00
|
|
|
ID: uuid.NewString(),
|
2023-02-14 18:19:38 -04:00
|
|
|
mapRef: sub,
|
2023-05-08 17:33:10 -04:00
|
|
|
PeerID: peerID,
|
|
|
|
C: make(chan *protocol.Envelope, 1024),
|
2023-09-29 10:43:25 +05:30
|
|
|
ContentFilter: protocol.ContentFilter{PubsubTopic: cf.PubsubTopic, ContentTopics: maps.Clone(cf.ContentTopics)},
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
sub.Items[peerID].SubsPerPubsubTopic[cf.PubsubTopic][details.ID] = details
|
2023-02-08 19:33:06 -04:00
|
|
|
|
|
|
|
return details
|
|
|
|
}
|
|
|
|
|
2023-04-20 10:44:06 -04:00
|
|
|
func (sub *SubscriptionsMap) IsSubscribedTo(peerID peer.ID) bool {
|
|
|
|
sub.RLock()
|
|
|
|
defer sub.RUnlock()
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
_, ok := sub.Items[peerID]
|
2023-04-20 10:44:06 -04:00
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
2023-09-19 15:52:11 +03:00
|
|
|
// Check if we have subscriptions for all (pubsubTopic, contentTopics[i]) pairs provided
|
2023-09-29 10:43:25 +05:30
|
|
|
func (sub *SubscriptionsMap) Has(peerID peer.ID, cf protocol.ContentFilter) bool {
|
2023-04-20 10:44:06 -04:00
|
|
|
sub.RLock()
|
|
|
|
defer sub.RUnlock()
|
|
|
|
|
2023-03-08 11:58:51 -04:00
|
|
|
// Check if peer exits
|
2023-09-29 10:43:25 +05:30
|
|
|
peerSubscription, ok := sub.Items[peerID]
|
2023-03-08 11:58:51 -04:00
|
|
|
if !ok {
|
|
|
|
return false
|
|
|
|
}
|
2023-09-12 18:04:43 +05:30
|
|
|
//TODO: Handle pubsubTopic as null
|
2023-03-08 11:58:51 -04:00
|
|
|
// Check if pubsub topic exists
|
2023-09-29 10:43:25 +05:30
|
|
|
subscriptions, ok := peerSubscription.SubsPerPubsubTopic[cf.PubsubTopic]
|
2023-03-08 11:58:51 -04:00
|
|
|
if !ok {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if the content topic exists within the list of subscriptions for this peer
|
2023-09-21 13:36:04 +03:00
|
|
|
for _, ct := range cf.ContentTopicsList() {
|
2023-03-08 11:58:51 -04:00
|
|
|
found := false
|
|
|
|
for _, subscription := range subscriptions {
|
2023-09-19 15:52:11 +03:00
|
|
|
_, exists := subscription.ContentFilter.ContentTopics[ct]
|
2023-03-08 11:58:51 -04:00
|
|
|
if exists {
|
|
|
|
found = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
2023-02-08 19:33:06 -04:00
|
|
|
func (sub *SubscriptionsMap) Delete(subscription *SubscriptionDetails) error {
|
|
|
|
sub.Lock()
|
|
|
|
defer sub.Unlock()
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
peerSubscription, ok := sub.Items[subscription.PeerID]
|
2023-02-08 19:33:06 -04:00
|
|
|
if !ok {
|
|
|
|
return ErrNotFound
|
|
|
|
}
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
delete(peerSubscription.SubsPerPubsubTopic[subscription.ContentFilter.PubsubTopic], subscription.ID)
|
2023-02-08 19:33:06 -04:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-02-14 18:19:38 -04:00
|
|
|
func (s *SubscriptionDetails) Add(contentTopics ...string) {
|
2023-02-08 19:33:06 -04:00
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
|
|
|
|
|
|
|
for _, ct := range contentTopics {
|
2023-09-19 15:52:11 +03:00
|
|
|
s.ContentFilter.ContentTopics[ct] = struct{}{}
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-14 18:19:38 -04:00
|
|
|
func (s *SubscriptionDetails) Remove(contentTopics ...string) {
|
2023-02-08 19:33:06 -04:00
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
|
|
|
|
|
|
|
for _, ct := range contentTopics {
|
2023-09-19 15:52:11 +03:00
|
|
|
delete(s.ContentFilter.ContentTopics, ct)
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
func (s *SubscriptionDetails) CloseC() {
|
2023-02-08 19:33:06 -04:00
|
|
|
s.once.Do(func() {
|
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
|
|
|
|
2023-05-08 17:33:10 -04:00
|
|
|
s.Closed = true
|
2023-02-08 19:33:06 -04:00
|
|
|
close(s.C)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *SubscriptionDetails) Close() error {
|
2023-09-29 10:43:25 +05:30
|
|
|
s.CloseC()
|
2023-02-08 19:33:06 -04:00
|
|
|
return s.mapRef.Delete(s)
|
|
|
|
}
|
|
|
|
|
2023-02-15 14:43:51 -04:00
|
|
|
func (s *SubscriptionDetails) Clone() *SubscriptionDetails {
|
|
|
|
s.RLock()
|
|
|
|
defer s.RUnlock()
|
|
|
|
|
|
|
|
result := &SubscriptionDetails{
|
2023-05-08 17:33:10 -04:00
|
|
|
ID: uuid.NewString(),
|
2023-02-15 14:43:51 -04:00
|
|
|
mapRef: s.mapRef,
|
2023-05-08 17:33:10 -04:00
|
|
|
Closed: false,
|
|
|
|
PeerID: s.PeerID,
|
2023-09-29 10:43:25 +05:30
|
|
|
ContentFilter: protocol.ContentFilter{PubsubTopic: s.ContentFilter.PubsubTopic, ContentTopics: maps.Clone(s.ContentFilter.ContentTopics)},
|
2023-02-15 14:43:51 -04:00
|
|
|
C: make(chan *protocol.Envelope),
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2023-02-08 19:33:06 -04:00
|
|
|
func (sub *SubscriptionsMap) clear() {
|
2023-09-29 10:43:25 +05:30
|
|
|
for _, peerSubscription := range sub.Items {
|
|
|
|
for _, subscriptionSet := range peerSubscription.SubsPerPubsubTopic {
|
2023-02-08 19:33:06 -04:00
|
|
|
for _, subscription := range subscriptionSet {
|
2023-09-29 10:43:25 +05:30
|
|
|
subscription.CloseC()
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
sub.Items = make(map[peer.ID]*PeerSubscription)
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func (sub *SubscriptionsMap) Clear() {
|
|
|
|
sub.Lock()
|
|
|
|
defer sub.Unlock()
|
|
|
|
sub.clear()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sub *SubscriptionsMap) Notify(peerID peer.ID, envelope *protocol.Envelope) {
|
|
|
|
sub.RLock()
|
|
|
|
defer sub.RUnlock()
|
|
|
|
|
2023-09-29 10:43:25 +05:30
|
|
|
subscriptions, ok := sub.Items[peerID].SubsPerPubsubTopic[envelope.PubsubTopic()]
|
2023-02-08 19:33:06 -04:00
|
|
|
if ok {
|
2023-05-08 17:33:10 -04:00
|
|
|
iterateSubscriptionSet(sub.logger, subscriptions, envelope)
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-08 17:33:10 -04:00
|
|
|
func iterateSubscriptionSet(logger *zap.Logger, subscriptions SubscriptionSet, envelope *protocol.Envelope) {
|
2023-02-08 19:33:06 -04:00
|
|
|
for _, subscription := range subscriptions {
|
|
|
|
func(subscription *SubscriptionDetails) {
|
|
|
|
subscription.RLock()
|
|
|
|
defer subscription.RUnlock()
|
|
|
|
|
2023-09-19 15:52:11 +03:00
|
|
|
_, ok := subscription.ContentFilter.ContentTopics[envelope.Message().ContentTopic]
|
2023-08-30 17:35:08 +07:00
|
|
|
if !ok { // only send the msg to subscriptions that have matching contentTopic
|
2023-02-08 19:33:06 -04:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-05-08 17:33:10 -04:00
|
|
|
if !subscription.Closed {
|
|
|
|
select {
|
|
|
|
case subscription.C <- envelope:
|
|
|
|
default:
|
|
|
|
logger.Warn("can't deliver message to subscription. subscriber too slow")
|
|
|
|
}
|
2023-02-08 19:33:06 -04:00
|
|
|
}
|
|
|
|
}(subscription)
|
|
|
|
}
|
|
|
|
}
|
2023-06-22 14:55:51 -04:00
|
|
|
|
|
|
|
func (s *SubscriptionDetails) MarshalJSON() ([]byte, error) {
|
|
|
|
type resultType struct {
|
|
|
|
PeerID string `json:"peerID"`
|
|
|
|
PubsubTopic string `json:"pubsubTopic"`
|
|
|
|
ContentTopics []string `json:"contentTopics"`
|
|
|
|
}
|
|
|
|
|
|
|
|
result := resultType{
|
|
|
|
PeerID: s.PeerID.Pretty(),
|
2023-09-19 15:52:11 +03:00
|
|
|
PubsubTopic: s.ContentFilter.PubsubTopic,
|
2023-06-22 14:55:51 -04:00
|
|
|
}
|
|
|
|
|
2023-09-19 15:52:11 +03:00
|
|
|
for c := range s.ContentFilter.ContentTopics {
|
2023-06-22 14:55:51 -04:00
|
|
|
result.ContentTopics = append(result.ContentTopics, c)
|
|
|
|
}
|
|
|
|
|
|
|
|
return json.Marshal(result)
|
|
|
|
}
|