Files
elect/elect_test.go

151 lines
4.2 KiB
Go
Raw Permalink Normal View History

2023-06-01 06:43:35 -07:00
package elect_test
import (
"testing"
"time"
"github.com/gopatchy/elect"
2023-06-01 06:43:35 -07:00
"github.com/stretchr/testify/require"
)
2023-06-04 10:43:32 -07:00
func TestOne(t *testing.T) {
2023-06-01 06:43:35 -07:00
t.Parallel()
2023-06-07 23:22:53 -07:00
ts := NewTestSystem(t, 1, 1)
2023-06-04 10:31:26 -07:00
defer ts.Stop()
2023-06-01 06:43:35 -07:00
2023-06-04 14:04:15 -07:00
require.False(t, ts.Candidate(0).IsLeader())
require.Eventually(t, ts.Candidate(0).IsLeader, 20*time.Second, 100*time.Millisecond)
2023-06-01 06:43:35 -07:00
}
2023-06-04 13:44:26 -07:00
func TestThree(t *testing.T) {
t.Parallel()
2023-06-07 23:22:53 -07:00
ts := NewTestSystem(t, 3, 3)
2023-06-04 13:44:26 -07:00
defer ts.Stop()
2023-06-04 14:04:15 -07:00
require.False(t, ts.Candidate(0).IsLeader())
require.False(t, ts.Candidate(1).IsLeader())
require.False(t, ts.Candidate(2).IsLeader())
2023-06-07 23:22:53 -07:00
{
w := NewWaiter()
w.Async(func() { require.Eventually(t, ts.Candidate(0).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(1).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(2).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Wait()
}
require.Equal(t, <-ts.Candidate(0).C, elect.StateLeader)
require.Empty(t, ts.Candidate(1).C)
require.Empty(t, ts.Candidate(2).C)
2023-06-04 13:44:26 -07:00
}
func TestFailover(t *testing.T) {
t.Parallel()
2023-06-07 23:22:53 -07:00
ts := NewTestSystem(t, 3, 3)
2023-06-04 13:44:26 -07:00
defer ts.Stop()
2023-06-04 14:04:15 -07:00
require.False(t, ts.Candidate(0).IsLeader())
require.False(t, ts.Candidate(1).IsLeader())
require.False(t, ts.Candidate(2).IsLeader())
2023-06-07 23:22:53 -07:00
{
w := NewWaiter()
w.Async(func() { require.Eventually(t, ts.Candidate(0).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(1).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(2).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Wait()
}
2023-06-04 13:44:26 -07:00
require.Equal(t, <-ts.Candidate(0).C, elect.StateLeader)
require.Empty(t, ts.Candidate(1).C)
require.Empty(t, ts.Candidate(2).C)
2023-06-04 13:44:26 -07:00
ts.SetServer(1)
require.Eventually(t, func() bool { return !ts.Candidate(0).IsLeader() }, 15*time.Second, 100*time.Millisecond)
// New candidate must not get leadership before old candidate loses it
require.False(t, ts.Candidate(1).IsLeader())
2023-06-04 14:04:15 -07:00
require.False(t, ts.Candidate(2).IsLeader())
2023-06-04 13:44:26 -07:00
require.Equal(t, <-ts.Candidate(0).C, elect.StateNotLeader)
require.Empty(t, ts.Candidate(1).C)
require.Empty(t, ts.Candidate(2).C)
2023-06-07 23:22:53 -07:00
{
w := NewWaiter()
w.Async(func() { require.Eventually(t, ts.Candidate(1).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(0).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(2).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Wait()
}
require.Equal(t, <-ts.Candidate(1).C, elect.StateLeader)
require.Empty(t, ts.Candidate(0).C)
require.Empty(t, ts.Candidate(2).C)
2023-06-04 14:04:15 -07:00
}
func TestPartialVotes(t *testing.T) {
t.Parallel()
2023-06-07 23:22:53 -07:00
ts := NewTestSystem(t, 3, 3)
2023-06-04 14:04:15 -07:00
defer ts.Stop()
ts.Proxy(0).SetRefuse(true)
2023-06-04 14:04:15 -07:00
require.False(t, ts.Candidate(0).IsLeader())
require.False(t, ts.Candidate(1).IsLeader())
require.False(t, ts.Candidate(2).IsLeader())
2023-06-07 23:22:53 -07:00
{
w := NewWaiter()
w.Async(func() { require.Eventually(t, ts.Candidate(0).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(1).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(2).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Wait()
}
require.Equal(t, <-ts.Candidate(0).C, elect.StateLeader)
require.Empty(t, ts.Candidate(1).C)
require.Empty(t, ts.Candidate(2).C)
2023-06-07 23:22:53 -07:00
}
func TestSplitVotes(t *testing.T) {
t.Parallel()
ts := NewTestSystem(t, 3, 3)
defer ts.Stop()
ts.SetServerForVoter(1, 1)
ts.SetServerForVoter(1, 2)
require.False(t, ts.Candidate(0).IsLeader())
2023-06-04 14:04:15 -07:00
require.False(t, ts.Candidate(1).IsLeader())
require.False(t, ts.Candidate(2).IsLeader())
2023-06-07 23:22:53 -07:00
{
w := NewWaiter()
w.Async(func() { require.Eventually(t, ts.Candidate(1).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(0).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Async(func() { require.Never(t, ts.Candidate(2).IsLeader, 20*time.Second, 100*time.Millisecond) })
w.Wait()
}
require.Equal(t, <-ts.Candidate(1).C, elect.StateLeader)
require.Empty(t, ts.Candidate(0).C)
require.Empty(t, ts.Candidate(2).C)
2023-06-04 13:44:26 -07:00
}