Signed-off-by: NotAShelf <raf@notashelf.dev> Change-Id: Iaf89cda3c480d6a8371e5f146ee95fcf6a6a6964
58 lines
1.2 KiB
Go
58 lines
1.2 KiB
Go
package normalize
|
|
|
|
import "sync"
|
|
|
|
// Bounded set of observed referrer domains.
|
|
type ReferrerRegistry struct {
|
|
mu sync.RWMutex
|
|
sources map[string]struct{}
|
|
maxSources int
|
|
overflowCount int
|
|
}
|
|
|
|
func NewReferrerRegistry(maxSources int) *ReferrerRegistry {
|
|
return &ReferrerRegistry{
|
|
sources: make(map[string]struct{}, maxSources),
|
|
maxSources: maxSources,
|
|
}
|
|
}
|
|
|
|
// Attempt to add a referrer source to the registry.
|
|
// Returns the source to use ("other" if rejected).
|
|
func (r *ReferrerRegistry) Add(source string) string {
|
|
if source == "direct" || source == "internal" {
|
|
return source
|
|
}
|
|
|
|
// Fast path: check with read lock first
|
|
r.mu.RLock()
|
|
if _, exists := r.sources[source]; exists {
|
|
r.mu.RUnlock()
|
|
return source
|
|
}
|
|
r.mu.RUnlock()
|
|
|
|
// Slow path: acquire write lock to add
|
|
r.mu.Lock()
|
|
defer r.mu.Unlock()
|
|
|
|
// Double-check after acquiring write lock, another goroutine might have added it beforehand
|
|
if _, exists := r.sources[source]; exists {
|
|
return source
|
|
}
|
|
|
|
// Check limit
|
|
if len(r.sources) >= r.maxSources {
|
|
r.overflowCount++
|
|
return "other"
|
|
}
|
|
|
|
r.sources[source] = struct{}{}
|
|
return source
|
|
}
|
|
|
|
func (r *ReferrerRegistry) OverflowCount() int {
|
|
r.mu.RLock()
|
|
defer r.mu.RUnlock()
|
|
return r.overflowCount
|
|
}
|