mirror of
https://github.com/ergochat/ergo.git
synced 2025-01-22 18:24:17 +01:00
168 lines
3.5 KiB
Go
168 lines
3.5 KiB
Go
|
// Copyright (c) 2012-2014 Jeremy Latt
|
||
|
// Copyright (c) 2016-2018 Daniel Oaks
|
||
|
// Copyright (c) 2019-2020 Shivaram Lingamneni
|
||
|
// released under the MIT license
|
||
|
|
||
|
package irc
|
||
|
|
||
|
import (
|
||
|
"regexp"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
"sync/atomic"
|
||
|
"time"
|
||
|
"unsafe"
|
||
|
|
||
|
"github.com/oragono/oragono/irc/utils"
|
||
|
)
|
||
|
|
||
|
type MaskInfo struct {
|
||
|
TimeCreated time.Time
|
||
|
CreatorNickmask string
|
||
|
CreatorAccount string
|
||
|
}
|
||
|
|
||
|
// UserMaskSet holds a set of client masks and lets you match hostnames to them.
|
||
|
type UserMaskSet struct {
|
||
|
sync.RWMutex
|
||
|
serialCacheUpdateMutex sync.Mutex
|
||
|
masks map[string]MaskInfo
|
||
|
regexp unsafe.Pointer
|
||
|
muteRegexp unsafe.Pointer
|
||
|
}
|
||
|
|
||
|
func NewUserMaskSet() *UserMaskSet {
|
||
|
return new(UserMaskSet)
|
||
|
}
|
||
|
|
||
|
// Add adds the given mask to this set.
|
||
|
func (set *UserMaskSet) Add(mask, creatorNickmask, creatorAccount string) (maskAdded string, err error) {
|
||
|
casefoldedMask, err := CanonicalizeMaskWildcard(mask)
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
set.serialCacheUpdateMutex.Lock()
|
||
|
defer set.serialCacheUpdateMutex.Unlock()
|
||
|
|
||
|
set.Lock()
|
||
|
if set.masks == nil {
|
||
|
set.masks = make(map[string]MaskInfo)
|
||
|
}
|
||
|
_, present := set.masks[casefoldedMask]
|
||
|
if !present {
|
||
|
maskAdded = casefoldedMask
|
||
|
set.masks[casefoldedMask] = MaskInfo{
|
||
|
TimeCreated: time.Now().UTC(),
|
||
|
CreatorNickmask: creatorNickmask,
|
||
|
CreatorAccount: creatorAccount,
|
||
|
}
|
||
|
}
|
||
|
set.Unlock()
|
||
|
|
||
|
if !present {
|
||
|
set.setRegexp()
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Remove removes the given mask from this set.
|
||
|
func (set *UserMaskSet) Remove(mask string) (maskRemoved string, err error) {
|
||
|
mask, err = CanonicalizeMaskWildcard(mask)
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
set.serialCacheUpdateMutex.Lock()
|
||
|
defer set.serialCacheUpdateMutex.Unlock()
|
||
|
|
||
|
set.Lock()
|
||
|
_, removed := set.masks[mask]
|
||
|
if removed {
|
||
|
maskRemoved = mask
|
||
|
delete(set.masks, mask)
|
||
|
}
|
||
|
set.Unlock()
|
||
|
|
||
|
if removed {
|
||
|
set.setRegexp()
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (set *UserMaskSet) SetMasks(masks map[string]MaskInfo) {
|
||
|
set.Lock()
|
||
|
set.masks = masks
|
||
|
set.Unlock()
|
||
|
set.setRegexp()
|
||
|
}
|
||
|
|
||
|
func (set *UserMaskSet) Masks() (result map[string]MaskInfo) {
|
||
|
set.RLock()
|
||
|
defer set.RUnlock()
|
||
|
|
||
|
result = make(map[string]MaskInfo, len(set.masks))
|
||
|
for mask, info := range set.masks {
|
||
|
result[mask] = info
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Match matches the given n!u@h against the standard (non-ext) bans.
|
||
|
func (set *UserMaskSet) Match(userhost string) bool {
|
||
|
regexp := (*regexp.Regexp)(atomic.LoadPointer(&set.regexp))
|
||
|
|
||
|
if regexp == nil {
|
||
|
return false
|
||
|
}
|
||
|
return regexp.MatchString(userhost)
|
||
|
}
|
||
|
|
||
|
// MatchMute matches the given NUH against the mute extbans.
|
||
|
func (set *UserMaskSet) MatchMute(userhost string) bool {
|
||
|
regexp := set.MuteRegexp()
|
||
|
|
||
|
if regexp == nil {
|
||
|
return false
|
||
|
}
|
||
|
return regexp.MatchString(userhost)
|
||
|
}
|
||
|
|
||
|
func (set *UserMaskSet) MuteRegexp() *regexp.Regexp {
|
||
|
return (*regexp.Regexp)(atomic.LoadPointer(&set.muteRegexp))
|
||
|
}
|
||
|
|
||
|
func (set *UserMaskSet) Length() int {
|
||
|
set.RLock()
|
||
|
defer set.RUnlock()
|
||
|
return len(set.masks)
|
||
|
}
|
||
|
|
||
|
func (set *UserMaskSet) setRegexp() {
|
||
|
set.RLock()
|
||
|
maskExprs := make([]string, 0, len(set.masks))
|
||
|
var muteExprs []string
|
||
|
for mask := range set.masks {
|
||
|
if strings.HasPrefix(mask, "m:") {
|
||
|
muteExprs = append(muteExprs, mask[2:])
|
||
|
} else {
|
||
|
maskExprs = append(maskExprs, mask)
|
||
|
}
|
||
|
}
|
||
|
set.RUnlock()
|
||
|
|
||
|
compileMasks := func(masks []string) *regexp.Regexp {
|
||
|
if len(masks) == 0 {
|
||
|
return nil
|
||
|
}
|
||
|
re, _ := utils.CompileMasks(masks)
|
||
|
return re
|
||
|
}
|
||
|
|
||
|
re := compileMasks(maskExprs)
|
||
|
muteRe := compileMasks(muteExprs)
|
||
|
|
||
|
atomic.StorePointer(&set.regexp, unsafe.Pointer(re))
|
||
|
atomic.StorePointer(&set.muteRegexp, unsafe.Pointer(muteRe))
|
||
|
}
|