2016-06-15 13:50:56 +02:00
|
|
|
// Copyright (c) 2012-2014 Jeremy Latt
|
|
|
|
// Copyright (c) 2014-2015 Edmund Huber
|
|
|
|
// Copyright (c) 2016- Daniel Oaks <daniel@danieloaks.net>
|
|
|
|
// released under the MIT license
|
|
|
|
|
2014-03-13 21:18:40 +01:00
|
|
|
package irc
|
|
|
|
|
|
|
|
import (
|
2016-06-28 08:22:35 +02:00
|
|
|
"fmt"
|
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
|
|
|
|
|
|
|
"github.com/DanielOaks/girc-go/ircmsg"
|
2014-03-13 21:18:40 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// user mode flags
|
|
|
|
type UserMode rune
|
|
|
|
|
|
|
|
func (mode UserMode) String() string {
|
|
|
|
return string(mode)
|
|
|
|
}
|
|
|
|
|
|
|
|
type UserModes []UserMode
|
|
|
|
|
|
|
|
func (modes UserModes) String() string {
|
|
|
|
strs := make([]string, len(modes))
|
|
|
|
for index, mode := range modes {
|
|
|
|
strs[index] = mode.String()
|
|
|
|
}
|
|
|
|
return strings.Join(strs, "")
|
|
|
|
}
|
|
|
|
|
2016-06-28 08:22:35 +02:00
|
|
|
type ModeChange struct {
|
|
|
|
mode UserMode
|
|
|
|
op ModeOp
|
|
|
|
}
|
|
|
|
|
|
|
|
func (change *ModeChange) String() string {
|
|
|
|
return fmt.Sprintf("%s%s", change.op, change.mode)
|
|
|
|
}
|
|
|
|
|
|
|
|
type ModeChanges []*ModeChange
|
|
|
|
|
|
|
|
func (changes ModeChanges) String() string {
|
|
|
|
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()
|
|
|
|
}
|
|
|
|
return str
|
|
|
|
}
|
|
|
|
|
2014-03-13 21:18:40 +01:00
|
|
|
// channel mode flags
|
|
|
|
type ChannelMode rune
|
|
|
|
|
|
|
|
func (mode ChannelMode) String() string {
|
|
|
|
return string(mode)
|
|
|
|
}
|
|
|
|
|
|
|
|
type ChannelModes []ChannelMode
|
|
|
|
|
|
|
|
func (modes ChannelModes) String() string {
|
|
|
|
strs := make([]string, len(modes))
|
|
|
|
for index, mode := range modes {
|
|
|
|
strs[index] = mode.String()
|
|
|
|
}
|
|
|
|
return strings.Join(strs, "")
|
|
|
|
}
|
|
|
|
|
2016-06-28 08:22:35 +02:00
|
|
|
type ChannelModeChange struct {
|
|
|
|
mode ChannelMode
|
|
|
|
op ModeOp
|
|
|
|
arg string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (change *ChannelModeChange) String() (str string) {
|
|
|
|
if (change.op == Add) || (change.op == Remove) {
|
|
|
|
str = change.op.String()
|
|
|
|
}
|
|
|
|
str += change.mode.String()
|
|
|
|
if change.arg != "" {
|
|
|
|
str += " " + change.arg
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
type ChannelModeChanges []*ChannelModeChange
|
|
|
|
|
|
|
|
func (changes ChannelModeChanges) String() string {
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
type ChannelModeCommand struct {
|
|
|
|
channel Name
|
|
|
|
changes ChannelModeChanges
|
|
|
|
}
|
|
|
|
|
2014-03-13 21:18:40 +01:00
|
|
|
type ModeOp rune
|
|
|
|
|
|
|
|
func (op ModeOp) String() string {
|
|
|
|
return string(op)
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
Add ModeOp = '+'
|
|
|
|
List ModeOp = '='
|
|
|
|
Remove ModeOp = '-'
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
Away UserMode = 'a'
|
|
|
|
Invisible UserMode = 'i'
|
|
|
|
LocalOperator UserMode = 'O'
|
|
|
|
Operator UserMode = 'o'
|
|
|
|
Restricted UserMode = 'r'
|
|
|
|
ServerNotice UserMode = 's' // deprecated
|
2016-06-28 17:09:07 +02:00
|
|
|
TLS UserMode = 'Z'
|
2014-03-13 21:18:40 +01:00
|
|
|
WallOps UserMode = 'w'
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
SupportedUserModes = UserModes{
|
|
|
|
Away, Invisible, Operator,
|
|
|
|
}
|
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
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2016-04-14 01:35:36 +02:00
|
|
|
ChannelFounder ChannelMode = 'q' // arg
|
|
|
|
ChannelAdmin ChannelMode = 'a' // arg
|
2014-03-13 21:18:40 +01:00
|
|
|
ChannelOperator ChannelMode = 'o' // arg
|
2016-04-14 01:35:36 +02:00
|
|
|
Halfop ChannelMode = 'h' // arg
|
2014-03-13 21:18:40 +01:00
|
|
|
Voice ChannelMode = 'v' // arg
|
2016-04-14 01:35:36 +02:00
|
|
|
|
|
|
|
BanMask ChannelMode = 'b' // arg
|
|
|
|
ExceptMask ChannelMode = 'e' // arg
|
|
|
|
InviteMask ChannelMode = 'I' // arg
|
|
|
|
InviteOnly ChannelMode = 'i' // flag
|
|
|
|
Key ChannelMode = 'k' // flag arg
|
|
|
|
Moderated ChannelMode = 'm' // flag
|
|
|
|
NoOutside ChannelMode = 'n' // flag
|
|
|
|
OpOnlyTopic ChannelMode = 't' // flag
|
|
|
|
Persistent ChannelMode = 'P' // flag
|
|
|
|
ReOp ChannelMode = 'r' // flag
|
2016-04-14 14:33:38 +02:00
|
|
|
Secret ChannelMode = 's' // flag
|
2016-04-14 01:35:36 +02:00
|
|
|
UserLimit ChannelMode = 'l' // flag arg
|
2014-03-13 21:18:40 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
SupportedChannelModes = ChannelModes{
|
|
|
|
BanMask, ExceptMask, InviteMask, InviteOnly, Key, NoOutside,
|
2016-09-12 04:31:25 +02:00
|
|
|
OpOnlyTopic, Persistent, 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
|
|
|
|
2016-04-21 11:29:50 +02:00
|
|
|
DefaultChannelModes = ChannelModes{
|
|
|
|
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.
|
|
|
|
ChannelPrivModes = ChannelModes{
|
|
|
|
ChannelFounder, ChannelAdmin, ChannelOperator, Halfop,
|
|
|
|
}
|
|
|
|
|
|
|
|
ChannelModePrefixes = map[ChannelMode]string{
|
|
|
|
ChannelFounder: "~",
|
|
|
|
ChannelAdmin: "&",
|
|
|
|
ChannelOperator: "@",
|
|
|
|
Halfop: "%",
|
|
|
|
Voice: "+",
|
|
|
|
}
|
2014-03-13 21:18:40 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
//
|
|
|
|
// commands
|
|
|
|
//
|
|
|
|
|
2016-06-20 14:53:45 +02:00
|
|
|
// MODE <target> [<modestring> [<mode arguments>...]]
|
|
|
|
func modeHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
|
|
|
|
name := NewName(msg.Params[0])
|
|
|
|
if name.IsChannel() {
|
2016-06-22 13:35:26 +02:00
|
|
|
return cmodeHandler(server, client, msg)
|
2016-06-20 14:53:45 +02:00
|
|
|
} else {
|
|
|
|
return umodeHandler(server, client, msg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MODE <target> [<modestring> [<mode arguments>...]]
|
|
|
|
func umodeHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
|
|
|
|
nickname := NewName(msg.Params[0])
|
|
|
|
|
|
|
|
target := server.clients.Get(nickname)
|
2014-03-13 21:18:40 +01:00
|
|
|
|
|
|
|
if target == nil {
|
2016-06-20 14:53:45 +02:00
|
|
|
client.Send(nil, server.nameString, ERR_NOSUCHNICK, client.nickString, msg.Params[0], "No such nick")
|
|
|
|
return false
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
|
|
|
|
2016-06-20 14:53:45 +02:00
|
|
|
//TODO(dan): restricting to Operator here should be done with SAMODE only
|
|
|
|
// point SAMODE at this handler too, if they are operator and SAMODE was called then fine
|
2014-03-13 21:18:40 +01:00
|
|
|
if client != target && !client.flags[Operator] {
|
2016-06-20 14:53:45 +02:00
|
|
|
if len(msg.Params) > 1 {
|
|
|
|
client.Send(nil, server.nameString, ERR_USERSDONTMATCH, client.nickString, "Can't change modes for other users")
|
|
|
|
} else {
|
|
|
|
client.Send(nil, server.nameString, ERR_USERSDONTMATCH, client.nickString, "Can't view modes for other users")
|
|
|
|
}
|
|
|
|
return false
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
|
|
|
|
2016-06-20 14:53:45 +02:00
|
|
|
// assemble changes
|
|
|
|
changes := make(ModeChanges, 0)
|
2016-06-22 13:35:26 +02:00
|
|
|
applied := make(ModeChanges, 0)
|
2016-06-20 14:53:45 +02:00
|
|
|
|
|
|
|
if len(msg.Params) > 1 {
|
2016-06-22 13:35:26 +02:00
|
|
|
modeArg := msg.Params[1]
|
2016-06-20 14:53:45 +02:00
|
|
|
op := ModeOp(modeArg[0])
|
|
|
|
if (op == Add) || (op == Remove) {
|
|
|
|
modeArg = modeArg[1:]
|
|
|
|
} else {
|
2016-06-22 13:35:26 +02:00
|
|
|
client.Send(nil, server.nameString, ERR_UNKNOWNMODE, client.nickString, string(modeArg[1]), "is an unknown mode character to me")
|
2016-06-20 14:53:45 +02:00
|
|
|
return false
|
|
|
|
}
|
2014-03-13 21:18:40 +01:00
|
|
|
|
2016-06-20 14:53:45 +02:00
|
|
|
for _, mode := range modeArg {
|
|
|
|
if mode == '-' || mode == '+' {
|
|
|
|
op = ModeOp(mode)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
changes = append(changes, &ModeChange{
|
|
|
|
mode: UserMode(mode),
|
|
|
|
op: op,
|
|
|
|
})
|
|
|
|
}
|
2014-03-13 21:18:40 +01:00
|
|
|
|
2016-06-20 14:53:45 +02:00
|
|
|
for _, change := range changes {
|
|
|
|
switch change.mode {
|
|
|
|
case Invisible, ServerNotice, WallOps:
|
|
|
|
switch change.op {
|
|
|
|
case Add:
|
|
|
|
if target.flags[change.mode] {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
target.flags[change.mode] = true
|
2016-06-22 13:35:26 +02:00
|
|
|
applied = append(applied, change)
|
2016-06-20 14:53:45 +02:00
|
|
|
|
|
|
|
case Remove:
|
|
|
|
if !target.flags[change.mode] {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
delete(target.flags, change.mode)
|
2016-06-22 13:35:26 +02:00
|
|
|
applied = append(applied, change)
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
|
|
|
|
2016-06-20 14:53:45 +02:00
|
|
|
case Operator, LocalOperator:
|
|
|
|
if change.op == Remove {
|
|
|
|
if !target.flags[change.mode] {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
delete(target.flags, change.mode)
|
2016-06-22 13:35:26 +02:00
|
|
|
applied = append(applied, change)
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
|
|
|
}
|
2016-06-28 17:09:07 +02:00
|
|
|
|
|
|
|
// can't do anything to TLS mode
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-28 17:09:07 +02:00
|
|
|
if len(applied) > 0 {
|
|
|
|
client.Send(nil, client.nickMaskString, "MODE", target.nickString, applied.String())
|
2014-03-22 22:15:52 +01:00
|
|
|
} else if client == target {
|
2016-06-20 14:53:45 +02:00
|
|
|
client.Send(nil, target.nickMaskString, RPL_UMODEIS, target.nickString, target.ModeString())
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
2016-06-20 14:53:45 +02:00
|
|
|
return false
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
|
|
|
|
2016-06-22 13:35:26 +02:00
|
|
|
// MODE <target> [<modestring> [<mode arguments>...]]
|
|
|
|
func cmodeHandler(server *Server, client *Client, msg ircmsg.IrcMessage) bool {
|
|
|
|
channelName := NewName(msg.Params[0])
|
|
|
|
channel := server.channels.Get(channelName)
|
|
|
|
|
2014-03-13 21:18:40 +01:00
|
|
|
if channel == nil {
|
2016-06-22 13:35:26 +02:00
|
|
|
client.Send(nil, server.nameString, ERR_NOSUCHCHANNEL, client.nickString, msg.Params[0], "No such channel")
|
|
|
|
return false
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|
|
|
|
|
2016-06-22 13:35:26 +02:00
|
|
|
// assemble changes
|
|
|
|
//TODO(dan): split out assembling changes into func that returns changes, err
|
|
|
|
changes := make(ChannelModeChanges, 0)
|
|
|
|
applied := make(ChannelModeChanges, 0)
|
|
|
|
|
2016-06-28 08:22:35 +02:00
|
|
|
// TODO(dan): look at separating these into the type A/B/C/D args and using those lists here
|
2016-06-22 13:35:26 +02:00
|
|
|
if len(msg.Params) > 1 {
|
|
|
|
modeArg := msg.Params[1]
|
|
|
|
op := ModeOp(modeArg[0])
|
|
|
|
if (op == Add) || (op == Remove) {
|
|
|
|
modeArg = modeArg[1:]
|
|
|
|
} else {
|
|
|
|
client.Send(nil, server.nameString, ERR_UNKNOWNMODE, client.nickString, string(modeArg[1]), "is an unknown mode character to me")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
skipArgs := 2
|
|
|
|
for _, mode := range modeArg {
|
|
|
|
if mode == '-' || mode == '+' {
|
|
|
|
op = ModeOp(mode)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
change := ChannelModeChange{
|
|
|
|
mode: ChannelMode(mode),
|
|
|
|
op: op,
|
|
|
|
}
|
|
|
|
|
|
|
|
// put arg into modechange if needed
|
|
|
|
switch ChannelMode(mode) {
|
|
|
|
case BanMask, ExceptMask, InviteMask:
|
|
|
|
if len(msg.Params) > skipArgs {
|
|
|
|
change.arg = msg.Params[skipArgs]
|
|
|
|
skipArgs += 1
|
|
|
|
} else {
|
|
|
|
change.op = List
|
|
|
|
}
|
|
|
|
case Key, UserLimit, ChannelFounder, ChannelAdmin, ChannelOperator, Halfop, Voice:
|
|
|
|
if len(msg.Params) > skipArgs {
|
|
|
|
change.arg = msg.Params[skipArgs]
|
|
|
|
skipArgs += 1
|
|
|
|
} else {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
applied = append(applied, &change)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, change := range changes {
|
|
|
|
switch change.mode {
|
|
|
|
case BanMask, ExceptMask, InviteMask:
|
|
|
|
mask := change.arg
|
|
|
|
list := channel.lists[change.mode]
|
|
|
|
if list == nil {
|
|
|
|
// This should never happen, but better safe than panicky.
|
|
|
|
client.Send(nil, server.nameString, ERR_UNKNOWNERROR, client.nickString, "MODE", "Could not complete MODE command")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if (change.op == List) || (mask == "") {
|
|
|
|
channel.ShowMaskList(client, change.mode)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
switch change.op {
|
|
|
|
case Add:
|
|
|
|
list.Add(Name(mask))
|
|
|
|
applied = append(applied, change)
|
|
|
|
|
|
|
|
case Remove:
|
|
|
|
list.Remove(Name(mask))
|
|
|
|
applied = append(applied, change)
|
|
|
|
}
|
|
|
|
|
|
|
|
case InviteOnly, Moderated, NoOutside, OpOnlyTopic, Persistent, Secret:
|
|
|
|
switch change.op {
|
|
|
|
case Add:
|
|
|
|
if channel.flags[change.mode] {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
channel.flags[change.mode] = true
|
|
|
|
applied = append(applied, change)
|
|
|
|
|
|
|
|
case Remove:
|
|
|
|
if !channel.flags[change.mode] {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
delete(channel.flags, change.mode)
|
|
|
|
applied = append(applied, change)
|
|
|
|
}
|
|
|
|
|
|
|
|
case ChannelFounder, ChannelAdmin, ChannelOperator, Halfop, Voice:
|
|
|
|
// make sure client has privs to edit the given prefix
|
|
|
|
var hasPrivs bool
|
|
|
|
|
|
|
|
for _, mode := range ChannelPrivModes {
|
|
|
|
if channel.members[client][mode] {
|
|
|
|
hasPrivs = true
|
|
|
|
|
|
|
|
// Admins can't give other people Admin or remove it from others,
|
|
|
|
// standard for that channel mode, we worry about this later
|
|
|
|
if mode == ChannelAdmin && change.mode == ChannelAdmin {
|
|
|
|
hasPrivs = false
|
|
|
|
}
|
|
|
|
|
|
|
|
break
|
|
|
|
} else if mode == change.mode {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
name := NewName(change.arg)
|
|
|
|
|
|
|
|
if !hasPrivs {
|
|
|
|
if change.op == Remove && name.ToLower() == client.nick.ToLower() {
|
|
|
|
// success!
|
|
|
|
} else {
|
|
|
|
client.Send(nil, client.server.nameString, ERR_CHANOPRIVSNEEDED, channel.nameString, "You're not a channel operator")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
change := channel.applyModeMember(client, change.mode, change.op, change.arg)
|
|
|
|
if change != nil {
|
|
|
|
applied = append(changes, change)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(applied) > 0 {
|
|
|
|
//TODO(dan): we should change the name of String and make it return a slice here
|
|
|
|
args := append([]string{channel.nameString}, strings.Split(applied.String(), " ")...)
|
|
|
|
client.Send(nil, client.nickMaskString, "MODE", args...)
|
|
|
|
} else {
|
|
|
|
//TODO(dan): we should just make ModeString return a slice here
|
|
|
|
args := append([]string{client.nickString, channel.nameString}, strings.Split(channel.ModeString(client), " ")...)
|
|
|
|
client.Send(nil, client.nickMaskString, RPL_CHANNELMODEIS, args...)
|
2016-06-26 13:06:28 +02:00
|
|
|
client.Send(nil, client.nickMaskString, RPL_CHANNELCREATED, client.nickString, channel.nameString, strconv.FormatInt(channel.createdTime.Unix(), 10))
|
2016-06-22 13:35:26 +02:00
|
|
|
}
|
|
|
|
return false
|
2014-03-13 21:18:40 +01:00
|
|
|
}
|