package server import ( "fmt" "log" "strings" "unicode" "github.com/kjk/betterguid" "github.com/khlieng/dispatch/irc" "github.com/khlieng/dispatch/storage" ) var excludedErrors = []string{ irc.ErrNicknameInUse, } type ircHandler struct { client *irc.Client session *Session whois WhoisReply userBuffers map[string][]string motdBuffer MOTD handlers map[string]func(*irc.Message) } func newIRCHandler(client *irc.Client, session *Session) *ircHandler { i := &ircHandler{ client: client, session: session, userBuffers: make(map[string][]string), } i.initHandlers() return i } func (i *ircHandler) run() { var lastConnErr error for { select { case msg, ok := <-i.client.Messages: if !ok { i.session.deleteIRC(i.client.Host) return } i.dispatchMessage(msg) case state := <-i.client.ConnectionChanged: i.session.sendJSON("connection_update", newConnectionUpdate(i.client.Host, state)) i.session.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") } } } } func (i *ircHandler) dispatchMessage(msg *irc.Message) { if msg.Command[0] == '4' && !isExcludedError(msg.Command) { i.session.printError(formatIRCError(msg)) } if handler, ok := i.handlers[msg.Command]; ok { handler(msg) } } func (i *ircHandler) nick(msg *irc.Message) { i.session.sendJSON("nick", Nick{ Server: i.client.Host, Old: msg.Nick, New: msg.LastParam(), }) channelStore.RenameUser(msg.Nick, msg.LastParam(), i.client.Host) if msg.LastParam() == i.client.GetNick() { go i.session.user.SetNick(msg.LastParam(), i.client.Host) } } func (i *ircHandler) join(msg *irc.Message) { i.session.sendJSON("join", Join{ Server: i.client.Host, User: msg.Nick, Channels: msg.Params, }) channel := msg.Params[0] channelStore.AddUser(msg.Nick, i.client.Host, channel) if msg.Nick == i.client.GetNick() { // Incase no topic is set and theres a cached one that needs to be cleared i.client.Topic(channel) i.session.sendLastMessages(i.client.Host, channel, 50) go i.session.user.AddChannel(storage.Channel{ Server: i.client.Host, Name: channel, }) } } 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.session.sendJSON("part", part) channelStore.RemoveUser(msg.Nick, i.client.Host, part.Channel) if msg.Nick == i.client.GetNick() { go i.session.user.RemoveChannel(i.client.Host, part.Channel) } } 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.session.sendJSON("mode", mode) channelStore.SetMode(i.client.Host, target, msg.Params[2], mode.Add, mode.Remove) } } func (i *ircHandler) message(msg *irc.Message) { message := Message{ ID: betterguid.New(), Server: i.client.Host, From: msg.Nick, Content: msg.LastParam(), } if msg.Params[0] == i.client.GetNick() { i.session.sendJSON("pm", message) } else { message.To = msg.Params[0] i.session.sendJSON("message", message) } if msg.Params[0] != "*" { go i.session.user.LogMessage(message.ID, i.client.Host, msg.Nick, msg.Params[0], msg.LastParam()) } } func (i *ircHandler) quit(msg *irc.Message) { i.session.sendJSON("quit", Quit{ Server: i.client.Host, User: msg.Nick, Reason: msg.LastParam(), }) channelStore.RemoveUserAll(msg.Nick, i.client.Host) } func (i *ircHandler) info(msg *irc.Message) { if msg.Command == irc.ReplyWelcome { i.session.sendJSON("nick", Nick{ Server: i.client.Host, New: msg.Params[0], }) go i.session.user.SetNick(msg.Params[0], i.client.Host) } i.session.sendJSON("pm", Message{ Server: i.client.Host, From: msg.Nick, Content: strings.Join(msg.Params[1:], " "), }) } 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) { i.whois.Channels = append(i.whois.Channels, strings.Split(strings.TrimRight(msg.LastParam(), " "), " ")...) } func (i *ircHandler) whoisEnd(msg *irc.Message) { if i.whois.Nick != "" { i.session.sendJSON("whois", i.whois) } 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.session.sendJSON("topic", Topic{ Server: i.client.Host, Channel: channel, 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.session.sendJSON("topic", Topic{ Server: i.client.Host, Channel: channel, }) channelStore.SetTopic("", i.client.Host, channel) } func (i *ircHandler) names(msg *irc.Message) { users := strings.Split(strings.TrimSuffix(msg.LastParam(), " "), " ") 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.session.sendJSON("users", Userlist{ 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 i.motdBuffer.Title = msg.LastParam() } func (i *ircHandler) motd(msg *irc.Message) { i.motdBuffer.Content = append(i.motdBuffer.Content, msg.LastParam()) } func (i *ircHandler) motdEnd(msg *irc.Message) { i.session.sendJSON("motd", i.motdBuffer) i.motdBuffer = MOTD{} } func (i *ircHandler) badNick(msg *irc.Message) { i.session.sendJSON("nick_fail", NickFail{ Server: i.client.Host, }) } func (i *ircHandler) error(msg *irc.Message) { i.session.printError(msg.LastParam()) } 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, irc.Error: i.error, irc.ReplyWelcome: i.info, irc.ReplyYourHost: i.info, irc.ReplyCreated: i.info, 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, irc.ErrErroneousNickname: i.badNick, } } func (i *ircHandler) log(v ...interface{}) { s := fmt.Sprintln(v...) log.Println("[IRC]", i.session.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 } func printMessage(msg *irc.Message, i *irc.Client) { log.Println(i.GetNick()+":", msg.Prefix, msg.Command, msg.Params) }