mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-01-19 05:45:21 +01:00
8e0534f756
This commit extends our healtcheck with an optional leader check. This is to ensure that given network partition or other cluster wide failure we act as soon as possible to avoid a split-brain situation where a new leader is elected but we still hold onto our etcd client.
103 lines
1.9 KiB
Go
103 lines
1.9 KiB
Go
//go:build kvdb_etcd
|
|
// +build kvdb_etcd
|
|
|
|
package cluster
|
|
|
|
import (
|
|
"context"
|
|
"os"
|
|
"runtime/pprof"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/lightningnetwork/lnd/kvdb/etcd"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
// GuardTimeout implements a test level timeout guard.
|
|
func GuardTimeout(t *testing.T, timeout time.Duration) func() {
|
|
done := make(chan struct{})
|
|
go func() {
|
|
select {
|
|
case <-time.After(timeout):
|
|
err := pprof.Lookup("goroutine").WriteTo(os.Stdout, 1)
|
|
require.NoError(t, err)
|
|
panic("test timeout")
|
|
|
|
case <-done:
|
|
}
|
|
}()
|
|
|
|
return func() {
|
|
close(done)
|
|
}
|
|
}
|
|
|
|
// TestEtcdElector tests that two candidates competing for leadership works as
|
|
// expected and that elected leader can resign and allow others to take on.
|
|
func TestEtcdElector(t *testing.T) {
|
|
guard := GuardTimeout(t, 5*time.Second)
|
|
defer guard()
|
|
|
|
tmpDir := t.TempDir()
|
|
|
|
etcdCfg, cleanup, err := etcd.NewEmbeddedEtcdInstance(tmpDir, 0, 0, "")
|
|
require.NoError(t, err)
|
|
t.Cleanup(cleanup)
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
defer cancel()
|
|
|
|
const (
|
|
election = "/election/"
|
|
id1 = "e1"
|
|
id2 = "e2"
|
|
ttl = 5
|
|
)
|
|
|
|
e1, err := newEtcdLeaderElector(
|
|
ctx, id1, election, ttl, etcdCfg,
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
e2, err := newEtcdLeaderElector(
|
|
ctx, id2, election, ttl, etcdCfg,
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
var wg sync.WaitGroup
|
|
ch := make(chan *etcdLeaderElector)
|
|
|
|
wg.Add(2)
|
|
ctxb := context.Background()
|
|
|
|
go func() {
|
|
defer wg.Done()
|
|
require.NoError(t, e1.Campaign(ctxb))
|
|
ch <- e1
|
|
}()
|
|
|
|
go func() {
|
|
defer wg.Done()
|
|
require.NoError(t, e2.Campaign(ctxb))
|
|
ch <- e2
|
|
}()
|
|
|
|
tmp := <-ch
|
|
first, err := tmp.Leader(ctxb)
|
|
require.NoError(t, err)
|
|
require.NoError(t, tmp.Resign(ctxb))
|
|
|
|
tmp = <-ch
|
|
second, err := tmp.Leader(ctxb)
|
|
require.NoError(t, err)
|
|
require.NoError(t, tmp.Resign(ctxb))
|
|
|
|
require.Contains(t, []string{id1, id2}, first)
|
|
require.Contains(t, []string{id1, id2}, second)
|
|
require.NotEqual(t, first, second)
|
|
|
|
wg.Wait()
|
|
}
|