2016-06-15 13:50:56 +02:00
|
|
|
// Copyright (c) 2012-2014 Jeremy Latt
|
|
|
|
// Copyright (c) 2014-2015 Edmund Huber
|
2017-03-27 14:15:02 +02:00
|
|
|
// Copyright (c) 2016 Daniel Oaks <daniel@danieloaks.net>
|
2016-06-15 13:50:56 +02:00
|
|
|
// released under the MIT license
|
|
|
|
|
2014-02-09 07:06:10 +01:00
|
|
|
package irc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2014-02-18 00:25:32 +01:00
|
|
|
"strings"
|
2017-04-17 13:01:39 +02:00
|
|
|
"sync"
|
2014-02-09 07:06:10 +01:00
|
|
|
)
|
|
|
|
|
2014-02-09 07:42:14 +01:00
|
|
|
//
|
2014-02-09 07:06:10 +01:00
|
|
|
// simple types
|
2014-02-09 07:42:14 +01:00
|
|
|
//
|
2014-02-09 07:06:10 +01:00
|
|
|
|
2017-04-17 13:01:39 +02:00
|
|
|
type ChannelNameMap struct {
|
|
|
|
ChansLock sync.RWMutex
|
|
|
|
Chans map[string]*Channel
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewChannelNameMap() ChannelNameMap {
|
|
|
|
var channels ChannelNameMap
|
|
|
|
channels.Chans = make(map[string]*Channel)
|
|
|
|
return channels
|
|
|
|
}
|
2014-02-09 08:15:05 +01:00
|
|
|
|
2016-10-11 15:51:46 +02:00
|
|
|
func (channels ChannelNameMap) Get(name string) *Channel {
|
|
|
|
name, err := CasefoldChannel(name)
|
|
|
|
if err == nil {
|
2017-04-17 13:01:39 +02:00
|
|
|
channels.ChansLock.RLock()
|
|
|
|
defer channels.ChansLock.RUnlock()
|
|
|
|
return channels.Chans[name]
|
2016-10-11 15:51:46 +02:00
|
|
|
}
|
|
|
|
return nil
|
2014-02-26 05:17:26 +01:00
|
|
|
}
|
|
|
|
|
2014-02-09 08:15:05 +01:00
|
|
|
func (channels ChannelNameMap) Add(channel *Channel) error {
|
2017-04-17 13:01:39 +02:00
|
|
|
channels.ChansLock.Lock()
|
|
|
|
defer channels.ChansLock.Unlock()
|
|
|
|
if channels.Chans[channel.nameCasefolded] != nil {
|
2014-02-09 08:15:05 +01:00
|
|
|
return fmt.Errorf("%s: already set", channel.name)
|
|
|
|
}
|
2017-04-17 13:01:39 +02:00
|
|
|
channels.Chans[channel.nameCasefolded] = channel
|
2014-02-09 08:15:05 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (channels ChannelNameMap) Remove(channel *Channel) error {
|
2017-04-17 13:01:39 +02:00
|
|
|
channels.ChansLock.Lock()
|
|
|
|
defer channels.ChansLock.Unlock()
|
|
|
|
if channel != channels.Chans[channel.nameCasefolded] {
|
2014-02-09 08:15:05 +01:00
|
|
|
return fmt.Errorf("%s: mismatch", channel.name)
|
|
|
|
}
|
2017-04-17 13:01:39 +02:00
|
|
|
delete(channels.Chans, channel.nameCasefolded)
|
2014-02-09 08:15:05 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-04-17 13:01:39 +02:00
|
|
|
func (channels ChannelNameMap) Len() int {
|
|
|
|
channels.ChansLock.RLock()
|
|
|
|
defer channels.ChansLock.RUnlock()
|
|
|
|
return len(channels.Chans)
|
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
type ModeSet map[Mode]bool
|
2014-02-15 06:57:08 +01:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
func (set ModeSet) String() string {
|
2014-02-26 00:57:35 +01:00
|
|
|
if len(set) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
2014-02-25 20:11:34 +01:00
|
|
|
strs := make([]string, len(set))
|
|
|
|
index := 0
|
|
|
|
for mode := range set {
|
|
|
|
strs[index] = mode.String()
|
|
|
|
index += 1
|
|
|
|
}
|
|
|
|
return strings.Join(strs, "")
|
|
|
|
}
|
|
|
|
|
2014-02-17 02:23:47 +01:00
|
|
|
type ClientSet map[*Client]bool
|
2014-02-09 08:15:05 +01:00
|
|
|
|
|
|
|
func (clients ClientSet) Add(client *Client) {
|
2014-02-17 02:23:47 +01:00
|
|
|
clients[client] = true
|
2014-02-09 08:15:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (clients ClientSet) Remove(client *Client) {
|
|
|
|
delete(clients, client)
|
|
|
|
}
|
|
|
|
|
2014-02-17 02:23:47 +01:00
|
|
|
func (clients ClientSet) Has(client *Client) bool {
|
|
|
|
return clients[client]
|
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
type MemberSet map[*Client]ModeSet
|
2014-02-17 02:23:47 +01:00
|
|
|
|
|
|
|
func (members MemberSet) Add(member *Client) {
|
2017-03-24 03:23:21 +01:00
|
|
|
members[member] = make(ModeSet)
|
2014-02-17 02:23:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (members MemberSet) Remove(member *Client) {
|
|
|
|
delete(members, member)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (members MemberSet) Has(member *Client) bool {
|
|
|
|
_, ok := members[member]
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
func (members MemberSet) HasMode(member *Client, mode Mode) bool {
|
2014-02-17 02:23:47 +01:00
|
|
|
modes, ok := members[member]
|
2014-02-15 06:57:08 +01:00
|
|
|
if !ok {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return modes[mode]
|
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
func (members MemberSet) AnyHasMode(mode Mode) bool {
|
2014-03-22 22:25:24 +01:00
|
|
|
for _, modes := range members {
|
2014-03-23 06:47:21 +01:00
|
|
|
if modes[mode] {
|
2014-03-22 22:25:24 +01:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2014-02-09 08:15:05 +01:00
|
|
|
type ChannelSet map[*Channel]bool
|
|
|
|
|
|
|
|
func (channels ChannelSet) Add(channel *Channel) {
|
|
|
|
channels[channel] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (channels ChannelSet) Remove(channel *Channel) {
|
|
|
|
delete(channels, channel)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (channels ChannelSet) First() *Channel {
|
|
|
|
for channel := range channels {
|
|
|
|
return channel
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-02-09 07:42:14 +01:00
|
|
|
//
|
2014-02-09 07:06:10 +01:00
|
|
|
// interfaces
|
2014-02-09 07:42:14 +01:00
|
|
|
//
|
2014-02-09 07:06:10 +01:00
|
|
|
|
2014-03-13 01:52:25 +01:00
|
|
|
type Identifiable interface {
|
2016-10-11 15:51:46 +02:00
|
|
|
Id() string
|
|
|
|
Nick() string
|
2014-02-09 17:48:11 +01:00
|
|
|
}
|