mirror of
https://github.com/google/alertmanager-irc-relay.git
synced 2025-01-14 06:02:38 +01:00
d47139a6d6
After a connection loss on an IRC session with a ngIRCd, the alertmanager-irc-relay was unable to reconnect. After some debugging, the error's origin was the state tracking within the used goirc library. When using an unidentified session, ngIRCd prefixes the user's ident with a `~`. The state tracking registers this and keeps `~${NICK}` as the current and the new ident for future reconnects. However, `~` is not a valid char for the `<user>` part in the `USER` command, at least not for ngIRCd. To clarify this behaviour, take a look at the following log. First, the initial connection is begin established correctly. Keep an eye on the `USER` command being sent to the server. > http.go:132: INFO Starting HTTP server > irc.go:308: INFO Connected to IRC server, waiting to establish session > connection.go:543: DEBUG -> NICK alertbot > connection.go:543: DEBUG -> USER alertbot 12 * :Alertmanager IRC Relay > connection.go:474: DEBUG <- :__SERVER__ 001 alertbot :Welcome to the Internet Relay Network alertbot!~alertbot@__IP__ Now, there was a network incident and the session needs to be recreated. > connection.go:466: ERROR irc.recv(): read tcp __REDACTED__: read: connection timed out > connection.go:577: INFO irc.Close(): Disconnected from server. > irc.go:150: INFO Disconnected from IRC > reconciler.go:129: INFO Channel #alerts monitor: context canceled while monitoring > irc.go:300: INFO Connecting to IRC __SERVER__ > backoff.go:111: INFO Backoff for 0s starts > backoff.go:114: INFO Backoff for 0s ends > connection.go:390: INFO irc.Connect(): Connecting to __SERVER__. > irc.go:308: INFO Connected to IRC server, waiting to establish session > connection.go:543: DEBUG -> NICK alertbot > connection.go:543: DEBUG -> USER ~alertbot 12 * :Alertmanager IRC Relay > connection.go:474: DEBUG <- ERROR :Invalid user name > connection.go:577: INFO irc.Close(): Disconnected from server. > irc.go:150: INFO Disconnected from IRC > irc.go:319: WARN Receiving a session down before the session is up, this is odd This time, the used `user` part of the `USER` command has the prefixed `~` and fails. However, without using `-debug` and taking a very close look, this error can be missed very easy. As the new ident is invalid, the alertmanager-irc-relay is now stuck in an endless reconnection loop. This fix is kind of straight forward and just checks if the ident has changed before trying to reconnect. It might not be the prettiest solution, but recreating the whole *irc.Config resulted in other bugs as it was still referenced - even after being `Close`d.
352 lines
10 KiB
Go
352 lines
10 KiB
Go
// Copyright 2018 Google LLC
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package main
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
irc "github.com/fluffle/goirc/client"
|
|
"github.com/google/alertmanager-irc-relay/logging"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
|
)
|
|
|
|
const (
|
|
pingFrequencySecs = 60
|
|
connectionTimeoutSecs = 30
|
|
nickservWaitSecs = 10
|
|
ircConnectMaxBackoffSecs = 300
|
|
ircConnectBackoffResetSecs = 1800
|
|
)
|
|
|
|
var (
|
|
ircConnectedGauge = promauto.NewGauge(prometheus.GaugeOpts{
|
|
Name: "irc_connected",
|
|
Help: "Whether the IRC connection is established",
|
|
})
|
|
ircSentMsgs = promauto.NewCounterVec(prometheus.CounterOpts{
|
|
Name: "irc_sent_msgs",
|
|
Help: "Number of IRC messages sent"},
|
|
[]string{"ircchannel"},
|
|
)
|
|
ircSendMsgErrors = promauto.NewCounterVec(prometheus.CounterOpts{
|
|
Name: "irc_send_msg_errors",
|
|
Help: "Errors while sending IRC messages"},
|
|
[]string{"ircchannel", "error"},
|
|
)
|
|
)
|
|
|
|
func loggerHandler(_ *irc.Conn, line *irc.Line) {
|
|
logging.Info("Received: '%s'", line.Raw)
|
|
}
|
|
|
|
func makeGOIRCConfig(config *Config) *irc.Config {
|
|
ircConfig := irc.NewConfig(config.IRCNick)
|
|
ircConfig.Me.Ident = config.IRCNick
|
|
ircConfig.Me.Name = config.IRCRealName
|
|
ircConfig.Server = strings.Join(
|
|
[]string{config.IRCHost, strconv.Itoa(config.IRCPort)}, ":")
|
|
ircConfig.Pass = config.IRCHostPass
|
|
ircConfig.SSL = config.IRCUseSSL
|
|
ircConfig.SSLConfig = &tls.Config{
|
|
ServerName: config.IRCHost,
|
|
InsecureSkipVerify: !config.IRCVerifySSL,
|
|
}
|
|
ircConfig.PingFreq = pingFrequencySecs * time.Second
|
|
ircConfig.Timeout = connectionTimeoutSecs * time.Second
|
|
ircConfig.NewNick = func(n string) string { return n + "^" }
|
|
|
|
return ircConfig
|
|
}
|
|
|
|
type IRCNotifier struct {
|
|
// Nick stores the nickname specified in the config, because irc.Client
|
|
// might change its copy.
|
|
Nick string
|
|
NickPassword string
|
|
|
|
NickservName string
|
|
NickservIdentifyPatterns []string
|
|
|
|
// As the goirc library might alter the irc.Config created by makeGOIRCConfig,
|
|
// we might also want to keep a reference to the original Config to restore
|
|
// the desired state.
|
|
Config *Config
|
|
IrcConfig *irc.Config
|
|
|
|
Client *irc.Conn
|
|
AlertMsgs chan AlertMsg
|
|
|
|
// irc.Conn has a Connected() method that can tell us wether the TCP
|
|
// connection is up, and thus if we should trigger connect/disconnect.
|
|
// We need to track the session establishment also at a higher level to
|
|
// understand when the server has accepted us and thus when we can join
|
|
// channels, send notices, etc.
|
|
sessionUp bool
|
|
sessionUpSignal chan bool
|
|
sessionDownSignal chan bool
|
|
sessionWg sync.WaitGroup
|
|
|
|
channelReconciler *ChannelReconciler
|
|
|
|
UsePrivmsg bool
|
|
|
|
NickservDelayWait time.Duration
|
|
BackoffCounter Delayer
|
|
timeTeller TimeTeller
|
|
}
|
|
|
|
func NewIRCNotifier(config *Config, alertMsgs chan AlertMsg, delayerMaker DelayerMaker, timeTeller TimeTeller) (*IRCNotifier, error) {
|
|
|
|
ircConfig := makeGOIRCConfig(config)
|
|
|
|
client := irc.Client(ircConfig)
|
|
|
|
backoffCounter := delayerMaker.NewDelayer(
|
|
ircConnectMaxBackoffSecs, ircConnectBackoffResetSecs,
|
|
time.Second)
|
|
|
|
channelReconciler := NewChannelReconciler(config, client, delayerMaker, timeTeller)
|
|
|
|
notifier := &IRCNotifier{
|
|
Nick: config.IRCNick,
|
|
NickPassword: config.IRCNickPass,
|
|
NickservName: config.NickservName,
|
|
NickservIdentifyPatterns: config.NickservIdentifyPatterns,
|
|
Config: config,
|
|
IrcConfig: ircConfig,
|
|
Client: client,
|
|
AlertMsgs: alertMsgs,
|
|
sessionUpSignal: make(chan bool),
|
|
sessionDownSignal: make(chan bool),
|
|
channelReconciler: channelReconciler,
|
|
UsePrivmsg: config.UsePrivmsg,
|
|
NickservDelayWait: nickservWaitSecs * time.Second,
|
|
BackoffCounter: backoffCounter,
|
|
timeTeller: timeTeller,
|
|
}
|
|
|
|
notifier.registerHandlers()
|
|
|
|
return notifier, nil
|
|
}
|
|
|
|
func (n *IRCNotifier) registerHandlers() {
|
|
n.Client.HandleFunc(irc.CONNECTED,
|
|
func(*irc.Conn, *irc.Line) {
|
|
logging.Info("Session established")
|
|
n.sessionUpSignal <- true
|
|
})
|
|
|
|
n.Client.HandleFunc(irc.DISCONNECTED,
|
|
func(*irc.Conn, *irc.Line) {
|
|
logging.Info("Disconnected from IRC")
|
|
n.sessionDownSignal <- false
|
|
})
|
|
|
|
n.Client.HandleFunc(irc.NOTICE,
|
|
func(_ *irc.Conn, line *irc.Line) {
|
|
n.HandleNotice(line.Nick, line.Text())
|
|
})
|
|
|
|
for _, event := range []string{"433"} {
|
|
n.Client.HandleFunc(event, loggerHandler)
|
|
}
|
|
}
|
|
|
|
func (n *IRCNotifier) HandleNotice(nick string, msg string) {
|
|
logging.Info("Received NOTICE from %s: %s", nick, msg)
|
|
if strings.ToLower(nick) == "nickserv" {
|
|
n.HandleNickservMsg(msg)
|
|
}
|
|
}
|
|
|
|
func (n *IRCNotifier) HandleNickservMsg(msg string) {
|
|
if n.NickPassword == "" {
|
|
logging.Debug("Skip processing NickServ request, no password configured")
|
|
return
|
|
}
|
|
|
|
// Remove most common formatting options from NickServ messages
|
|
cleaner := strings.NewReplacer(
|
|
"\001", "", // bold
|
|
"\002", "", // faint
|
|
"\004", "", // underline
|
|
"\037", "", // underline
|
|
)
|
|
cleanedMsg := cleaner.Replace(msg)
|
|
|
|
for _, identifyPattern := range n.NickservIdentifyPatterns {
|
|
logging.Debug("Checking if NickServ message matches identify request '%s'", identifyPattern)
|
|
if strings.Contains(cleanedMsg, identifyPattern) {
|
|
logging.Info("Handling NickServ request to IDENTIFY")
|
|
n.Client.Privmsgf(n.NickservName, "IDENTIFY %s", n.NickPassword)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (n *IRCNotifier) MaybeGhostNick() {
|
|
if n.NickPassword == "" {
|
|
logging.Debug("Skip GHOST check, no password configured")
|
|
return
|
|
}
|
|
|
|
currentNick := n.Client.Me().Nick
|
|
if currentNick != n.Nick {
|
|
logging.Info("My nick is '%s', sending GHOST to NickServ to get '%s'",
|
|
currentNick, n.Nick)
|
|
n.Client.Privmsgf(n.NickservName, "GHOST %s %s", n.Nick,
|
|
n.NickPassword)
|
|
time.Sleep(n.NickservDelayWait)
|
|
|
|
logging.Info("Changing nick to '%s'", n.Nick)
|
|
n.Client.Nick(n.Nick)
|
|
time.Sleep(n.NickservDelayWait)
|
|
}
|
|
}
|
|
|
|
func (n *IRCNotifier) MaybeWaitForNickserv() {
|
|
if n.NickPassword == "" {
|
|
logging.Debug("Skip NickServ wait, no password configured")
|
|
return
|
|
}
|
|
|
|
// Very lazy/optimistic, but this is good enough for my irssi config,
|
|
// so it should work here as well.
|
|
logging.Info("Waiting for NickServ to notice us and issue an identify request")
|
|
time.Sleep(n.NickservDelayWait)
|
|
}
|
|
|
|
func (n *IRCNotifier) ChannelJoined(ctx context.Context, channel string) bool {
|
|
|
|
isJoined, waitJoined := n.channelReconciler.JoinChannel(channel)
|
|
if isJoined {
|
|
return true
|
|
}
|
|
|
|
select {
|
|
case <-waitJoined:
|
|
return true
|
|
case <-n.timeTeller.After(ircJoinWaitSecs * time.Second):
|
|
logging.Warn("Channel %s not joined after %d seconds, giving bad news to caller", channel, ircJoinWaitSecs)
|
|
return false
|
|
case <-ctx.Done():
|
|
logging.Info("Context canceled while waiting for join on channel %s", channel)
|
|
return false
|
|
}
|
|
}
|
|
|
|
func (n *IRCNotifier) SendAlertMsg(ctx context.Context, alertMsg *AlertMsg) {
|
|
if !n.sessionUp {
|
|
logging.Error("Cannot send alert to %s : IRC not connected", alertMsg.Channel)
|
|
ircSendMsgErrors.WithLabelValues(alertMsg.Channel, "not_connected").Inc()
|
|
return
|
|
}
|
|
if !n.ChannelJoined(ctx, alertMsg.Channel) {
|
|
logging.Error("Cannot send alert to %s : cannot join channel", alertMsg.Channel)
|
|
ircSendMsgErrors.WithLabelValues(alertMsg.Channel, "not_joined").Inc()
|
|
return
|
|
}
|
|
|
|
if n.UsePrivmsg {
|
|
n.Client.Privmsg(alertMsg.Channel, alertMsg.Alert)
|
|
} else {
|
|
n.Client.Notice(alertMsg.Channel, alertMsg.Alert)
|
|
}
|
|
ircSentMsgs.WithLabelValues(alertMsg.Channel).Inc()
|
|
}
|
|
|
|
func (n *IRCNotifier) ShutdownPhase() {
|
|
if n.sessionUp {
|
|
logging.Info("IRC client connected, quitting")
|
|
n.Client.Quit("see ya")
|
|
|
|
logging.Info("Wait for IRC disconnect to complete")
|
|
select {
|
|
case <-n.sessionDownSignal:
|
|
case <-n.timeTeller.After(n.Client.Config().Timeout):
|
|
logging.Warn("Timeout while waiting for IRC disconnect to complete, stopping anyway")
|
|
}
|
|
n.sessionWg.Done()
|
|
}
|
|
logging.Info("IRC shutdown complete")
|
|
}
|
|
|
|
func (n *IRCNotifier) ConnectedPhase(ctx context.Context) {
|
|
select {
|
|
case alertMsg := <-n.AlertMsgs:
|
|
n.SendAlertMsg(ctx, &alertMsg)
|
|
case <-n.sessionDownSignal:
|
|
n.sessionUp = false
|
|
n.sessionWg.Done()
|
|
n.channelReconciler.Stop()
|
|
n.Client.Quit("see ya")
|
|
ircConnectedGauge.Set(0)
|
|
case <-ctx.Done():
|
|
logging.Info("IRC routine asked to terminate")
|
|
}
|
|
}
|
|
|
|
func (n *IRCNotifier) SetupPhase(ctx context.Context) {
|
|
if !n.Client.Connected() {
|
|
if n.IrcConfig.Me.Ident != n.Config.IRCNick {
|
|
logging.Debug("Restoring IRC nick from %s to %s", n.IrcConfig.Me.Ident, n.Config.IRCNick)
|
|
n.IrcConfig.Me.Ident = n.Config.IRCNick
|
|
}
|
|
|
|
logging.Info("Connecting to IRC %s", n.Client.Config().Server)
|
|
if ok := n.BackoffCounter.DelayContext(ctx); !ok {
|
|
return
|
|
}
|
|
if err := n.Client.ConnectContext(WithWaitGroup(ctx, &n.sessionWg)); err != nil {
|
|
logging.Error("Could not connect to IRC: %s", err)
|
|
return
|
|
}
|
|
logging.Info("Connected to IRC server, waiting to establish session")
|
|
}
|
|
select {
|
|
case <-n.sessionUpSignal:
|
|
n.sessionUp = true
|
|
n.sessionWg.Add(1)
|
|
n.MaybeGhostNick()
|
|
n.MaybeWaitForNickserv()
|
|
n.channelReconciler.Start(ctx)
|
|
ircConnectedGauge.Set(1)
|
|
case <-n.sessionDownSignal:
|
|
logging.Warn("Receiving a session down before the session is up, this is odd")
|
|
case <-ctx.Done():
|
|
logging.Info("IRC routine asked to terminate")
|
|
}
|
|
}
|
|
|
|
func (n *IRCNotifier) Run(ctx context.Context, stopWg *sync.WaitGroup) {
|
|
defer stopWg.Done()
|
|
|
|
for ctx.Err() != context.Canceled {
|
|
if !n.sessionUp {
|
|
n.SetupPhase(ctx)
|
|
} else {
|
|
n.ConnectedPhase(ctx)
|
|
}
|
|
}
|
|
n.ShutdownPhase()
|
|
}
|