dispatch/server/irc_handler.go

469 lines
10 KiB
Go
Raw Normal View History

package server
2015-01-17 01:37:21 +00:00
import (
"fmt"
"log"
2019-01-23 06:34:39 +00:00
"strconv"
2015-01-17 01:37:21 +00:00
"strings"
"unicode"
2015-01-17 01:37:21 +00:00
2017-04-20 03:32:22 +00:00
"github.com/kjk/betterguid"
"github.com/khlieng/dispatch/pkg/irc"
2015-12-11 03:35:48 +00:00
"github.com/khlieng/dispatch/storage"
2015-01-17 01:37:21 +00:00
)
var excludedErrors = []string{
irc.ErrNicknameInUse,
irc.ErrForward,
}
2015-06-16 21:20:53 +00:00
type ircHandler struct {
client *irc.Client
state *State
2015-01-17 01:37:21 +00:00
2015-06-16 21:20:53 +00:00
whois WhoisReply
userBuffers map[string][]string
motdBuffer MOTD
2019-01-23 06:34:39 +00:00
listBuffer storage.ChannelListIndex
2015-06-16 21:20:53 +00:00
handlers map[string]func(*irc.Message)
}
func newIRCHandler(client *irc.Client, state *State) *ircHandler {
2015-06-16 21:20:53 +00:00
i := &ircHandler{
client: client,
state: state,
2015-06-16 21:20:53 +00:00
userBuffers: make(map[string][]string),
}
i.initHandlers()
return i
}
func (i *ircHandler) run() {
var lastConnErr error
for {
2016-01-13 17:53:54 +00:00
select {
case msg, ok := <-i.client.Messages:
if !ok {
i.state.deleteIRC(i.client.Host)
2016-01-13 17:53:54 +00:00
return
}
i.dispatchMessage(msg)
case state := <-i.client.ConnectionChanged:
i.state.sendJSON("connection_update", newConnectionUpdate(i.client.Host, state))
i.state.setConnectionState(i.client.Host, state)
if state.Error != nil && (lastConnErr == nil ||
state.Error.Error() != lastConnErr.Error()) {
lastConnErr = state.Error
i.log("Connection error:", state.Error)
} else if state.Connected {
i.log("Connected")
}
2016-01-13 17:53:54 +00:00
}
2015-06-16 21:20:53 +00:00
}
}
func (i *ircHandler) dispatchMessage(msg *irc.Message) {
if msg.Command[0] == '4' && !isExcludedError(msg.Command) {
err := IRCError{
Server: i.client.Host,
Message: msg.LastParam(),
}
if len(msg.Params) > 2 {
for i := 1; i < len(msg.Params); i++ {
if isChannel(msg.Params[i]) {
err.Target = msg.Params[i]
break
}
}
}
i.state.sendJSON("error", err)
}
2015-06-16 21:20:53 +00:00
if handler, ok := i.handlers[msg.Command]; ok {
handler(msg)
}
}
func (i *ircHandler) nick(msg *irc.Message) {
i.state.sendJSON("nick", Nick{
Server: i.client.Host,
Old: msg.Nick,
New: msg.LastParam(),
2015-06-16 21:20:53 +00:00
})
2016-12-16 23:11:44 +00:00
channelStore.RenameUser(msg.Nick, msg.LastParam(), i.client.Host)
2017-04-11 04:04:59 +00:00
if msg.LastParam() == i.client.GetNick() {
go i.state.user.SetNick(msg.LastParam(), i.client.Host)
2017-04-11 04:04:59 +00:00
}
2015-06-16 21:20:53 +00:00
}
func (i *ircHandler) join(msg *irc.Message) {
i.state.sendJSON("join", Join{
2015-06-16 21:20:53 +00:00
Server: i.client.Host,
User: msg.Nick,
Channels: msg.Params,
})
channel := msg.Params[0]
channelStore.AddUser(msg.Nick, i.client.Host, channel)
2015-06-16 21:20:53 +00:00
if msg.Nick == i.client.GetNick() {
2018-06-18 13:47:51 +00:00
// In case no topic is set and there's a cached one that needs to be cleared
i.client.Topic(channel)
i.state.sendLastMessages(i.client.Host, channel, 50)
go i.state.user.AddChannel(&storage.Channel{
2015-06-16 21:20:53 +00:00
Server: i.client.Host,
Name: channel,
2015-06-16 21:20:53 +00:00
})
}
}
func (i *ircHandler) part(msg *irc.Message) {
part := Part{
Server: i.client.Host,
User: msg.Nick,
Channel: msg.Params[0],
}
if len(msg.Params) == 2 {
part.Reason = msg.Params[1]
}
i.state.sendJSON("part", part)
2015-06-16 21:20:53 +00:00
channelStore.RemoveUser(msg.Nick, i.client.Host, part.Channel)
2015-06-16 21:20:53 +00:00
if msg.Nick == i.client.GetNick() {
go i.state.user.RemoveChannel(i.client.Host, part.Channel)
2015-06-16 21:20:53 +00:00
}
}
func (i *ircHandler) mode(msg *irc.Message) {
target := msg.Params[0]
if len(msg.Params) > 2 && isChannel(target) {
mode := parseMode(msg.Params[1])
mode.Server = i.client.Host
mode.Channel = target
mode.User = msg.Params[2]
i.state.sendJSON("mode", mode)
2015-06-16 21:20:53 +00:00
channelStore.SetMode(i.client.Host, target, msg.Params[2], mode.Add, mode.Remove)
}
}
func (i *ircHandler) message(msg *irc.Message) {
message := Message{
2017-04-20 03:32:22 +00:00
ID: betterguid.New(),
2015-06-16 21:20:53 +00:00
Server: i.client.Host,
From: msg.Nick,
Content: msg.LastParam(),
2015-06-16 21:20:53 +00:00
}
target := msg.Params[0]
2015-06-16 21:20:53 +00:00
if target == i.client.GetNick() {
i.state.sendJSON("pm", message)
i.state.user.AddOpenDM(i.client.Host, message.From)
target = message.From
2015-06-16 21:20:53 +00:00
} else {
message.To = target
i.state.sendJSON("message", message)
2015-06-16 21:20:53 +00:00
}
if target != "*" {
go i.state.user.LogMessage(message.ID,
i.client.Host, msg.Nick, target, msg.LastParam())
2015-06-16 21:20:53 +00:00
}
}
func (i *ircHandler) quit(msg *irc.Message) {
i.state.sendJSON("quit", Quit{
Server: i.client.Host,
User: msg.Nick,
Reason: msg.LastParam(),
2015-06-16 21:20:53 +00:00
})
channelStore.RemoveUserAll(msg.Nick, i.client.Host)
}
func (i *ircHandler) info(msg *irc.Message) {
if msg.Command == irc.ReplyWelcome {
i.state.sendJSON("nick", Nick{
Server: i.client.Host,
New: msg.Params[0],
})
2019-01-23 06:34:39 +00:00
_, needsUpdate := channelIndexes.Get(i.client.Host)
if needsUpdate {
i.listBuffer = storage.NewMapChannelListIndex()
i.client.List()
}
go i.state.user.SetNick(msg.Params[0], i.client.Host)
}
i.state.sendJSON("pm", Message{
2015-06-16 21:20:53 +00:00
Server: i.client.Host,
From: msg.Nick,
Content: strings.Join(msg.Params[1:], " "),
2015-06-16 21:20:53 +00:00
})
}
2019-01-27 07:53:07 +00:00
func (i *ircHandler) features(msg *irc.Message) {
i.state.sendJSON("features", Features{
Server: i.client.Host,
Features: i.client.Features.Map(),
})
if name := i.client.Features.String("NETWORK"); name != "" {
go func() {
server, err := i.state.user.GetServer(i.client.Host)
if err == nil && server.Name == "" {
i.state.user.SetServerName(name, server.Host)
}
}()
}
}
2015-06-16 21:20:53 +00:00
func (i *ircHandler) whoisUser(msg *irc.Message) {
i.whois.Nick = msg.Params[1]
i.whois.Username = msg.Params[2]
i.whois.Host = msg.Params[3]
i.whois.Realname = msg.Params[5]
}
func (i *ircHandler) whoisServer(msg *irc.Message) {
i.whois.Server = msg.Params[2]
}
func (i *ircHandler) whoisChannels(msg *irc.Message) {
2016-12-16 23:11:44 +00:00
i.whois.Channels = append(i.whois.Channels, strings.Split(strings.TrimRight(msg.LastParam(), " "), " ")...)
2015-06-16 21:20:53 +00:00
}
func (i *ircHandler) whoisEnd(msg *irc.Message) {
if i.whois.Nick != "" {
i.state.sendJSON("whois", i.whois)
}
2015-06-16 21:20:53 +00:00
i.whois = WhoisReply{}
}
func (i *ircHandler) topic(msg *irc.Message) {
var channel string
var nick string
if msg.Command == irc.Topic {
channel = msg.Params[0]
nick = msg.Nick
} else {
channel = msg.Params[1]
}
i.state.sendJSON("topic", Topic{
2015-06-16 21:20:53 +00:00
Server: i.client.Host,
Channel: channel,
2016-12-16 23:11:44 +00:00
Topic: msg.LastParam(),
Nick: nick,
})
channelStore.SetTopic(msg.LastParam(), i.client.Host, channel)
}
func (i *ircHandler) noTopic(msg *irc.Message) {
channel := msg.Params[1]
i.state.sendJSON("topic", Topic{
Server: i.client.Host,
Channel: channel,
2015-06-16 21:20:53 +00:00
})
channelStore.SetTopic("", i.client.Host, channel)
2015-06-16 21:20:53 +00:00
}
func (i *ircHandler) names(msg *irc.Message) {
2016-12-16 23:11:44 +00:00
users := strings.Split(strings.TrimSuffix(msg.LastParam(), " "), " ")
2015-06-16 21:20:53 +00:00
userBuffer := i.userBuffers[msg.Params[2]]
i.userBuffers[msg.Params[2]] = append(userBuffer, users...)
}
func (i *ircHandler) namesEnd(msg *irc.Message) {
channel := msg.Params[1]
users := i.userBuffers[channel]
i.state.sendJSON("users", Userlist{
2015-06-16 21:20:53 +00:00
Server: i.client.Host,
Channel: channel,
Users: users,
})
channelStore.SetUsers(users, i.client.Host, channel)
delete(i.userBuffers, channel)
}
func (i *ircHandler) motdStart(msg *irc.Message) {
i.motdBuffer.Server = i.client.Host
2016-12-16 23:11:44 +00:00
i.motdBuffer.Title = msg.LastParam()
2015-06-16 21:20:53 +00:00
}
func (i *ircHandler) motd(msg *irc.Message) {
2016-12-16 23:11:44 +00:00
i.motdBuffer.Content = append(i.motdBuffer.Content, msg.LastParam())
2015-06-16 21:20:53 +00:00
}
func (i *ircHandler) motdEnd(msg *irc.Message) {
i.state.sendJSON("motd", i.motdBuffer)
2015-06-16 21:20:53 +00:00
i.motdBuffer = MOTD{}
}
2019-01-23 06:34:39 +00:00
func (i *ircHandler) list(msg *irc.Message) {
if i.listBuffer == nil && i.state.Bool("update_chanlist_"+i.client.Host) {
i.listBuffer = storage.NewMapChannelListIndex()
}
if i.listBuffer != nil {
c, _ := strconv.Atoi(msg.Params[2])
i.listBuffer.Add(&storage.ChannelListItem{
Name: msg.Params[1],
UserCount: c,
Topic: msg.LastParam(),
})
}
}
func (i *ircHandler) listEnd(msg *irc.Message) {
if i.listBuffer != nil {
i.state.Set("update_chanlist_"+i.client.Host, false)
go func(idx storage.ChannelListIndex) {
idx.Finish()
channelIndexes.Set(i.client.Host, idx)
}(i.listBuffer)
i.listBuffer = nil
}
}
func (i *ircHandler) badNick(msg *irc.Message) {
i.state.sendJSON("nick_fail", NickFail{
Server: i.client.Host,
})
}
func (i *ircHandler) forward(msg *irc.Message) {
if len(msg.Params) > 2 {
i.state.sendJSON("channel_forward", ChannelForward{
Server: i.client.Host,
Old: msg.Params[1],
New: msg.Params[2],
})
}
}
2017-07-03 05:25:38 +00:00
func (i *ircHandler) error(msg *irc.Message) {
i.state.sendJSON("error", IRCError{
Server: i.client.Host,
Message: msg.LastParam(),
})
2017-07-03 05:25:38 +00:00
}
2015-06-16 21:20:53 +00:00
func (i *ircHandler) initHandlers() {
i.handlers = map[string]func(*irc.Message){
irc.Nick: i.nick,
irc.Join: i.join,
irc.Part: i.part,
irc.Mode: i.mode,
irc.Privmsg: i.message,
irc.Notice: i.message,
irc.Quit: i.quit,
irc.Topic: i.topic,
2017-07-03 05:25:38 +00:00
irc.Error: i.error,
irc.ReplyWelcome: i.info,
irc.ReplyYourHost: i.info,
irc.ReplyCreated: i.info,
2019-01-27 07:53:07 +00:00
irc.ReplyISupport: i.features,
irc.ReplyLUserClient: i.info,
irc.ReplyLUserOp: i.info,
irc.ReplyLUserUnknown: i.info,
irc.ReplyLUserChannels: i.info,
irc.ReplyLUserMe: i.info,
irc.ReplyWhoisUser: i.whoisUser,
irc.ReplyWhoisServer: i.whoisServer,
irc.ReplyWhoisChannels: i.whoisChannels,
irc.ReplyEndOfWhois: i.whoisEnd,
irc.ReplyNoTopic: i.noTopic,
irc.ReplyTopic: i.topic,
irc.ReplyNamReply: i.names,
irc.ReplyEndOfNames: i.namesEnd,
irc.ReplyMotdStart: i.motdStart,
irc.ReplyMotd: i.motd,
irc.ReplyEndOfMotd: i.motdEnd,
2019-01-23 06:34:39 +00:00
irc.ReplyList: i.list,
irc.ReplyListEnd: i.listEnd,
irc.ErrErroneousNickname: i.badNick,
irc.ErrForward: i.forward,
2015-01-17 01:37:21 +00:00
}
}
func (i *ircHandler) log(v ...interface{}) {
s := fmt.Sprintln(v...)
log.Println("[IRC]", i.state.user.ID, i.client.Host, s[:len(s)-1])
}
func parseMode(mode string) *Mode {
m := Mode{}
add := false
for _, c := range mode {
if c == '+' {
add = true
} else if c == '-' {
add = false
} else if add {
m.Add += string(c)
} else {
m.Remove += string(c)
}
}
return &m
}
func isChannel(s string) bool {
return strings.IndexAny(s, "&#+!") == 0
}
func isExcludedError(cmd string) bool {
for _, err := range excludedErrors {
if cmd == err {
return true
}
}
return false
}
func formatIRCError(msg *irc.Message) string {
errMsg := strings.TrimSuffix(msg.LastParam(), ".")
if len(msg.Params) > 2 {
for _, c := range msg.LastParam() {
if unicode.IsLower(c) {
return msg.Params[1] + " " + errMsg
}
return msg.Params[1] + ": " + errMsg
}
}
return errMsg
}
2015-06-05 22:34:13 +00:00
func printMessage(msg *irc.Message, i *irc.Client) {
log.Println(i.GetNick()+":", msg.Prefix, msg.Command, msg.Params)
2015-01-17 01:37:21 +00:00
}