2016-06-15 13:50:56 +02:00
|
|
|
// Copyright (c) 2012-2014 Jeremy Latt
|
|
|
|
// Copyright (c) 2014-2015 Edmund Huber
|
2017-03-12 23:08:18 +01:00
|
|
|
// Copyright (c) 2016-2017 Daniel Oaks <daniel@danieloaks.net>
|
2016-06-15 13:50:56 +02:00
|
|
|
// released under the MIT license
|
|
|
|
|
2014-03-13 21:18:40 +01:00
|
|
|
package irc
|
|
|
|
|
|
|
|
import (
|
2016-06-26 13:06:28 +02:00
|
|
|
"strconv"
|
2014-03-13 21:18:40 +01:00
|
|
|
"strings"
|
2016-06-20 14:53:45 +02:00
|
|
|
|
2017-06-14 20:00:53 +02:00
|
|
|
"github.com/oragono/oragono/irc/sno"
|
2014-03-13 21:18:40 +01:00
|
|
|
)
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
// ModeOp is an operation performed with modes
|
|
|
|
type ModeOp rune
|
2016-06-28 08:22:35 +02:00
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
func (op ModeOp) String() string {
|
|
|
|
return string(op)
|
2016-06-28 08:22:35 +02:00
|
|
|
}
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
const (
|
2017-06-19 22:53:16 +02:00
|
|
|
// Add is used when adding the given key.
|
|
|
|
Add ModeOp = '+'
|
|
|
|
// List is used when listing modes (for instance, listing the current bans on a channel).
|
|
|
|
List ModeOp = '='
|
|
|
|
// Remove is used when taking away the given key.
|
2017-03-12 23:08:18 +01:00
|
|
|
Remove ModeOp = '-'
|
|
|
|
)
|
2016-06-28 08:22:35 +02:00
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
// Mode represents a user/channel/server mode
|
|
|
|
type Mode rune
|
2014-03-13 21:18:40 +01:00
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
func (mode Mode) String() string {
|
2014-03-13 21:18:40 +01:00
|
|
|
return string(mode)
|
|
|
|
}
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
// ModeChange is a single mode changing
|
|
|
|
type ModeChange struct {
|
|
|
|
mode Mode
|
2016-06-28 08:22:35 +02:00
|
|
|
op ModeOp
|
|
|
|
arg string
|
|
|
|
}
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
func (change *ModeChange) String() (str string) {
|
2016-06-28 08:22:35 +02:00
|
|
|
if (change.op == Add) || (change.op == Remove) {
|
|
|
|
str = change.op.String()
|
|
|
|
}
|
|
|
|
str += change.mode.String()
|
|
|
|
if change.arg != "" {
|
|
|
|
str += " " + change.arg
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
// ModeChanges are a collection of 'ModeChange's
|
|
|
|
type ModeChanges []ModeChange
|
2016-06-28 08:22:35 +02:00
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
func (changes ModeChanges) String() string {
|
2016-06-28 08:22:35 +02:00
|
|
|
if len(changes) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
op := changes[0].op
|
|
|
|
str := changes[0].op.String()
|
|
|
|
|
|
|
|
for _, change := range changes {
|
|
|
|
if change.op != op {
|
|
|
|
op = change.op
|
|
|
|
str += change.op.String()
|
|
|
|
}
|
|
|
|
str += change.mode.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, change := range changes {
|
|
|
|
if change.arg == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
str += " " + change.arg
|
|
|
|
}
|
|
|
|
return str
|
|
|
|
}
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
// Modes is just a raw list of modes
|
|
|
|
type Modes []Mode
|
2014-03-13 21:18:40 +01:00
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
func (modes Modes) String() string {
|
|
|
|
strs := make([]string, len(modes))
|
|
|
|
for index, mode := range modes {
|
|
|
|
strs[index] = mode.String()
|
|
|
|
}
|
|
|
|
return strings.Join(strs, "")
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
// User Modes
|
2014-03-13 21:18:40 +01:00
|
|
|
const (
|
2017-03-12 23:08:18 +01:00
|
|
|
Away Mode = 'a'
|
2018-01-07 03:56:51 +01:00
|
|
|
Bot Mode = 'B'
|
2017-03-12 23:08:18 +01:00
|
|
|
Invisible Mode = 'i'
|
|
|
|
LocalOperator Mode = 'O'
|
|
|
|
Operator Mode = 'o'
|
|
|
|
Restricted Mode = 'r'
|
2017-08-17 09:52:30 +02:00
|
|
|
RegisteredOnly Mode = 'R'
|
2017-05-08 01:15:16 +02:00
|
|
|
ServerNotice Mode = 's'
|
2017-03-12 23:08:18 +01:00
|
|
|
TLS Mode = 'Z'
|
|
|
|
UserRoleplaying Mode = 'E'
|
|
|
|
WallOps Mode = 'w'
|
2014-03-13 21:18:40 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2017-06-19 22:53:16 +02:00
|
|
|
// SupportedUserModes are the user modes that we actually support (modifying).
|
2017-03-12 23:08:18 +01:00
|
|
|
SupportedUserModes = Modes{
|
2018-01-07 03:56:51 +01:00
|
|
|
Away, Bot, Invisible, Operator, RegisteredOnly, ServerNotice, UserRoleplaying,
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
2016-06-19 13:59:18 +02:00
|
|
|
// supportedUserModesString acts as a cache for when we introduce users
|
|
|
|
supportedUserModesString = SupportedUserModes.String()
|
2014-03-13 21:18:40 +01:00
|
|
|
)
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
// Channel Modes
|
2014-03-13 21:18:40 +01:00
|
|
|
const (
|
2017-03-12 23:08:18 +01:00
|
|
|
BanMask Mode = 'b' // arg
|
|
|
|
ChanRoleplaying Mode = 'E' // flag
|
|
|
|
ExceptMask Mode = 'e' // arg
|
|
|
|
InviteMask Mode = 'I' // arg
|
|
|
|
InviteOnly Mode = 'i' // flag
|
|
|
|
Key Mode = 'k' // flag arg
|
|
|
|
Moderated Mode = 'm' // flag
|
|
|
|
NoOutside Mode = 'n' // flag
|
|
|
|
OpOnlyTopic Mode = 't' // flag
|
2017-08-17 09:52:30 +02:00
|
|
|
// RegisteredOnly mode is reused here from umode definition
|
|
|
|
Secret Mode = 's' // flag
|
|
|
|
UserLimit Mode = 'l' // flag arg
|
2014-03-13 21:18:40 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2017-03-12 23:08:18 +01:00
|
|
|
ChannelFounder Mode = 'q' // arg
|
|
|
|
ChannelAdmin Mode = 'a' // arg
|
|
|
|
ChannelOperator Mode = 'o' // arg
|
|
|
|
Halfop Mode = 'h' // arg
|
|
|
|
Voice Mode = 'v' // arg
|
2016-10-22 16:45:51 +02:00
|
|
|
|
2017-06-19 22:53:16 +02:00
|
|
|
// SupportedChannelModes are the channel modes that we support.
|
2017-03-12 23:08:18 +01:00
|
|
|
SupportedChannelModes = Modes{
|
2017-10-01 07:14:32 +02:00
|
|
|
BanMask, ChanRoleplaying, ExceptMask, InviteMask, InviteOnly, Key,
|
|
|
|
Moderated, NoOutside, OpOnlyTopic, RegisteredOnly, Secret, UserLimit,
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
2016-06-19 13:59:18 +02:00
|
|
|
// supportedChannelModesString acts as a cache for when we introduce users
|
|
|
|
supportedChannelModesString = SupportedChannelModes.String()
|
2016-04-14 01:35:36 +02:00
|
|
|
|
2017-06-19 22:53:16 +02:00
|
|
|
// DefaultChannelModes are enabled on brand new channels when they're created.
|
2017-09-06 23:34:38 +02:00
|
|
|
// this can be overridden in the `channels` config, with the `default-modes` key
|
2017-03-12 23:08:18 +01:00
|
|
|
DefaultChannelModes = Modes{
|
2016-04-21 11:29:50 +02:00
|
|
|
NoOutside, OpOnlyTopic,
|
|
|
|
}
|
|
|
|
|
2016-04-14 01:35:36 +02:00
|
|
|
// ChannelPrivModes holds the list of modes that are privileged, ie founder/op/halfop, in order.
|
|
|
|
// voice is not in this list because it cannot perform channel operator actions.
|
2017-03-12 23:08:18 +01:00
|
|
|
ChannelPrivModes = Modes{
|
2016-04-14 01:35:36 +02:00
|
|
|
ChannelFounder, ChannelAdmin, ChannelOperator, Halfop,
|
|
|
|
}
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
ChannelModePrefixes = map[Mode]string{
|
2016-04-14 01:35:36 +02:00
|
|
|
ChannelFounder: "~",
|
|
|
|
ChannelAdmin: "&",
|
|
|
|
ChannelOperator: "@",
|
|
|
|
Halfop: "%",
|
|
|
|
Voice: "+",
|
|
|
|
}
|
2014-03-13 21:18:40 +01:00
|
|
|
)
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
//
|
|
|
|
// channel membership prefixes
|
|
|
|
//
|
|
|
|
|
2016-10-22 16:45:51 +02:00
|
|
|
// SplitChannelMembershipPrefixes takes a target and returns the prefixes on it, then the name.
|
|
|
|
func SplitChannelMembershipPrefixes(target string) (prefixes string, name string) {
|
|
|
|
name = target
|
|
|
|
for {
|
2016-11-04 12:38:47 +01:00
|
|
|
if len(name) > 0 && strings.Contains("~&@%+", string(name[0])) {
|
2016-10-22 16:45:51 +02:00
|
|
|
prefixes += string(name[0])
|
|
|
|
name = name[1:]
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return prefixes, name
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetLowestChannelModePrefix returns the lowest channel prefix mode out of the given prefixes.
|
2017-03-24 03:23:21 +01:00
|
|
|
func GetLowestChannelModePrefix(prefixes string) *Mode {
|
|
|
|
var lowest *Mode
|
2016-10-22 16:45:51 +02:00
|
|
|
|
|
|
|
if strings.Contains(prefixes, "+") {
|
|
|
|
lowest = &Voice
|
|
|
|
} else {
|
|
|
|
for i, mode := range ChannelPrivModes {
|
|
|
|
if strings.Contains(prefixes, ChannelModePrefixes[mode]) {
|
|
|
|
lowest = &ChannelPrivModes[i]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return lowest
|
|
|
|
}
|
|
|
|
|
2014-03-13 21:18:40 +01:00
|
|
|
//
|
|
|
|
// commands
|
|
|
|
//
|
|
|
|
|
2017-05-08 01:15:16 +02:00
|
|
|
// ParseUserModeChanges returns the valid changes, and the list of unknown chars.
|
|
|
|
func ParseUserModeChanges(params ...string) (ModeChanges, map[rune]bool) {
|
|
|
|
changes := make(ModeChanges, 0)
|
|
|
|
unknown := make(map[rune]bool)
|
|
|
|
|
2017-09-25 23:58:17 +02:00
|
|
|
op := List
|
|
|
|
|
2017-05-08 01:15:16 +02:00
|
|
|
if 0 < len(params) {
|
|
|
|
modeArg := params[0]
|
|
|
|
skipArgs := 1
|
|
|
|
|
|
|
|
for _, mode := range modeArg {
|
|
|
|
if mode == '-' || mode == '+' {
|
|
|
|
op = ModeOp(mode)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
change := ModeChange{
|
|
|
|
mode: Mode(mode),
|
|
|
|
op: op,
|
|
|
|
}
|
|
|
|
|
|
|
|
// put arg into modechange if needed
|
|
|
|
switch Mode(mode) {
|
|
|
|
case ServerNotice:
|
|
|
|
// always require arg
|
|
|
|
if len(params) > skipArgs {
|
|
|
|
change.arg = params[skipArgs]
|
|
|
|
skipArgs++
|
|
|
|
} else {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var isKnown bool
|
|
|
|
for _, supportedMode := range SupportedUserModes {
|
|
|
|
if rune(supportedMode) == mode {
|
|
|
|
isKnown = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !isKnown {
|
|
|
|
unknown[mode] = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
changes = append(changes, change)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return changes, unknown
|
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
// applyUserModeChanges applies the given changes, and returns the applied changes.
|
2017-05-08 01:15:16 +02:00
|
|
|
func (client *Client) applyUserModeChanges(force bool, changes ModeChanges) ModeChanges {
|
2017-03-12 23:08:18 +01:00
|
|
|
applied := make(ModeChanges, 0)
|
|
|
|
|
|
|
|
for _, change := range changes {
|
|
|
|
switch change.mode {
|
2018-01-07 03:56:51 +01:00
|
|
|
case Bot, Invisible, WallOps, UserRoleplaying, Operator, LocalOperator, RegisteredOnly:
|
2017-03-12 23:08:18 +01:00
|
|
|
switch change.op {
|
|
|
|
case Add:
|
2017-05-08 01:15:16 +02:00
|
|
|
if !force && (change.mode == Operator || change.mode == LocalOperator) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2017-03-12 23:08:18 +01:00
|
|
|
if client.flags[change.mode] {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
client.flags[change.mode] = true
|
|
|
|
applied = append(applied, change)
|
|
|
|
|
|
|
|
case Remove:
|
|
|
|
if !client.flags[change.mode] {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
delete(client.flags, change.mode)
|
|
|
|
applied = append(applied, change)
|
|
|
|
}
|
|
|
|
|
2017-05-08 01:15:16 +02:00
|
|
|
case ServerNotice:
|
|
|
|
if !client.flags[Operator] {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
var masks []sno.Mask
|
|
|
|
if change.op == Add || change.op == Remove {
|
|
|
|
for _, char := range change.arg {
|
|
|
|
masks = append(masks, sno.Mask(char))
|
2017-03-12 23:08:18 +01:00
|
|
|
}
|
2017-05-08 01:15:16 +02:00
|
|
|
}
|
|
|
|
if change.op == Add {
|
|
|
|
client.server.snomasks.AddMasks(client, masks...)
|
|
|
|
applied = append(applied, change)
|
|
|
|
} else if change.op == Remove {
|
|
|
|
client.server.snomasks.RemoveMasks(client, masks...)
|
2017-03-12 23:08:18 +01:00
|
|
|
applied = append(applied, change)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// can't do anything to TLS mode
|
|
|
|
}
|
|
|
|
|
|
|
|
// return the changes we could actually apply
|
|
|
|
return applied
|
2016-06-20 14:53:45 +02:00
|
|
|
}
|
|
|
|
|
2017-09-06 23:34:38 +02:00
|
|
|
// ParseDefaultChannelModes parses the `default-modes` line of the config
|
|
|
|
func ParseDefaultChannelModes(config *Config) Modes {
|
|
|
|
if config.Channels.DefaultModes == nil {
|
|
|
|
// not present in config, fall back to compile-time default
|
|
|
|
return DefaultChannelModes
|
|
|
|
}
|
|
|
|
modeChangeStrings := strings.Split(strings.TrimSpace(*config.Channels.DefaultModes), " ")
|
|
|
|
modeChanges, _ := ParseChannelModeChanges(modeChangeStrings...)
|
|
|
|
defaultChannelModes := make(Modes, 0)
|
|
|
|
for _, modeChange := range modeChanges {
|
|
|
|
if modeChange.op == Add {
|
|
|
|
defaultChannelModes = append(defaultChannelModes, modeChange.mode)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return defaultChannelModes
|
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
// ParseChannelModeChanges returns the valid changes, and the list of unknown chars.
|
|
|
|
func ParseChannelModeChanges(params ...string) (ModeChanges, map[rune]bool) {
|
|
|
|
changes := make(ModeChanges, 0)
|
|
|
|
unknown := make(map[rune]bool)
|
2016-06-22 13:35:26 +02:00
|
|
|
|
2017-09-25 23:58:17 +02:00
|
|
|
op := List
|
|
|
|
|
2017-03-26 12:37:13 +02:00
|
|
|
if 0 < len(params) {
|
2017-03-24 03:23:21 +01:00
|
|
|
modeArg := params[0]
|
|
|
|
skipArgs := 1
|
|
|
|
|
2016-06-22 13:35:26 +02:00
|
|
|
for _, mode := range modeArg {
|
|
|
|
if mode == '-' || mode == '+' {
|
|
|
|
op = ModeOp(mode)
|
|
|
|
continue
|
|
|
|
}
|
2017-03-24 03:23:21 +01:00
|
|
|
change := ModeChange{
|
|
|
|
mode: Mode(mode),
|
2016-06-22 13:35:26 +02:00
|
|
|
op: op,
|
|
|
|
}
|
|
|
|
|
|
|
|
// put arg into modechange if needed
|
2017-03-24 03:23:21 +01:00
|
|
|
switch Mode(mode) {
|
2016-06-22 13:35:26 +02:00
|
|
|
case BanMask, ExceptMask, InviteMask:
|
2017-03-24 03:23:21 +01:00
|
|
|
if len(params) > skipArgs {
|
|
|
|
change.arg = params[skipArgs]
|
2016-09-14 11:48:47 +02:00
|
|
|
skipArgs++
|
2016-06-22 13:35:26 +02:00
|
|
|
} else {
|
|
|
|
change.op = List
|
|
|
|
}
|
2016-09-14 11:48:47 +02:00
|
|
|
case ChannelFounder, ChannelAdmin, ChannelOperator, Halfop, Voice:
|
2017-03-24 03:23:21 +01:00
|
|
|
if len(params) > skipArgs {
|
|
|
|
change.arg = params[skipArgs]
|
2016-09-14 11:48:47 +02:00
|
|
|
skipArgs++
|
2016-06-22 13:35:26 +02:00
|
|
|
} else {
|
|
|
|
continue
|
|
|
|
}
|
2016-09-14 11:48:47 +02:00
|
|
|
case Key, UserLimit:
|
|
|
|
// don't require value when removing
|
|
|
|
if change.op == Add {
|
2017-03-24 03:23:21 +01:00
|
|
|
if len(params) > skipArgs {
|
|
|
|
change.arg = params[skipArgs]
|
2016-09-14 11:48:47 +02:00
|
|
|
skipArgs++
|
|
|
|
} else {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
2017-04-17 15:05:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var isKnown bool
|
|
|
|
for _, supportedMode := range SupportedChannelModes {
|
|
|
|
if rune(supportedMode) == mode {
|
|
|
|
isKnown = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2017-06-29 16:59:51 +02:00
|
|
|
for _, supportedMode := range ChannelPrivModes {
|
|
|
|
if rune(supportedMode) == mode {
|
|
|
|
isKnown = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2017-06-29 17:14:38 +02:00
|
|
|
if mode == rune(Voice) {
|
|
|
|
isKnown = true
|
|
|
|
}
|
2017-04-17 15:05:24 +02:00
|
|
|
if !isKnown {
|
2017-03-24 03:23:21 +01:00
|
|
|
unknown[mode] = true
|
2017-05-08 01:15:16 +02:00
|
|
|
continue
|
2016-06-22 13:35:26 +02:00
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
changes = append(changes, change)
|
2016-06-22 13:35:26 +02:00
|
|
|
}
|
2017-03-24 03:23:21 +01:00
|
|
|
}
|
2016-06-22 13:35:26 +02:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
return changes, unknown
|
|
|
|
}
|
2016-10-23 16:50:18 +02:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
// ApplyChannelModeChanges applies a given set of mode changes.
|
2017-10-23 01:50:16 +02:00
|
|
|
func (channel *Channel) ApplyChannelModeChanges(client *Client, isSamode bool, changes ModeChanges) ModeChanges {
|
2017-03-24 03:23:21 +01:00
|
|
|
// so we only output one warning for each list type when full
|
|
|
|
listFullWarned := make(map[Mode]bool)
|
2016-11-03 07:59:57 +01:00
|
|
|
|
2017-10-23 01:50:16 +02:00
|
|
|
clientIsOp := channel.ClientIsAtLeast(client, ChannelOperator)
|
2017-03-24 03:23:21 +01:00
|
|
|
var alreadySentPrivError bool
|
|
|
|
|
|
|
|
applied := make(ModeChanges, 0)
|
|
|
|
|
2017-11-13 08:55:26 +01:00
|
|
|
isListOp := func(change ModeChange) bool {
|
|
|
|
return (change.op == List) || (change.arg == "")
|
|
|
|
}
|
|
|
|
|
|
|
|
hasPrivs := func(change ModeChange) bool {
|
|
|
|
if isSamode {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
switch change.mode {
|
|
|
|
case ChannelFounder, ChannelAdmin, ChannelOperator, Halfop, Voice:
|
|
|
|
// Admins can't give other people Admin or remove it from others
|
|
|
|
if change.mode == ChannelAdmin {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if change.op == List {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
cfarg, _ := CasefoldName(change.arg)
|
|
|
|
if change.op == Remove && cfarg == client.nickCasefolded {
|
|
|
|
// "There is no restriction, however, on anyone `deopping' themselves"
|
|
|
|
// <https://tools.ietf.org/html/rfc2812#section-3.1.5>
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return channel.ClientIsAtLeast(client, change.mode)
|
|
|
|
case BanMask:
|
|
|
|
// #163: allow unprivileged users to list ban masks
|
|
|
|
return clientIsOp || isListOp(change)
|
|
|
|
default:
|
|
|
|
return clientIsOp
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
for _, change := range changes {
|
2017-11-13 08:55:26 +01:00
|
|
|
if !hasPrivs(change) {
|
2017-03-24 03:23:21 +01:00
|
|
|
if !alreadySentPrivError {
|
|
|
|
alreadySentPrivError = true
|
2018-01-22 12:26:01 +01:00
|
|
|
client.Send(nil, client.server.name, ERR_CHANOPRIVSNEEDED, channel.name, client.t("You're not a channel operator"))
|
2016-11-03 07:59:57 +01:00
|
|
|
}
|
2017-03-24 03:23:21 +01:00
|
|
|
continue
|
|
|
|
}
|
2016-11-03 07:59:57 +01:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
switch change.mode {
|
|
|
|
case BanMask, ExceptMask, InviteMask:
|
2017-11-13 08:55:26 +01:00
|
|
|
if isListOp(change) {
|
2017-03-24 03:23:21 +01:00
|
|
|
channel.ShowMaskList(client, change.mode)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// confirm mask looks valid
|
2017-11-13 08:55:26 +01:00
|
|
|
mask, err := Casefold(change.arg)
|
2017-03-24 03:23:21 +01:00
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
2016-06-22 13:35:26 +02:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
switch change.op {
|
|
|
|
case Add:
|
2017-11-03 07:36:55 +01:00
|
|
|
if channel.lists[change.mode].Length() >= client.server.Limits().ChanListModes {
|
2017-03-24 03:23:21 +01:00
|
|
|
if !listFullWarned[change.mode] {
|
2018-01-22 12:26:01 +01:00
|
|
|
client.Send(nil, client.server.name, ERR_BANLISTFULL, client.Nick(), channel.Name(), change.mode.String(), client.t("Channel list is full"))
|
2017-03-24 03:23:21 +01:00
|
|
|
listFullWarned[change.mode] = true
|
|
|
|
}
|
2016-10-11 15:51:46 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2017-10-23 01:50:16 +02:00
|
|
|
channel.lists[change.mode].Add(mask)
|
2017-03-24 03:23:21 +01:00
|
|
|
applied = append(applied, change)
|
2016-10-23 16:50:18 +02:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
case Remove:
|
2017-10-23 01:50:16 +02:00
|
|
|
channel.lists[change.mode].Remove(mask)
|
2017-03-24 03:23:21 +01:00
|
|
|
applied = append(applied, change)
|
|
|
|
}
|
2016-06-22 13:35:26 +02:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
case UserLimit:
|
|
|
|
switch change.op {
|
|
|
|
case Add:
|
|
|
|
val, err := strconv.ParseUint(change.arg, 10, 64)
|
|
|
|
if err == nil {
|
2017-10-23 01:50:16 +02:00
|
|
|
channel.setUserLimit(val)
|
2016-06-22 13:35:26 +02:00
|
|
|
applied = append(applied, change)
|
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
case Remove:
|
2017-10-23 01:50:16 +02:00
|
|
|
channel.setUserLimit(0)
|
2017-03-24 03:23:21 +01:00
|
|
|
applied = append(applied, change)
|
|
|
|
}
|
2016-09-14 11:48:47 +02:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
case Key:
|
|
|
|
switch change.op {
|
|
|
|
case Add:
|
2017-10-23 01:50:16 +02:00
|
|
|
channel.setKey(change.arg)
|
2016-09-14 11:48:47 +02:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
case Remove:
|
2017-10-23 01:50:16 +02:00
|
|
|
channel.setKey("")
|
2017-03-24 03:23:21 +01:00
|
|
|
}
|
|
|
|
applied = append(applied, change)
|
2016-09-14 11:48:47 +02:00
|
|
|
|
2017-03-28 09:32:03 +02:00
|
|
|
case InviteOnly, Moderated, NoOutside, OpOnlyTopic, RegisteredOnly, Secret, ChanRoleplaying:
|
2017-10-23 01:50:16 +02:00
|
|
|
if change.op == List {
|
|
|
|
continue
|
|
|
|
}
|
2016-09-14 11:48:47 +02:00
|
|
|
|
2017-10-23 01:50:16 +02:00
|
|
|
already := channel.setMode(change.mode, change.op == Add)
|
|
|
|
if !already {
|
2017-03-24 03:23:21 +01:00
|
|
|
applied = append(applied, change)
|
|
|
|
}
|
2016-06-22 13:35:26 +02:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
case ChannelFounder, ChannelAdmin, ChannelOperator, Halfop, Voice:
|
2017-10-23 01:50:16 +02:00
|
|
|
if change.op == List {
|
|
|
|
continue
|
|
|
|
}
|
2016-06-22 13:35:26 +02:00
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
change := channel.applyModeMemberNoMutex(client, change.mode, change.op, change.arg)
|
|
|
|
if change != nil {
|
|
|
|
applied = append(applied, *change)
|
2016-06-22 13:35:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-24 03:23:21 +01:00
|
|
|
return applied
|
|
|
|
}
|