mirror of https://github.com/fluffle/goirc
The great state tracker privatisation 1/3: channels.
This commit is contained in:
parent
8231942086
commit
bffe946388
163
state/channel.go
163
state/channel.go
|
@ -3,18 +3,24 @@ package state
|
||||||
import (
|
import (
|
||||||
"github.com/fluffle/goirc/logging"
|
"github.com/fluffle/goirc/logging"
|
||||||
|
|
||||||
"fmt"
|
|
||||||
"reflect"
|
"reflect"
|
||||||
"sort"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
)
|
)
|
||||||
|
|
||||||
// A struct representing an IRC channel
|
// A Channel is returned from the state tracker and contains
|
||||||
|
// a copy of the channel state at a particular time.
|
||||||
type Channel struct {
|
type Channel struct {
|
||||||
Name, Topic string
|
Name, Topic string
|
||||||
Modes *ChanMode
|
Modes *ChanMode
|
||||||
lookup map[string]*Nick
|
Nicks map[string]*ChanPrivs
|
||||||
nicks map[*Nick]*ChanPrivs
|
}
|
||||||
|
|
||||||
|
// Internal bookkeeping struct for channels.
|
||||||
|
type channel struct {
|
||||||
|
name, topic string
|
||||||
|
modes *ChanMode
|
||||||
|
lookup map[string]*nick
|
||||||
|
nicks map[*nick]*ChanPrivs
|
||||||
}
|
}
|
||||||
|
|
||||||
// A struct representing the modes of an IRC Channel
|
// A struct representing the modes of an IRC Channel
|
||||||
|
@ -98,48 +104,57 @@ func init() {
|
||||||
* Channel methods for state management
|
* Channel methods for state management
|
||||||
\******************************************************************************/
|
\******************************************************************************/
|
||||||
|
|
||||||
func NewChannel(name string) *Channel {
|
func newChannel(name string) *channel {
|
||||||
return &Channel{
|
return &channel{
|
||||||
Name: name,
|
name: name,
|
||||||
Modes: new(ChanMode),
|
modes: new(ChanMode),
|
||||||
nicks: make(map[*Nick]*ChanPrivs),
|
nicks: make(map[*nick]*ChanPrivs),
|
||||||
lookup: make(map[string]*Nick),
|
lookup: make(map[string]*nick),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns true if the Nick is associated with the Channel
|
// Returns a copy of the internal tracker channel state at this time.
|
||||||
func (ch *Channel) IsOn(nk *Nick) (*ChanPrivs, bool) {
|
// Relies on tracker-level locking for concurrent access.
|
||||||
|
func (ch *channel) Channel() *Channel {
|
||||||
|
c := &Channel{
|
||||||
|
Name: ch.name,
|
||||||
|
Topic: ch.topic,
|
||||||
|
Modes: ch.modes.Copy(),
|
||||||
|
Nicks: make(map[string]*ChanPrivs),
|
||||||
|
}
|
||||||
|
for n, cp := range ch.nicks {
|
||||||
|
c.Nicks[n.nick] = cp.Copy()
|
||||||
|
}
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ch *channel) isOn(nk *nick) (*ChanPrivs, bool) {
|
||||||
cp, ok := ch.nicks[nk]
|
cp, ok := ch.nicks[nk]
|
||||||
return cp, ok
|
return cp.Copy(), ok
|
||||||
}
|
|
||||||
|
|
||||||
func (ch *Channel) IsOnStr(n string) (*Nick, bool) {
|
|
||||||
nk, ok := ch.lookup[n]
|
|
||||||
return nk, ok
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Associates a Nick with a Channel
|
// Associates a Nick with a Channel
|
||||||
func (ch *Channel) addNick(nk *Nick, cp *ChanPrivs) {
|
func (ch *channel) addNick(nk *nick, cp *ChanPrivs) {
|
||||||
if _, ok := ch.nicks[nk]; !ok {
|
if _, ok := ch.nicks[nk]; !ok {
|
||||||
ch.nicks[nk] = cp
|
ch.nicks[nk] = cp
|
||||||
ch.lookup[nk.Nick] = nk
|
ch.lookup[nk.nick] = nk
|
||||||
} else {
|
} else {
|
||||||
logging.Warn("Channel.addNick(): %s already on %s.", nk.Nick, ch.Name)
|
logging.Warn("Channel.addNick(): %s already on %s.", nk.nick, ch.name)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Disassociates a Nick from a Channel.
|
// Disassociates a Nick from a Channel.
|
||||||
func (ch *Channel) delNick(nk *Nick) {
|
func (ch *channel) delNick(nk *nick) {
|
||||||
if _, ok := ch.nicks[nk]; ok {
|
if _, ok := ch.nicks[nk]; ok {
|
||||||
delete(ch.nicks, nk)
|
delete(ch.nicks, nk)
|
||||||
delete(ch.lookup, nk.Nick)
|
delete(ch.lookup, nk.nick)
|
||||||
} else {
|
} else {
|
||||||
logging.Warn("Channel.delNick(): %s not on %s.", nk.Nick, ch.Name)
|
logging.Warn("Channel.delNick(): %s not on %s.", nk.nick, ch.name)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parses mode strings for a channel.
|
// Parses mode strings for a channel.
|
||||||
func (ch *Channel) ParseModes(modes string, modeargs ...string) {
|
func (ch *channel) parseModes(modes string, modeargs ...string) {
|
||||||
var modeop bool // true => add mode, false => remove mode
|
var modeop bool // true => add mode, false => remove mode
|
||||||
var modestr string
|
var modestr string
|
||||||
for i := 0; i < len(modes); i++ {
|
for i := 0; i < len(modes); i++ {
|
||||||
|
@ -151,43 +166,43 @@ func (ch *Channel) ParseModes(modes string, modeargs ...string) {
|
||||||
modeop = false
|
modeop = false
|
||||||
modestr = string(m)
|
modestr = string(m)
|
||||||
case 'i':
|
case 'i':
|
||||||
ch.Modes.InviteOnly = modeop
|
ch.modes.InviteOnly = modeop
|
||||||
case 'm':
|
case 'm':
|
||||||
ch.Modes.Moderated = modeop
|
ch.modes.Moderated = modeop
|
||||||
case 'n':
|
case 'n':
|
||||||
ch.Modes.NoExternalMsg = modeop
|
ch.modes.NoExternalMsg = modeop
|
||||||
case 'p':
|
case 'p':
|
||||||
ch.Modes.Private = modeop
|
ch.modes.Private = modeop
|
||||||
case 'r':
|
case 'r':
|
||||||
ch.Modes.Registered = modeop
|
ch.modes.Registered = modeop
|
||||||
case 's':
|
case 's':
|
||||||
ch.Modes.Secret = modeop
|
ch.modes.Secret = modeop
|
||||||
case 't':
|
case 't':
|
||||||
ch.Modes.ProtectedTopic = modeop
|
ch.modes.ProtectedTopic = modeop
|
||||||
case 'z':
|
case 'z':
|
||||||
ch.Modes.SSLOnly = modeop
|
ch.modes.SSLOnly = modeop
|
||||||
case 'Z':
|
case 'Z':
|
||||||
ch.Modes.AllSSL = modeop
|
ch.modes.AllSSL = modeop
|
||||||
case 'O':
|
case 'O':
|
||||||
ch.Modes.OperOnly = modeop
|
ch.modes.OperOnly = modeop
|
||||||
case 'k':
|
case 'k':
|
||||||
if modeop && len(modeargs) != 0 {
|
if modeop && len(modeargs) != 0 {
|
||||||
ch.Modes.Key, modeargs = modeargs[0], modeargs[1:]
|
ch.modes.Key, modeargs = modeargs[0], modeargs[1:]
|
||||||
} else if !modeop {
|
} else if !modeop {
|
||||||
ch.Modes.Key = ""
|
ch.modes.Key = ""
|
||||||
} else {
|
} else {
|
||||||
logging.Warn("Channel.ParseModes(): not enough arguments to "+
|
logging.Warn("Channel.ParseModes(): not enough arguments to "+
|
||||||
"process MODE %s %s%c", ch.Name, modestr, m)
|
"process MODE %s %s%c", ch.name, modestr, m)
|
||||||
}
|
}
|
||||||
case 'l':
|
case 'l':
|
||||||
if modeop && len(modeargs) != 0 {
|
if modeop && len(modeargs) != 0 {
|
||||||
ch.Modes.Limit, _ = strconv.Atoi(modeargs[0])
|
ch.modes.Limit, _ = strconv.Atoi(modeargs[0])
|
||||||
modeargs = modeargs[1:]
|
modeargs = modeargs[1:]
|
||||||
} else if !modeop {
|
} else if !modeop {
|
||||||
ch.Modes.Limit = 0
|
ch.modes.Limit = 0
|
||||||
} else {
|
} else {
|
||||||
logging.Warn("Channel.ParseModes(): not enough arguments to "+
|
logging.Warn("Channel.ParseModes(): not enough arguments to "+
|
||||||
"process MODE %s %s%c", ch.Name, modestr, m)
|
"process MODE %s %s%c", ch.name, modestr, m)
|
||||||
}
|
}
|
||||||
case 'q', 'a', 'o', 'h', 'v':
|
case 'q', 'a', 'o', 'h', 'v':
|
||||||
if len(modeargs) != 0 {
|
if len(modeargs) != 0 {
|
||||||
|
@ -208,11 +223,11 @@ func (ch *Channel) ParseModes(modes string, modeargs ...string) {
|
||||||
modeargs = modeargs[1:]
|
modeargs = modeargs[1:]
|
||||||
} else {
|
} else {
|
||||||
logging.Warn("Channel.ParseModes(): untracked nick %s "+
|
logging.Warn("Channel.ParseModes(): untracked nick %s "+
|
||||||
"received MODE on channel %s", modeargs[0], ch.Name)
|
"received MODE on channel %s", modeargs[0], ch.name)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
logging.Warn("Channel.ParseModes(): not enough arguments to "+
|
logging.Warn("Channel.ParseModes(): not enough arguments to "+
|
||||||
"process MODE %s %s%c", ch.Name, modestr, m)
|
"process MODE %s %s%c", ch.name, modestr, m)
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
logging.Info("Channel.ParseModes(): unknown mode char %c", m)
|
logging.Info("Channel.ParseModes(): unknown mode char %c", m)
|
||||||
|
@ -220,29 +235,39 @@ func (ch *Channel) ParseModes(modes string, modeargs ...string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type byNick []*Nick
|
// Returns true if the Nick is associated with the Channel
|
||||||
|
func (ch *Channel) IsOn(nk string) (*ChanPrivs, bool) {
|
||||||
func (b byNick) Len() int { return len(b) }
|
cp, ok := ch.Nicks[nk]
|
||||||
func (b byNick) Less(i, j int) bool { return b[i].Nick < b[j].Nick }
|
return cp, ok
|
||||||
func (b byNick) Swap(i, j int) { b[i], b[j] = b[j], b[i] }
|
|
||||||
|
|
||||||
// Nicks returns a list of *Nick that are on the channel, sorted by nick.
|
|
||||||
func (ch *Channel) Nicks() []*Nick {
|
|
||||||
nicks := make([]*Nick, 0, len(ch.lookup))
|
|
||||||
for _, nick := range ch.lookup {
|
|
||||||
nicks = append(nicks, nick)
|
|
||||||
}
|
|
||||||
sort.Sort(byNick(nicks))
|
|
||||||
return nicks
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// NicksStr returns a list of nick strings that are on the channel, sorted by nick.
|
// Test Channel equality.
|
||||||
func (ch *Channel) NicksStr() []string {
|
func (ch *Channel) Equals(other *Channel) bool {
|
||||||
var nicks []string
|
return reflect.DeepEqual(ch, other)
|
||||||
for _, nick := range ch.Nicks() {
|
|
||||||
nicks = append(nicks, nick.Nick)
|
|
||||||
}
|
}
|
||||||
return nicks
|
|
||||||
|
// Duplicates a ChanMode struct.
|
||||||
|
func (cm *ChanMode) Copy() *ChanMode {
|
||||||
|
if cm == nil { return nil }
|
||||||
|
c := *cm
|
||||||
|
return &c
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test ChanMode equality.
|
||||||
|
func (cm *ChanMode) Equals(other *ChanMode) bool {
|
||||||
|
return reflect.DeepEqual(cm, other)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Duplicates a ChanPrivs struct.
|
||||||
|
func (cp *ChanPrivs) Copy() *ChanPrivs {
|
||||||
|
if cp == nil { return nil }
|
||||||
|
c := *cp
|
||||||
|
return &c
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test ChanPrivs equality.
|
||||||
|
func (cp *ChanPrivs) Equals(other *ChanPrivs) bool {
|
||||||
|
return reflect.DeepEqual(cp, other)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns a string representing the channel. Looks like:
|
// Returns a string representing the channel. Looks like:
|
||||||
|
@ -257,12 +282,16 @@ func (ch *Channel) String() string {
|
||||||
str += "Topic: " + ch.Topic + "\n\t"
|
str += "Topic: " + ch.Topic + "\n\t"
|
||||||
str += "Modes: " + ch.Modes.String() + "\n\t"
|
str += "Modes: " + ch.Modes.String() + "\n\t"
|
||||||
str += "Nicks: \n"
|
str += "Nicks: \n"
|
||||||
for nk, cp := range ch.nicks {
|
for nk, cp := range ch.Nicks {
|
||||||
str += "\t\t" + nk.Nick + ": " + cp.String() + "\n"
|
str += "\t\t" + nk + ": " + cp.String() + "\n"
|
||||||
}
|
}
|
||||||
return str
|
return str
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (ch *channel) String() string {
|
||||||
|
return ch.Channel().String()
|
||||||
|
}
|
||||||
|
|
||||||
// Returns a string representing the channel modes. Looks like:
|
// Returns a string representing the channel modes. Looks like:
|
||||||
// +npk key
|
// +npk key
|
||||||
func (cm *ChanMode) String() string {
|
func (cm *ChanMode) String() string {
|
||||||
|
@ -284,7 +313,7 @@ func (cm *ChanMode) String() string {
|
||||||
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
if f.Int() != 0 {
|
if f.Int() != 0 {
|
||||||
str += ChanModeToString[t.Field(i).Name]
|
str += ChanModeToString[t.Field(i).Name]
|
||||||
a = append(a, fmt.Sprintf("%d", f.Int()))
|
a = append(a, strconv.FormatInt(f.Int(), 10))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,23 +1,35 @@
|
||||||
package state
|
package state
|
||||||
|
|
||||||
import (
|
import "testing"
|
||||||
"testing"
|
|
||||||
)
|
func compareChannel(t *testing.T, ch *channel) {
|
||||||
|
c := ch.Channel()
|
||||||
|
if c.Name != ch.name || c.Topic != ch.topic ||
|
||||||
|
!c.Modes.Equals(ch.modes) || len(c.Nicks) != len(ch.nicks) {
|
||||||
|
t.Errorf("Channel not duped correctly from internal state.")
|
||||||
|
}
|
||||||
|
for nk, cp := range ch.nicks {
|
||||||
|
if other, ok := c.Nicks[nk.nick]; !ok || !cp.Equals(other) {
|
||||||
|
t.Errorf("Nick not duped correctly from internal state.")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestNewChannel(t *testing.T) {
|
func TestNewChannel(t *testing.T) {
|
||||||
ch := NewChannel("#test1")
|
ch := newChannel("#test1")
|
||||||
|
|
||||||
if ch.Name != "#test1" {
|
if ch.name != "#test1" {
|
||||||
t.Errorf("Channel not created correctly by NewChannel()")
|
t.Errorf("Channel not created correctly by NewChannel()")
|
||||||
}
|
}
|
||||||
if len(ch.nicks) != 0 || len(ch.lookup) != 0 {
|
if len(ch.nicks) != 0 || len(ch.lookup) != 0 {
|
||||||
t.Errorf("Channel maps contain data after NewChannel()")
|
t.Errorf("Channel maps contain data after NewChannel()")
|
||||||
}
|
}
|
||||||
|
compareChannel(t, ch)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestAddNick(t *testing.T) {
|
func TestAddNick(t *testing.T) {
|
||||||
ch := NewChannel("#test1")
|
ch := newChannel("#test1")
|
||||||
nk := NewNick("test1")
|
nk := newNick("test1")
|
||||||
cp := new(ChanPrivs)
|
cp := new(ChanPrivs)
|
||||||
|
|
||||||
ch.addNick(nk, cp)
|
ch.addNick(nk, cp)
|
||||||
|
@ -31,11 +43,12 @@ func TestAddNick(t *testing.T) {
|
||||||
if n, ok := ch.lookup["test1"]; !ok || n != nk {
|
if n, ok := ch.lookup["test1"]; !ok || n != nk {
|
||||||
t.Errorf("Nick test1 not properly stored in lookup map.")
|
t.Errorf("Nick test1 not properly stored in lookup map.")
|
||||||
}
|
}
|
||||||
|
compareChannel(t, ch)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDelNick(t *testing.T) {
|
func TestDelNick(t *testing.T) {
|
||||||
ch := NewChannel("#test1")
|
ch := newChannel("#test1")
|
||||||
nk := NewNick("test1")
|
nk := newNick("test1")
|
||||||
cp := new(ChanPrivs)
|
cp := new(ChanPrivs)
|
||||||
|
|
||||||
ch.addNick(nk, cp)
|
ch.addNick(nk, cp)
|
||||||
|
@ -49,18 +62,20 @@ func TestDelNick(t *testing.T) {
|
||||||
if n, ok := ch.lookup["#test1"]; ok || n != nil {
|
if n, ok := ch.lookup["#test1"]; ok || n != nil {
|
||||||
t.Errorf("Nick test1 not properly removed from lookup map.")
|
t.Errorf("Nick test1 not properly removed from lookup map.")
|
||||||
}
|
}
|
||||||
|
compareChannel(t, ch)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestChannelParseModes(t *testing.T) {
|
func TestChannelParseModes(t *testing.T) {
|
||||||
ch := NewChannel("#test1")
|
ch := newChannel("#test1")
|
||||||
md := ch.Modes
|
md := ch.modes
|
||||||
|
|
||||||
// Channel modes can adjust channel privs too, so we need a Nick
|
// Channel modes can adjust channel privs too, so we need a Nick
|
||||||
nk := NewNick("test1")
|
nk := newNick("test1")
|
||||||
cp := new(ChanPrivs)
|
cp := new(ChanPrivs)
|
||||||
ch.addNick(nk, cp)
|
ch.addNick(nk, cp)
|
||||||
|
|
||||||
// Test bools first.
|
// Test bools first.
|
||||||
|
compareChannel(t, ch)
|
||||||
if md.Private || md.Secret || md.ProtectedTopic || md.NoExternalMsg ||
|
if md.Private || md.Secret || md.ProtectedTopic || md.NoExternalMsg ||
|
||||||
md.Moderated || md.InviteOnly || md.OperOnly || md.SSLOnly {
|
md.Moderated || md.InviteOnly || md.OperOnly || md.SSLOnly {
|
||||||
t.Errorf("Modes for new channel set to true.")
|
t.Errorf("Modes for new channel set to true.")
|
||||||
|
@ -72,8 +87,9 @@ func TestChannelParseModes(t *testing.T) {
|
||||||
md.InviteOnly = true
|
md.InviteOnly = true
|
||||||
|
|
||||||
// Flip some MOAR bits.
|
// Flip some MOAR bits.
|
||||||
ch.ParseModes("+s-p+tm-i")
|
ch.parseModes("+s-p+tm-i")
|
||||||
|
|
||||||
|
compareChannel(t, ch)
|
||||||
if md.Private || !md.Secret || !md.ProtectedTopic || !md.NoExternalMsg ||
|
if md.Private || !md.Secret || !md.ProtectedTopic || !md.NoExternalMsg ||
|
||||||
!md.Moderated || md.InviteOnly || md.OperOnly || md.SSLOnly {
|
!md.Moderated || md.InviteOnly || md.OperOnly || md.SSLOnly {
|
||||||
t.Errorf("Modes not flipped correctly by ParseModes.")
|
t.Errorf("Modes not flipped correctly by ParseModes.")
|
||||||
|
@ -85,19 +101,22 @@ func TestChannelParseModes(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// enable limit correctly
|
// enable limit correctly
|
||||||
ch.ParseModes("+l", "256")
|
ch.parseModes("+l", "256")
|
||||||
|
compareChannel(t, ch)
|
||||||
if md.Limit != 256 {
|
if md.Limit != 256 {
|
||||||
t.Errorf("Limit for channel not set correctly")
|
t.Errorf("Limit for channel not set correctly")
|
||||||
}
|
}
|
||||||
|
|
||||||
// enable limit incorrectly
|
// enable limit incorrectly
|
||||||
ch.ParseModes("+l")
|
ch.parseModes("+l")
|
||||||
|
compareChannel(t, ch)
|
||||||
if md.Limit != 256 {
|
if md.Limit != 256 {
|
||||||
t.Errorf("Bad limit value caused limit to be unset.")
|
t.Errorf("Bad limit value caused limit to be unset.")
|
||||||
}
|
}
|
||||||
|
|
||||||
// disable limit correctly
|
// disable limit correctly
|
||||||
ch.ParseModes("-l")
|
ch.parseModes("-l")
|
||||||
|
compareChannel(t, ch)
|
||||||
if md.Limit != 0 {
|
if md.Limit != 0 {
|
||||||
t.Errorf("Limit for channel not unset correctly")
|
t.Errorf("Limit for channel not unset correctly")
|
||||||
}
|
}
|
||||||
|
@ -108,19 +127,22 @@ func TestChannelParseModes(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// enable key correctly
|
// enable key correctly
|
||||||
ch.ParseModes("+k", "foobar")
|
ch.parseModes("+k", "foobar")
|
||||||
|
compareChannel(t, ch)
|
||||||
if md.Key != "foobar" {
|
if md.Key != "foobar" {
|
||||||
t.Errorf("Key for channel not set correctly")
|
t.Errorf("Key for channel not set correctly")
|
||||||
}
|
}
|
||||||
|
|
||||||
// enable key incorrectly
|
// enable key incorrectly
|
||||||
ch.ParseModes("+k")
|
ch.parseModes("+k")
|
||||||
|
compareChannel(t, ch)
|
||||||
if md.Key != "foobar" {
|
if md.Key != "foobar" {
|
||||||
t.Errorf("Bad key value caused key to be unset.")
|
t.Errorf("Bad key value caused key to be unset.")
|
||||||
}
|
}
|
||||||
|
|
||||||
// disable key correctly
|
// disable key correctly
|
||||||
ch.ParseModes("-k")
|
ch.parseModes("-k")
|
||||||
|
compareChannel(t, ch)
|
||||||
if md.Key != "" {
|
if md.Key != "" {
|
||||||
t.Errorf("Key for channel not unset correctly")
|
t.Errorf("Key for channel not unset correctly")
|
||||||
}
|
}
|
||||||
|
@ -128,16 +150,18 @@ func TestChannelParseModes(t *testing.T) {
|
||||||
// Test chan privs parsing.
|
// Test chan privs parsing.
|
||||||
cp.Op = true
|
cp.Op = true
|
||||||
cp.HalfOp = true
|
cp.HalfOp = true
|
||||||
ch.ParseModes("+aq-o", "test1", "test1", "test1")
|
ch.parseModes("+aq-o", "test1", "test1", "test1")
|
||||||
|
|
||||||
|
compareChannel(t, ch)
|
||||||
if !cp.Owner || !cp.Admin || cp.Op || !cp.HalfOp || cp.Voice {
|
if !cp.Owner || !cp.Admin || cp.Op || !cp.HalfOp || cp.Voice {
|
||||||
t.Errorf("Channel privileges not flipped correctly by ParseModes.")
|
t.Errorf("Channel privileges not flipped correctly by ParseModes.")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Test a random mix of modes, just to be sure
|
// Test a random mix of modes, just to be sure
|
||||||
md.Limit = 256
|
md.Limit = 256
|
||||||
ch.ParseModes("+zpt-qsl+kv-h", "test1", "foobar", "test1")
|
ch.parseModes("+zpt-qsl+kv-h", "test1", "foobar", "test1")
|
||||||
|
|
||||||
|
compareChannel(t, ch)
|
||||||
if !md.Private || md.Secret || !md.ProtectedTopic || !md.NoExternalMsg ||
|
if !md.Private || md.Secret || !md.ProtectedTopic || !md.NoExternalMsg ||
|
||||||
!md.Moderated || md.InviteOnly || md.OperOnly || !md.SSLOnly {
|
!md.Moderated || md.InviteOnly || md.OperOnly || !md.SSLOnly {
|
||||||
t.Errorf("Modes not flipped correctly by ParseModes (2).")
|
t.Errorf("Modes not flipped correctly by ParseModes (2).")
|
||||||
|
|
Loading…
Reference in New Issue