| 1 | package birc |
| 2 | |
| 3 | import ( |
| 4 | "bytes" |
| 5 | "fmt" |
| 6 | "io" |
| 7 | "strconv" |
| 8 | "strings" |
| 9 | "time" |
| 10 | |
| 11 | "github.com/lrstanley/girc" |
| 12 | "github.com/matterbridge-org/matterbridge/bridge/config" |
| 13 | "github.com/matterbridge-org/matterbridge/bridge/helper" |
| 14 | "github.com/paulrosania/go-charset/charset" |
| 15 | "github.com/saintfish/chardet" |
| 16 | |
| 17 | // We need to import the 'data' package as an implicit dependency. |
| 18 | // See: https://godoc.org/github.com/paulrosania/go-charset/charset |
| 19 | _ "github.com/paulrosania/go-charset/data" |
| 20 | ) |
| 21 | |
| 22 | func (b *Birc) handleCharset(msg *config.Message) error { |
| 23 | if b.GetString("Charset") != "" { |
| 24 | switch b.GetString("Charset") { |
| 25 | case "gbk", "gb18030", "gb2312", "big5", "euc-kr", "euc-jp", "shift-jis", "iso-2022-jp": |
| 26 | msg.Text = toUTF8(b.GetString("Charset"), msg.Text) |
| 27 | default: |
| 28 | buf := new(bytes.Buffer) |
| 29 | w, err := charset.NewWriter(b.GetString("Charset"), buf) |
| 30 | if err != nil { |
| 31 | b.Log.Errorf("charset to utf-8 conversion failed: %s", err) |
| 32 | return err |
| 33 | } |
| 34 | fmt.Fprint(w, msg.Text) |
| 35 | w.Close() |
| 36 | msg.Text = buf.String() |
| 37 | } |
| 38 | } |
| 39 | return nil |
| 40 | } |
| 41 | |
| 42 | // handleFiles returns true if we have handled the files, otherwise return false |
| 43 | func (b *Birc) handleFiles(msg *config.Message) bool { |
| 44 | if msg.Extra == nil { |
| 45 | return false |
| 46 | } |
| 47 | for _, rmsg := range helper.HandleExtra(msg, b.General) { |
| 48 | b.Local <- rmsg |
| 49 | } |
| 50 | if len(msg.Extra["file"]) == 0 { |
| 51 | return false |
| 52 | } |
| 53 | for _, f := range msg.Extra["file"] { |
| 54 | fi := f.(config.FileInfo) |
| 55 | if fi.Comment != "" { |
| 56 | msg.Text += fi.Comment + " : " |
| 57 | } |
| 58 | if fi.URL != "" { |
| 59 | msg.Text = fi.URL |
| 60 | if fi.Comment != "" { |
| 61 | msg.Text = fi.Comment + " : " + fi.URL |
| 62 | } |
| 63 | } |
| 64 | b.Local <- config.Message{Text: msg.Text, Username: msg.Username, Channel: msg.Channel, Event: msg.Event} |
| 65 | } |
| 66 | return true |
| 67 | } |
| 68 | |
| 69 | func (b *Birc) handleInvite(client *girc.Client, event girc.Event) { |
| 70 | if len(event.Params) != 2 { |
| 71 | return |
| 72 | } |
| 73 | |
| 74 | channel := event.Params[1] |
| 75 | |
| 76 | b.Log.Debugf("got invite for %s", channel) |
| 77 | |
| 78 | if _, ok := b.channels[channel]; ok { |
| 79 | b.i.Cmd.Join(channel) |
| 80 | } |
| 81 | } |
| 82 | |
| 83 | func (b *Birc) handleJoinPart(client *girc.Client, event girc.Event) { |
| 84 | if len(event.Params) == 0 { |
| 85 | b.Log.Debugf("handleJoinPart: empty Params? %#v", event) |
| 86 | return |
| 87 | } |
| 88 | channel := strings.ToLower(event.Params[0]) |
| 89 | if event.Command == "KICK" && event.Params[1] == b.Nick { |
| 90 | b.Log.Infof("Got kicked from %s by %s", channel, event.Source.Name) |
| 91 | time.Sleep(time.Duration(b.GetInt("RejoinDelay")) * time.Second) |
| 92 | b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: channel, Account: b.Account, Event: config.EventRejoinChannels} |
| 93 | return |
| 94 | } |
| 95 | if event.Command == "QUIT" { |
| 96 | if event.Source.Name == b.Nick && strings.Contains(event.Last(), "Ping timeout") { |
| 97 | b.Log.Infof("%s reconnecting ..", b.Account) |
| 98 | b.Remote <- config.Message{Username: "system", Text: "reconnect", Channel: channel, Account: b.Account, Event: config.EventFailure} |
| 99 | return |
| 100 | } |
| 101 | } |
| 102 | if event.Source.Name != b.Nick { |
| 103 | if b.GetBool("nosendjoinpart") { |
| 104 | return |
| 105 | } |
| 106 | msg := config.Message{Username: "system", Text: event.Source.Name + " " + strings.ToLower(event.Command) + "s", Channel: channel, Account: b.Account, Event: config.EventJoinLeave} |
| 107 | if b.GetBool("verbosejoinpart") { |
| 108 | b.Log.Debugf("<= Sending verbose JOIN_LEAVE event from %s to gateway", b.Account) |
| 109 | msg = config.Message{Username: "system", Text: event.Source.Name + " (" + event.Source.Ident + "@" + event.Source.Host + ") " + strings.ToLower(event.Command) + "s", Channel: channel, Account: b.Account, Event: config.EventJoinLeave} |
| 110 | } else { |
| 111 | b.Log.Debugf("<= Sending JOIN_LEAVE event from %s to gateway", b.Account) |
| 112 | } |
| 113 | b.Log.Debugf("<= Message is %#v", msg) |
| 114 | b.Remote <- msg |
| 115 | return |
| 116 | } |
| 117 | b.Log.Debugf("handle %#v", event) |
| 118 | } |
| 119 | |
| 120 | func (b *Birc) handleNewConnection(client *girc.Client, event girc.Event) { |
| 121 | b.Log.Debug("Registering callbacks") |
| 122 | i := b.i |
| 123 | b.Nick = event.Params[0] |
| 124 | |
| 125 | b.Log.Debug("Clearing handlers before adding in case of BNC reconnect") |
| 126 | i.Handlers.Clear("PRIVMSG") |
| 127 | i.Handlers.Clear("CTCP_ACTION") |
| 128 | i.Handlers.Clear(girc.RPL_TOPICWHOTIME) |
| 129 | i.Handlers.Clear(girc.NOTICE) |
| 130 | i.Handlers.Clear("JOIN") |
| 131 | i.Handlers.Clear("PART") |
| 132 | i.Handlers.Clear("QUIT") |
| 133 | i.Handlers.Clear("KICK") |
| 134 | i.Handlers.Clear("INVITE") |
| 135 | |
| 136 | i.Handlers.AddBg("PRIVMSG", b.handlePrivMsg) |
| 137 | i.Handlers.Add(girc.RPL_TOPICWHOTIME, b.handleTopicWhoTime) |
| 138 | i.Handlers.AddBg(girc.NOTICE, b.handleNotice) |
| 139 | i.Handlers.AddBg("JOIN", b.handleJoinPart) |
| 140 | i.Handlers.AddBg("PART", b.handleJoinPart) |
| 141 | i.Handlers.AddBg("QUIT", b.handleJoinPart) |
| 142 | i.Handlers.AddBg("KICK", b.handleJoinPart) |
| 143 | i.Handlers.Add("INVITE", b.handleInvite) |
| 144 | } |
| 145 | |
| 146 | func (b *Birc) handleNickServ() { |
| 147 | if !b.GetBool("UseSASL") && b.GetString("NickServNick") != "" && b.GetString("NickServPassword") != "" { |
| 148 | b.Log.Debugf("Sending identify to nickserv %s", b.GetString("NickServNick")) |
| 149 | b.i.Cmd.Message(b.GetString("NickServNick"), "IDENTIFY "+b.GetString("NickServPassword")) |
| 150 | } |
| 151 | if strings.EqualFold(b.GetString("NickServNick"), "Q@CServe.quakenet.org") { |
| 152 | b.Log.Debugf("Authenticating %s against %s", b.GetString("NickServUsername"), b.GetString("NickServNick")) |
| 153 | b.i.Cmd.Message(b.GetString("NickServNick"), "AUTH "+b.GetString("NickServUsername")+" "+b.GetString("NickServPassword")) |
| 154 | } |
| 155 | // give nickserv some slack |
| 156 | time.Sleep(time.Second * 5) |
| 157 | b.authDone = true |
| 158 | } |
| 159 | |
| 160 | func (b *Birc) handleNotice(client *girc.Client, event girc.Event) { |
| 161 | if strings.Contains(event.String(), "This nickname is registered") && event.Source.Name == b.GetString("NickServNick") { |
| 162 | b.handleNickServ() |
| 163 | } else { |
| 164 | b.handlePrivMsg(client, event) |
| 165 | } |
| 166 | } |
| 167 | |
| 168 | func (b *Birc) handleOther(client *girc.Client, event girc.Event) { |
| 169 | if b.GetInt("DebugLevel") == 1 { |
| 170 | if event.Command != "CLIENT_STATE_UPDATED" && |
| 171 | event.Command != "CLIENT_GENERAL_UPDATED" { |
| 172 | b.Log.Debugf("%#v", event.String()) |
| 173 | } |
| 174 | return |
| 175 | } |
| 176 | switch event.Command { |
| 177 | case "372", "375", "376", "250", "251", "252", "253", "254", "255", "265", "266", "002", "003", "004", "005": |
| 178 | return |
| 179 | } |
| 180 | b.Log.Debugf("%#v", event.String()) |
| 181 | } |
| 182 | |
| 183 | func (b *Birc) handleOtherAuth(client *girc.Client, event girc.Event) { |
| 184 | b.handleNickServ() |
| 185 | b.handleRunCommands() |
| 186 | // we are now fully connected |
| 187 | // only send on first connection |
| 188 | if b.FirstConnection { |
| 189 | b.connected <- nil |
| 190 | } |
| 191 | } |
| 192 | |
| 193 | func (b *Birc) handlePrivMsg(client *girc.Client, event girc.Event) { |
| 194 | if b.skipPrivMsg(event) { |
| 195 | return |
| 196 | } |
| 197 | |
| 198 | rmsg := config.Message{ |
| 199 | Username: event.Source.Name, |
| 200 | Channel: strings.ToLower(event.Params[0]), |
| 201 | Account: b.Account, |
| 202 | UserID: event.Source.Ident + "@" + event.Source.Host, |
| 203 | } |
| 204 | |
| 205 | b.Log.Debugf("== Receiving PRIVMSG: %s %s %#v", event.Source.Name, event.Last(), event) |
| 206 | |
| 207 | // set action event |
| 208 | if ok, ctcp := event.IsCTCP(); ok { |
| 209 | if ctcp.Command != girc.CTCP_ACTION { |
| 210 | b.Log.Debugf("dropping user ctcp, command: %s", ctcp.Command) |
| 211 | return |
| 212 | } |
| 213 | rmsg.Event = config.EventUserAction |
| 214 | } |
| 215 | |
| 216 | // set NOTICE event |
| 217 | if event.Command == "NOTICE" { |
| 218 | rmsg.Event = config.EventNoticeIRC |
| 219 | } |
| 220 | |
| 221 | // strip action, we made an event if it was an action |
| 222 | rmsg.Text += event.StripAction() |
| 223 | |
| 224 | // start detecting the charset |
| 225 | mycharset := b.GetString("Charset") |
| 226 | if mycharset == "" { |
| 227 | // detect what were sending so that we convert it to utf-8 |
| 228 | detector := chardet.NewTextDetector() |
| 229 | result, err := detector.DetectBest([]byte(rmsg.Text)) |
| 230 | if err != nil { |
| 231 | b.Log.Infof("detection failed for rmsg.Text: %#v", rmsg.Text) |
| 232 | return |
| 233 | } |
| 234 | b.Log.Debugf("detected %s confidence %#v", result.Charset, result.Confidence) |
| 235 | mycharset = result.Charset |
| 236 | // if we're not sure, just pick ISO-8859-1 |
| 237 | if result.Confidence < 80 { |
| 238 | mycharset = "ISO-8859-1" |
| 239 | } |
| 240 | } |
| 241 | switch mycharset { |
| 242 | case "gbk", "gb18030", "gb2312", "big5", "euc-kr", "euc-jp", "shift-jis", "iso-2022-jp": |
| 243 | rmsg.Text = toUTF8(b.GetString("Charset"), rmsg.Text) |
| 244 | default: |
| 245 | r, err := charset.NewReader(mycharset, strings.NewReader(rmsg.Text)) |
| 246 | if err != nil { |
| 247 | b.Log.Errorf("charset to utf-8 conversion failed: %s", err) |
| 248 | return |
| 249 | } |
| 250 | |
| 251 | output, _ := io.ReadAll(r) |
| 252 | rmsg.Text = string(output) |
| 253 | } |
| 254 | |
| 255 | b.Log.Debugf("<= Sending message from %s on %s to gateway", event.Params[0], b.Account) |
| 256 | b.Remote <- rmsg |
| 257 | } |
| 258 | |
| 259 | func (b *Birc) handleRunCommands() { |
| 260 | for _, cmd := range b.GetStringSlice("RunCommands") { |
| 261 | cmd = strings.ReplaceAll(cmd, "{BOTNICK}", b.Nick) |
| 262 | if err := b.i.Cmd.SendRaw(cmd); err != nil { |
| 263 | b.Log.Errorf("RunCommands %s failed: %s", cmd, err) |
| 264 | } |
| 265 | time.Sleep(time.Second) |
| 266 | } |
| 267 | } |
| 268 | |
| 269 | func (b *Birc) handleTopicWhoTime(client *girc.Client, event girc.Event) { |
| 270 | parts := strings.Split(event.Params[2], "!") |
| 271 | t, err := strconv.ParseInt(event.Params[3], 10, 64) |
| 272 | if err != nil { |
| 273 | b.Log.Errorf("Invalid time stamp: %s", event.Params[3]) |
| 274 | } |
| 275 | user := parts[0] |
| 276 | if len(parts) > 1 { |
| 277 | user += " [" + parts[1] + "]" |
| 278 | } |
| 279 | b.Log.Debugf("%s: Topic set by %s [%s]", event.Command, user, time.Unix(t, 0)) |
| 280 | } |
| 281 | |