100 lines
2.8 KiB
Go
100 lines
2.8 KiB
Go
package shhext
|
|
|
|
import (
|
|
"fmt"
|
|
"hash/fnv"
|
|
"sync"
|
|
"time"
|
|
|
|
whispertypes "github.com/status-im/status-protocol-go/transport/whisper/types"
|
|
statusproto "github.com/status-im/status-protocol-go/types"
|
|
)
|
|
|
|
const (
|
|
// defaultRequestsDelay will be used in RequestsRegistry if no other was provided.
|
|
defaultRequestsDelay = 3 * time.Second
|
|
)
|
|
|
|
type requestMeta struct {
|
|
timestamp time.Time
|
|
lastUID statusproto.Hash
|
|
}
|
|
|
|
// NewRequestsRegistry creates instance of the RequestsRegistry and returns pointer to it.
|
|
func NewRequestsRegistry(delay time.Duration) *RequestsRegistry {
|
|
r := &RequestsRegistry{
|
|
delay: delay,
|
|
}
|
|
r.Clear()
|
|
return r
|
|
}
|
|
|
|
// RequestsRegistry keeps map for all requests with timestamp when they were made.
|
|
type RequestsRegistry struct {
|
|
mu sync.Mutex
|
|
delay time.Duration
|
|
uidToTopics map[statusproto.Hash]statusproto.Hash
|
|
byTopicsHash map[statusproto.Hash]requestMeta
|
|
}
|
|
|
|
// Register request with given topics. If request with same topics was made in less then configured delay then error
|
|
// will be returned.
|
|
func (r *RequestsRegistry) Register(uid statusproto.Hash, topics []whispertypes.TopicType) error {
|
|
r.mu.Lock()
|
|
defer r.mu.Unlock()
|
|
topicsHash := topicsToHash(topics)
|
|
if meta, exist := r.byTopicsHash[topicsHash]; exist {
|
|
if time.Since(meta.timestamp) < r.delay {
|
|
return fmt.Errorf("another request with the same topics was sent less than %s ago. Please wait for a bit longer, or set `force` to true in request parameters", r.delay)
|
|
}
|
|
}
|
|
newMeta := requestMeta{
|
|
timestamp: time.Now(),
|
|
lastUID: uid,
|
|
}
|
|
r.uidToTopics[uid] = topicsHash
|
|
r.byTopicsHash[topicsHash] = newMeta
|
|
return nil
|
|
}
|
|
|
|
// Has returns true if given uid is stored in registry.
|
|
func (r *RequestsRegistry) Has(uid statusproto.Hash) bool {
|
|
r.mu.Lock()
|
|
defer r.mu.Unlock()
|
|
_, exist := r.uidToTopics[uid]
|
|
return exist
|
|
}
|
|
|
|
// Unregister removes request with given UID from registry.
|
|
func (r *RequestsRegistry) Unregister(uid statusproto.Hash) {
|
|
r.mu.Lock()
|
|
defer r.mu.Unlock()
|
|
topicsHash, exist := r.uidToTopics[uid]
|
|
if !exist {
|
|
return
|
|
}
|
|
delete(r.uidToTopics, uid)
|
|
meta := r.byTopicsHash[topicsHash]
|
|
// remove topicsHash only if we are trying to unregister last request with this topic.
|
|
if meta.lastUID == uid {
|
|
delete(r.byTopicsHash, topicsHash)
|
|
}
|
|
}
|
|
|
|
// Clear recreates all structures used for caching requests.
|
|
func (r *RequestsRegistry) Clear() {
|
|
r.mu.Lock()
|
|
defer r.mu.Unlock()
|
|
r.uidToTopics = map[statusproto.Hash]statusproto.Hash{}
|
|
r.byTopicsHash = map[statusproto.Hash]requestMeta{}
|
|
}
|
|
|
|
// topicsToHash returns non-cryptographic hash of the topics.
|
|
func topicsToHash(topics []whispertypes.TopicType) statusproto.Hash {
|
|
hash := fnv.New32()
|
|
for i := range topics {
|
|
_, _ = hash.Write(topics[i][:]) // never returns error per documentation
|
|
}
|
|
return statusproto.BytesToHash(hash.Sum(nil))
|
|
}
|