From 436eb578aa6f85aec3a42c1e610679e1ddb3d179 Mon Sep 17 00:00:00 2001 From: James Lu Date: Sun, 20 Dec 2020 19:27:02 +0000 Subject: [PATCH 1/5] irc: add support for stateless bridging via draft/relaymsg As discussed at https://github.com/42wim/matterbridge/issues/667#issuecomment-634214165 --- bridge/irc/irc.go | 85 +++++++++++++++++++++++++++++++++-------------- 1 file changed, 60 insertions(+), 25 deletions(-) diff --git a/bridge/irc/irc.go b/bridge/irc/irc.go index 2352fcd76a..087e685d88 100644 --- a/bridge/irc/irc.go +++ b/bridge/irc/irc.go @@ -205,27 +205,57 @@ func (b *Birc) doConnect() { } } +// Sanitize nicks for RELAYMSG: replace IRC characters with special meanings with "-" +func sanitizeNick(nick string) string { + sanitize := func(r rune) rune { + if strings.ContainsRune("!+%@&#$:'\"?*,. ", r) { + return '-'; + } + return r + } + return strings.Map(sanitize, nick) +} + func (b *Birc) doSend() { rate := time.Millisecond * time.Duration(b.MessageDelay) throttle := time.NewTicker(rate) for msg := range b.Local { <-throttle.C username := msg.Username - if b.GetBool("Colornicks") && len(username) > 1 { - checksum := crc32.ChecksumIEEE([]byte(msg.Username)) - colorCode := checksum%14 + 2 // quick fix - prevent white or black color codes - username = fmt.Sprintf("\x03%02d%s\x0F", colorCode, msg.Username) - } + // Optional support for the proposed RELAYMSG extension, described at + // https://github.com/jlu5/ircv3-specifications/blob/master/extensions/relaymsg.md + if (b.i.HasCapability("overdrivenetworks.com/relaymsg") || b.i.HasCapability("draft/relaymsg")) && + b.GetBool("UseRelayMsg") { + username = sanitizeNick(username) + text := msg.Text + + // Work around girc chomping leading commas on single word messages? + if strings.HasPrefix(text, ":") && !strings.ContainsRune(text, ' ') { + text = ":" + text + } - switch msg.Event { - case config.EventUserAction: - b.i.Cmd.Action(msg.Channel, username+msg.Text) - case config.EventNoticeIRC: - b.Log.Debugf("Sending notice to channel %s", msg.Channel) - b.i.Cmd.Notice(msg.Channel, username+msg.Text) - default: - b.Log.Debugf("Sending to channel %s", msg.Channel) - b.i.Cmd.Message(msg.Channel, username+msg.Text) + if msg.Event == config.EventUserAction { + b.i.Cmd.SendRawf("RELAYMSG %s %s :\x01ACTION %s\x01", msg.Channel, username, text) + } else { + b.Log.Debugf("Sending RELAYMSG to channel %s: nick=%s", msg.Channel, username) + b.i.Cmd.SendRawf("RELAYMSG %s %s :%s", msg.Channel, username, text) + } + } else { + if b.GetBool("Colornicks") { + checksum := crc32.ChecksumIEEE([]byte(msg.Username)) + colorCode := checksum%14 + 2 // quick fix - prevent white or black color codes + username = fmt.Sprintf("\x03%02d%s\x0F", colorCode, msg.Username) + } + switch msg.Event { + case config.EventUserAction: + b.i.Cmd.Action(msg.Channel, username+msg.Text) + case config.EventNoticeIRC: + b.Log.Debugf("Sending notice to channel %s", msg.Channel) + b.i.Cmd.Notice(msg.Channel, username+msg.Text) + default: + b.Log.Debugf("Sending to channel %s", msg.Channel) + b.i.Cmd.Message(msg.Channel, username+msg.Text) + } } } } @@ -263,18 +293,19 @@ func (b *Birc) getClient() (*girc.Client, error) { b.Log.Debugf("setting pingdelay to %s", pingDelay) i := girc.New(girc.Config{ - Server: server, - ServerPass: b.GetString("Password"), - Port: port, - Nick: b.GetString("Nick"), - User: user, - Name: b.GetString("Nick"), - SSL: b.GetBool("UseTLS"), - TLSConfig: &tls.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), ServerName: server}, //nolint:gosec - PingDelay: pingDelay, + Server: server, + ServerPass: b.GetString("Password"), + Port: port, + Nick: b.GetString("Nick"), + User: user, + Name: b.GetString("Nick"), + SSL: b.GetBool("UseTLS"), + TLSConfig: &tls.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), ServerName: server}, //nolint:gosec + PingDelay: pingDelay, // skip gIRC internal rate limiting, since we have our own throttling - AllowFlood: true, - Debug: debug, + AllowFlood: true, + Debug: debug, + SupportedCaps: map[string][]string{ "overdrivenetworks.com/relaymsg": nil, "draft/relaymsg": nil }, }) return i, nil } @@ -311,6 +342,10 @@ func (b *Birc) skipPrivMsg(event girc.Event) bool { if event.Source.Name == b.Nick { return true } + // don't forward messages we sent via RELAYMSG + if relayedNick, ok := event.Tags.Get("relaymsg"); ok && relayedNick == b.Nick { + return true + } return false } From 9241f901c1de4273f373c55d614bd1a8b0dce261 Mon Sep 17 00:00:00 2001 From: James Lu Date: Sun, 20 Dec 2020 23:41:30 +0000 Subject: [PATCH 2/5] irc: handle the draft/relaymsg tag in spoofed messages too --- bridge/irc/irc.go | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/bridge/irc/irc.go b/bridge/irc/irc.go index 087e685d88..f16d43341e 100644 --- a/bridge/irc/irc.go +++ b/bridge/irc/irc.go @@ -343,6 +343,11 @@ func (b *Birc) skipPrivMsg(event girc.Event) bool { return true } // don't forward messages we sent via RELAYMSG + if relayedNick, ok := event.Tags.Get("draft/relaymsg"); ok && relayedNick == b.Nick { + return true + } + // This is the old name of the cap sent in spoofed messages; I've kept this in + // for compatibility reasons if relayedNick, ok := event.Tags.Get("relaymsg"); ok && relayedNick == b.Nick { return true } From caaffd7b83e5e8fbbe1aa5e0d9c4227d8f9040f7 Mon Sep 17 00:00:00 2001 From: James Lu Date: Tue, 29 Dec 2020 13:49:57 -0800 Subject: [PATCH 3/5] Apply suggestions from code review Co-authored-by: Wim --- bridge/irc/irc.go | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/bridge/irc/irc.go b/bridge/irc/irc.go index f16d43341e..78e3924f9a 100644 --- a/bridge/irc/irc.go +++ b/bridge/irc/irc.go @@ -224,6 +224,7 @@ func (b *Birc) doSend() { username := msg.Username // Optional support for the proposed RELAYMSG extension, described at // https://github.com/jlu5/ircv3-specifications/blob/master/extensions/relaymsg.md + // nolint:nestif if (b.i.HasCapability("overdrivenetworks.com/relaymsg") || b.i.HasCapability("draft/relaymsg")) && b.GetBool("UseRelayMsg") { username = sanitizeNick(username) @@ -235,10 +236,10 @@ func (b *Birc) doSend() { } if msg.Event == config.EventUserAction { - b.i.Cmd.SendRawf("RELAYMSG %s %s :\x01ACTION %s\x01", msg.Channel, username, text) + b.i.Cmd.SendRawf("RELAYMSG %s %s :\x01ACTION %s\x01", msg.Channel, username, text) //nolint:errcheck } else { b.Log.Debugf("Sending RELAYMSG to channel %s: nick=%s", msg.Channel, username) - b.i.Cmd.SendRawf("RELAYMSG %s %s :%s", msg.Channel, username, text) + b.i.Cmd.SendRawf("RELAYMSG %s %s :%s", msg.Channel, username, text) //nolint:errcheck } } else { if b.GetBool("Colornicks") { From cf0715cede266b0b4fbbfd97997e2b76c563eb44 Mon Sep 17 00:00:00 2001 From: James Lu Date: Tue, 29 Dec 2020 21:53:16 +0000 Subject: [PATCH 4/5] Run gofmt on irc.go --- bridge/irc/irc.go | 24 ++++++++++++------------ 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/bridge/irc/irc.go b/bridge/irc/irc.go index 78e3924f9a..4ccdee95fe 100644 --- a/bridge/irc/irc.go +++ b/bridge/irc/irc.go @@ -209,7 +209,7 @@ func (b *Birc) doConnect() { func sanitizeNick(nick string) string { sanitize := func(r rune) rune { if strings.ContainsRune("!+%@&#$:'\"?*,. ", r) { - return '-'; + return '-' } return r } @@ -226,7 +226,7 @@ func (b *Birc) doSend() { // https://github.com/jlu5/ircv3-specifications/blob/master/extensions/relaymsg.md // nolint:nestif if (b.i.HasCapability("overdrivenetworks.com/relaymsg") || b.i.HasCapability("draft/relaymsg")) && - b.GetBool("UseRelayMsg") { + b.GetBool("UseRelayMsg") { username = sanitizeNick(username) text := msg.Text @@ -294,19 +294,19 @@ func (b *Birc) getClient() (*girc.Client, error) { b.Log.Debugf("setting pingdelay to %s", pingDelay) i := girc.New(girc.Config{ - Server: server, - ServerPass: b.GetString("Password"), - Port: port, - Nick: b.GetString("Nick"), - User: user, - Name: b.GetString("Nick"), - SSL: b.GetBool("UseTLS"), - TLSConfig: &tls.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), ServerName: server}, //nolint:gosec - PingDelay: pingDelay, + Server: server, + ServerPass: b.GetString("Password"), + Port: port, + Nick: b.GetString("Nick"), + User: user, + Name: b.GetString("Nick"), + SSL: b.GetBool("UseTLS"), + TLSConfig: &tls.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), ServerName: server}, //nolint:gosec + PingDelay: pingDelay, // skip gIRC internal rate limiting, since we have our own throttling AllowFlood: true, Debug: debug, - SupportedCaps: map[string][]string{ "overdrivenetworks.com/relaymsg": nil, "draft/relaymsg": nil }, + SupportedCaps: map[string][]string{"overdrivenetworks.com/relaymsg": nil, "draft/relaymsg": nil}, }) return i, nil } From a1e0de800d04733ccb0f8ae3e0e7de07c76beb82 Mon Sep 17 00:00:00 2001 From: James Lu Date: Tue, 29 Dec 2020 22:02:12 +0000 Subject: [PATCH 5/5] Document relaymsg in matterbridge.toml.sample --- matterbridge.toml.sample | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/matterbridge.toml.sample b/matterbridge.toml.sample index 9749080fd6..a92892b018 100644 --- a/matterbridge.toml.sample +++ b/matterbridge.toml.sample @@ -193,6 +193,19 @@ ShowTopicChange=false #OPTIONAL (default 0) JoinDelay=0 +#Use the optional RELAYMSG extension for username spoofing on IRC. +#This requires an IRCd that supports the draft/relaymsg specification: currently this includes +#Oragono 2.4.0+ and InspIRCd 3 with the m_relaymsg contrib module. +#See https://github.com/42wim/matterbridge/issues/667#issuecomment-634214165 for more details. +#Spoofed nicks will use the configured RemoteNickFormat, replacing reserved IRC characters +#(!+%@&#$:'"?*,.) with a hyphen (-). +#On most configurations, the RemoteNickFormat must include a separator character such as "/". +#You should make sure that the settings here match your IRCd. +#This option overrides ColorNicks. +#OPTIONAL (default false) +UseRelayMsg=false +#RemoteNickFormat="{NICK}/{PROTOCOL}" + ################################################################### #XMPP section ###################################################################