mirror of
				https://github.com/google/alertmanager-irc-relay.git
				synced 2025-10-26 11:27:25 +01:00 
			
		
		
		
	Add tests for channel join retry logic
Also adopt interface for querying time information, so it can be faked properly during at test time Signed-off-by: Luca Bigliardi <shammash@google.com>
This commit is contained in:
		
							parent
							
								
									e7d5eefc49
								
							
						
					
					
						commit
						559b817262
					
				
							
								
								
									
										16
									
								
								backoff.go
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								backoff.go
									
									
									
									
									
								
							| @ -50,22 +50,6 @@ func jitterFunc(input int) int { | ||||
| 	return rand.Intn(input) | ||||
| } | ||||
| 
 | ||||
| // TimeTeller interface allows injection of fake time during testing | ||||
| type TimeTeller interface { | ||||
| 	Now() time.Time | ||||
| 	After(time.Duration) <-chan time.Time | ||||
| } | ||||
| 
 | ||||
| type RealTime struct{} | ||||
| 
 | ||||
| func (r *RealTime) Now() time.Time { | ||||
| 	return time.Now() | ||||
| } | ||||
| 
 | ||||
| func (r *RealTime) After(d time.Duration) <-chan time.Time { | ||||
| 	return time.After(d) | ||||
| } | ||||
| 
 | ||||
| type BackoffMaker struct{} | ||||
| 
 | ||||
| func (bm *BackoffMaker) NewDelayer(maxBackoff float64, resetDelta float64, durationUnit time.Duration) Delayer { | ||||
|  | ||||
| @ -20,24 +20,6 @@ import ( | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| type FakeTime struct { | ||||
| 	timeseries   []int | ||||
| 	lastIndex    int | ||||
| 	durationUnit time.Duration | ||||
| 	afterChan    chan time.Time | ||||
| } | ||||
| 
 | ||||
| func (f *FakeTime) Now() time.Time { | ||||
| 	timeDelta := time.Duration(f.timeseries[f.lastIndex]) * f.durationUnit | ||||
| 	fakeTime := time.Unix(0, 0).Add(timeDelta) | ||||
| 	f.lastIndex++ | ||||
| 	return fakeTime | ||||
| } | ||||
| 
 | ||||
| func (f *FakeTime) After(d time.Duration) <-chan time.Time { | ||||
| 	return f.afterChan | ||||
| } | ||||
| 
 | ||||
| func FakeJitter(input int) int { | ||||
| 	return input | ||||
| } | ||||
|  | ||||
							
								
								
									
										37
									
								
								fake_timeteller.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								fake_timeteller.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,37 @@ | ||||
| // Copyright 2021 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 ( | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| type FakeTime struct { | ||||
| 	timeseries   []int | ||||
| 	lastIndex    int | ||||
| 	durationUnit time.Duration | ||||
| 	afterChan    chan time.Time | ||||
| } | ||||
| 
 | ||||
| func (f *FakeTime) Now() time.Time { | ||||
| 	timeDelta := time.Duration(f.timeseries[f.lastIndex]) * f.durationUnit | ||||
| 	fakeTime := time.Unix(0, 0).Add(timeDelta) | ||||
| 	f.lastIndex++ | ||||
| 	return fakeTime | ||||
| } | ||||
| 
 | ||||
| func (f *FakeTime) After(d time.Duration) <-chan time.Time { | ||||
| 	return f.afterChan | ||||
| } | ||||
							
								
								
									
										10
									
								
								irc.go
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								irc.go
									
									
									
									
									
								
							| @ -100,9 +100,10 @@ type IRCNotifier struct { | ||||
| 
 | ||||
| 	NickservDelayWait time.Duration | ||||
| 	BackoffCounter    Delayer | ||||
| 	timeTeller        TimeTeller | ||||
| } | ||||
| 
 | ||||
| func NewIRCNotifier(config *Config, alertMsgs chan AlertMsg, delayerMaker DelayerMaker) (*IRCNotifier, error) { | ||||
| func NewIRCNotifier(config *Config, alertMsgs chan AlertMsg, delayerMaker DelayerMaker, timeTeller TimeTeller) (*IRCNotifier, error) { | ||||
| 
 | ||||
| 	ircConfig := makeGOIRCConfig(config) | ||||
| 
 | ||||
| @ -112,7 +113,7 @@ func NewIRCNotifier(config *Config, alertMsgs chan AlertMsg, delayerMaker Delaye | ||||
| 		ircConnectMaxBackoffSecs, ircConnectBackoffResetSecs, | ||||
| 		time.Second) | ||||
| 
 | ||||
| 	channelReconciler := NewChannelReconciler(config, client, delayerMaker) | ||||
| 	channelReconciler := NewChannelReconciler(config, client, delayerMaker, timeTeller) | ||||
| 
 | ||||
| 	notifier := &IRCNotifier{ | ||||
| 		Nick:              config.IRCNick, | ||||
| @ -125,6 +126,7 @@ func NewIRCNotifier(config *Config, alertMsgs chan AlertMsg, delayerMaker Delaye | ||||
| 		UsePrivmsg:        config.UsePrivmsg, | ||||
| 		NickservDelayWait: nickservWaitSecs * time.Second, | ||||
| 		BackoffCounter:    backoffCounter, | ||||
| 		timeTeller:        timeTeller, | ||||
| 	} | ||||
| 
 | ||||
| 	notifier.registerHandlers() | ||||
| @ -183,7 +185,7 @@ func (n *IRCNotifier) ChannelJoined(ctx context.Context, channel string) bool { | ||||
| 	select { | ||||
| 	case <-waitJoined: | ||||
| 		return true | ||||
| 	case <-time.After(ircJoinWaitSecs * time.Second): | ||||
| 	case <-n.timeTeller.After(ircJoinWaitSecs * time.Second): | ||||
| 		log.Printf("Channel %s not joined after %d seconds, giving bad news to caller", channel, ircJoinWaitSecs) | ||||
| 		return false | ||||
| 	case <-ctx.Done(): | ||||
| @ -220,7 +222,7 @@ func (n *IRCNotifier) ShutdownPhase() { | ||||
| 		log.Printf("Wait for IRC disconnect to complete") | ||||
| 		select { | ||||
| 		case <-n.sessionDownSignal: | ||||
| 		case <-time.After(n.Client.Config().Timeout): | ||||
| 		case <-n.timeTeller.After(n.Client.Config().Timeout): | ||||
| 			log.Printf("Timeout while waiting for IRC disconnect to complete, stopping anyway") | ||||
| 		} | ||||
| 		n.sessionWg.Done() | ||||
|  | ||||
| @ -16,6 +16,7 @@ package main | ||||
| 
 | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"log" | ||||
| @ -130,6 +131,17 @@ func (s *testServer) handleConnection(conn net.Conn) { | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (s *testServer) SendMsg(msg string) error { | ||||
| 	if s.Client == nil { | ||||
| 		return errors.New("Cannot write without client connected") | ||||
| 	} | ||||
| 	bufConn := bufio.NewWriter(s.Client) | ||||
| 	log.Printf("=Server= sending to client: %s", msg) | ||||
| 	_, err := bufConn.WriteString(msg) | ||||
| 	bufConn.Flush() | ||||
| 	return err | ||||
| } | ||||
| 
 | ||||
| func (s *testServer) SetCloseEarly(h closeEarlyHandler) { | ||||
| 	s.closeEarlyMu.Lock() | ||||
| 	defer s.closeEarlyMu.Unlock() | ||||
|  | ||||
| @ -44,11 +44,14 @@ func makeTestIRCConfig(IRCPort int) *Config { | ||||
| 
 | ||||
| func makeTestNotifier(t *testing.T, config *Config) (*IRCNotifier, chan AlertMsg, context.Context, context.CancelFunc, *sync.WaitGroup) { | ||||
| 	fakeDelayerMaker := &FakeDelayerMaker{} | ||||
| 	fakeTime := &FakeTime{ | ||||
| 		afterChan: make(chan time.Time, 1), | ||||
| 	} | ||||
| 	alertMsgs := make(chan AlertMsg) | ||||
| 	ctx, cancel := context.WithCancel(context.Background()) | ||||
| 	stopWg := sync.WaitGroup{} | ||||
| 	stopWg.Add(1) | ||||
| 	notifier, err := NewIRCNotifier(config, alertMsgs, fakeDelayerMaker) | ||||
| 	notifier, err := NewIRCNotifier(config, alertMsgs, fakeDelayerMaker, fakeTime) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(fmt.Sprintf("Could not create IRC notifier: %s", err)) | ||||
| 	} | ||||
|  | ||||
							
								
								
									
										2
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								main.go
									
									
									
									
									
								
							| @ -40,7 +40,7 @@ func main() { | ||||
| 	alertMsgs := make(chan AlertMsg, config.AlertBufferSize) | ||||
| 
 | ||||
| 	stopWg.Add(1) | ||||
| 	ircNotifier, err := NewIRCNotifier(config, alertMsgs, &BackoffMaker{}) | ||||
| 	ircNotifier, err := NewIRCNotifier(config, alertMsgs, &BackoffMaker{}, &RealTime{}) | ||||
| 	if err != nil { | ||||
| 		log.Printf("Could not create IRC notifier: %s", err) | ||||
| 		return | ||||
|  | ||||
| @ -32,7 +32,9 @@ const ( | ||||
| type channelState struct { | ||||
| 	channel IRCChannel | ||||
| 	client  *irc.Conn | ||||
| 	delayer Delayer | ||||
| 
 | ||||
| 	delayer    Delayer | ||||
| 	timeTeller TimeTeller | ||||
| 
 | ||||
| 	joinDone chan struct{} // joined when channel is closed | ||||
| 	joined   bool | ||||
| @ -42,13 +44,14 @@ type channelState struct { | ||||
| 	mu sync.Mutex | ||||
| } | ||||
| 
 | ||||
| func newChannelState(channel *IRCChannel, client *irc.Conn, delayerMaker DelayerMaker) *channelState { | ||||
| func newChannelState(channel *IRCChannel, client *irc.Conn, delayerMaker DelayerMaker, timeTeller TimeTeller) *channelState { | ||||
| 	delayer := delayerMaker.NewDelayer(ircJoinMaxBackoffSecs, ircJoinBackoffResetSecs, time.Second) | ||||
| 
 | ||||
| 	return &channelState{ | ||||
| 		channel:         *channel, | ||||
| 		client:          client, | ||||
| 		delayer:         delayer, | ||||
| 		timeTeller:      timeTeller, | ||||
| 		joinDone:        make(chan struct{}), | ||||
| 		joined:          false, | ||||
| 		joinUnsetSignal: make(chan bool), | ||||
| @ -108,7 +111,7 @@ func (c *channelState) join(ctx context.Context) { | ||||
| 	select { | ||||
| 	case <-c.JoinDone(): | ||||
| 		log.Printf("Channel %s monitor: join succeeded", c.channel.Name) | ||||
| 	case <-time.After(ircJoinWaitSecs * time.Second): | ||||
| 	case <-c.timeTeller.After(ircJoinWaitSecs * time.Second): | ||||
| 		log.Printf("Channel %s monitor: could not join after %d seconds, will retry", c.channel.Name, ircJoinWaitSecs) | ||||
| 	case <-ctx.Done(): | ||||
| 		log.Printf("Channel %s monitor: context canceled while waiting for join", c.channel.Name) | ||||
| @ -147,6 +150,7 @@ type ChannelReconciler struct { | ||||
| 	client          *irc.Conn | ||||
| 
 | ||||
| 	delayerMaker DelayerMaker | ||||
| 	timeTeller   TimeTeller | ||||
| 
 | ||||
| 	channels map[string]*channelState | ||||
| 
 | ||||
| @ -157,11 +161,12 @@ type ChannelReconciler struct { | ||||
| 	mu sync.Mutex | ||||
| } | ||||
| 
 | ||||
| func NewChannelReconciler(config *Config, client *irc.Conn, delayerMaker DelayerMaker) *ChannelReconciler { | ||||
| func NewChannelReconciler(config *Config, client *irc.Conn, delayerMaker DelayerMaker, timeTeller TimeTeller) *ChannelReconciler { | ||||
| 	reconciler := &ChannelReconciler{ | ||||
| 		preJoinChannels: config.IRCChannels, | ||||
| 		client:          client, | ||||
| 		delayerMaker:    delayerMaker, | ||||
| 		timeTeller:      timeTeller, | ||||
| 		channels:        make(map[string]*channelState), | ||||
| 	} | ||||
| 
 | ||||
| @ -219,7 +224,7 @@ func (r *ChannelReconciler) HandleKick(nick string, channel string) { | ||||
| } | ||||
| 
 | ||||
| func (r *ChannelReconciler) unsafeAddChannel(channel *IRCChannel) *channelState { | ||||
| 	c := newChannelState(channel, r.client, r.delayerMaker) | ||||
| 	c := newChannelState(channel, r.client, r.delayerMaker, r.timeTeller) | ||||
| 
 | ||||
| 	r.stopWg.Add(1) | ||||
| 	go c.Monitor(r.stopCtx, &r.stopWg) | ||||
|  | ||||
| @ -21,21 +21,12 @@ import ( | ||||
| 	"sort" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| 
 | ||||
| 	irc "github.com/fluffle/goirc/client" | ||||
| ) | ||||
| 
 | ||||
| func makeReconcilerTestIRCConfig(IRCPort int) *Config { | ||||
| 	config := makeTestIRCConfig(IRCPort) | ||||
| 	config.IRCChannels = []IRCChannel{ | ||||
| 		IRCChannel{Name: "#foo"}, | ||||
| 		IRCChannel{Name: "#bar"}, | ||||
| 		IRCChannel{Name: "#baz"}, | ||||
| 	} | ||||
| 	return config | ||||
| } | ||||
| 
 | ||||
| func makeTestReconciler(config *Config) (*ChannelReconciler, chan bool, chan bool) { | ||||
| func makeTestReconciler(config *Config) (*ChannelReconciler, chan bool, chan bool, *FakeTime) { | ||||
| 
 | ||||
| 	sessionUp := make(chan bool) | ||||
| 	sessionDown := make(chan bool) | ||||
| @ -53,15 +44,23 @@ func makeTestReconciler(config *Config) (*ChannelReconciler, chan bool, chan boo | ||||
| 		}) | ||||
| 
 | ||||
| 	fakeDelayerMaker := &FakeDelayerMaker{} | ||||
| 	reconciler := NewChannelReconciler(config, client, fakeDelayerMaker) | ||||
| 	fakeTime := &FakeTime{ | ||||
| 		afterChan: make(chan time.Time, 1), | ||||
| 	} | ||||
| 	reconciler := NewChannelReconciler(config, client, fakeDelayerMaker, fakeTime) | ||||
| 
 | ||||
| 	return reconciler, sessionUp, sessionDown | ||||
| 	return reconciler, sessionUp, sessionDown, fakeTime | ||||
| } | ||||
| 
 | ||||
| func TestPreJoinChannels(t *testing.T) { | ||||
| 	server, port := makeTestServer(t) | ||||
| 	config := makeReconcilerTestIRCConfig(port) | ||||
| 	reconciler, sessionUp, sessionDown := makeTestReconciler(config) | ||||
| 	config := makeTestIRCConfig(port) | ||||
| 	config.IRCChannels = []IRCChannel{ | ||||
| 		IRCChannel{Name: "#foo"}, | ||||
| 		IRCChannel{Name: "#bar"}, | ||||
| 		IRCChannel{Name: "#baz"}, | ||||
| 	} | ||||
| 	reconciler, sessionUp, sessionDown, _ := makeTestReconciler(config) | ||||
| 
 | ||||
| 	var testStep sync.WaitGroup | ||||
| 
 | ||||
| @ -98,3 +97,86 @@ func TestPreJoinChannels(t *testing.T) { | ||||
| 		t.Error("Did not pre-join channels") | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestKeepJoining(t *testing.T) { | ||||
| 	server, port := makeTestServer(t) | ||||
| 	config := makeTestIRCConfig(port) | ||||
| 	reconciler, sessionUp, sessionDown, fakeTime := makeTestReconciler(config) | ||||
| 
 | ||||
| 	var testStep sync.WaitGroup | ||||
| 
 | ||||
| 	var joinedCounter int | ||||
| 
 | ||||
| 	// Confirm join only after a few attempts | ||||
| 	joinHandler := func(conn *bufio.ReadWriter, line *irc.Line) error { | ||||
| 		joinedCounter++ | ||||
| 
 | ||||
| 		if joinedCounter == 3 { | ||||
| 			testStep.Done() | ||||
| 			return hJOIN(conn, line) | ||||
| 		} else { | ||||
| 			fakeTime.afterChan <- time.Now() | ||||
| 		} | ||||
| 		return nil | ||||
| 	} | ||||
| 	server.SetHandler("JOIN", joinHandler) | ||||
| 
 | ||||
| 	testStep.Add(1) | ||||
| 
 | ||||
| 	reconciler.client.Connect() | ||||
| 
 | ||||
| 	<-sessionUp | ||||
| 	reconciler.Start(context.Background()) | ||||
| 
 | ||||
| 	testStep.Wait() | ||||
| 
 | ||||
| 	reconciler.client.Quit("see ya") | ||||
| 	<-sessionDown | ||||
| 	reconciler.Stop() | ||||
| 
 | ||||
| 	server.Stop() | ||||
| 
 | ||||
| 	expectedJoinedCounter := 3 | ||||
| 
 | ||||
| 	if !reflect.DeepEqual(expectedJoinedCounter, joinedCounter) { | ||||
| 		t.Error("Did not keep joining") | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestKickRejoin(t *testing.T) { | ||||
| 	server, port := makeTestServer(t) | ||||
| 	config := makeTestIRCConfig(port) | ||||
| 	reconciler, sessionUp, sessionDown, _ := makeTestReconciler(config) | ||||
| 
 | ||||
| 	var testStep sync.WaitGroup | ||||
| 
 | ||||
| 	// Wait for channel to be joined | ||||
| 	joinHandler := func(conn *bufio.ReadWriter, line *irc.Line) error { | ||||
| 		hJOIN(conn, line) | ||||
| 		testStep.Done() | ||||
| 		return nil | ||||
| 	} | ||||
| 	server.SetHandler("JOIN", joinHandler) | ||||
| 
 | ||||
| 	testStep.Add(1) | ||||
| 
 | ||||
| 	reconciler.client.Connect() | ||||
| 
 | ||||
| 	<-sessionUp | ||||
| 	reconciler.Start(context.Background()) | ||||
| 
 | ||||
| 	testStep.Wait() | ||||
| 
 | ||||
| 	// Kick and wait for channel to be joined again | ||||
| 	testStep.Add(1) | ||||
| 	server.SendMsg(":test!~test@example.com KICK #foo foo :Bye!\n") | ||||
| 
 | ||||
| 	testStep.Wait() | ||||
| 
 | ||||
| 	reconciler.client.Quit("see ya") | ||||
| 	<-sessionDown | ||||
| 	reconciler.Stop() | ||||
| 
 | ||||
| 	server.Stop() | ||||
| 
 | ||||
| } | ||||
|  | ||||
							
								
								
									
										35
									
								
								time.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								time.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,35 @@ | ||||
| // Copyright 2021 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 ( | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| // TimeTeller interface allows injection of fake time during testing | ||||
| type TimeTeller interface { | ||||
| 	Now() time.Time | ||||
| 	After(time.Duration) <-chan time.Time | ||||
| } | ||||
| 
 | ||||
| type RealTime struct{} | ||||
| 
 | ||||
| func (r *RealTime) Now() time.Time { | ||||
| 	return time.Now() | ||||
| } | ||||
| 
 | ||||
| func (r *RealTime) After(d time.Duration) <-chan time.Time { | ||||
| 	return time.After(d) | ||||
| } | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Luca Bigliardi
						Luca Bigliardi