mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-08 18:51:19 +00:00
119 lines
3.0 KiB
Go
119 lines
3.0 KiB
Go
|
package beacon
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"sync"
|
||
|
"testing"
|
||
|
|
||
|
healthTesting "github.com/prysmaticlabs/prysm/v5/api/client/beacon/testing"
|
||
|
"go.uber.org/mock/gomock"
|
||
|
)
|
||
|
|
||
|
func TestNodeHealth_IsHealthy(t *testing.T) {
|
||
|
tests := []struct {
|
||
|
name string
|
||
|
isHealthy bool
|
||
|
want bool
|
||
|
}{
|
||
|
{"initially healthy", true, true},
|
||
|
{"initially unhealthy", false, false},
|
||
|
}
|
||
|
for _, tt := range tests {
|
||
|
t.Run(tt.name, func(t *testing.T) {
|
||
|
n := &NodeHealthTracker{
|
||
|
isHealthy: &tt.isHealthy,
|
||
|
healthChan: make(chan bool, 1),
|
||
|
}
|
||
|
if got := n.IsHealthy(); got != tt.want {
|
||
|
t.Errorf("IsHealthy() = %v, want %v", got, tt.want)
|
||
|
}
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestNodeHealth_UpdateNodeHealth(t *testing.T) {
|
||
|
tests := []struct {
|
||
|
name string
|
||
|
initial bool // Initial health status
|
||
|
newStatus bool // Status to update to
|
||
|
shouldSend bool // Should a message be sent through the channel
|
||
|
}{
|
||
|
{"healthy to unhealthy", true, false, true},
|
||
|
{"unhealthy to healthy", false, true, true},
|
||
|
{"remain healthy", true, true, false},
|
||
|
{"remain unhealthy", false, false, false},
|
||
|
}
|
||
|
for _, tt := range tests {
|
||
|
t.Run(tt.name, func(t *testing.T) {
|
||
|
ctrl := gomock.NewController(t)
|
||
|
defer ctrl.Finish()
|
||
|
client := healthTesting.NewMockHealthClient(ctrl)
|
||
|
client.EXPECT().IsHealthy(gomock.Any()).Return(tt.newStatus)
|
||
|
n := &NodeHealthTracker{
|
||
|
isHealthy: &tt.initial,
|
||
|
node: client,
|
||
|
healthChan: make(chan bool, 1),
|
||
|
}
|
||
|
|
||
|
s := n.CheckHealth(context.Background())
|
||
|
// Check if health status was updated
|
||
|
if s != tt.newStatus {
|
||
|
t.Errorf("UpdateNodeHealth() failed to update isHealthy from %v to %v", tt.initial, tt.newStatus)
|
||
|
}
|
||
|
|
||
|
select {
|
||
|
case status := <-n.HealthUpdates():
|
||
|
if !tt.shouldSend {
|
||
|
t.Errorf("UpdateNodeHealth() unexpectedly sent status %v to HealthCh", status)
|
||
|
} else if status != tt.newStatus {
|
||
|
t.Errorf("UpdateNodeHealth() sent wrong status %v, want %v", status, tt.newStatus)
|
||
|
}
|
||
|
default:
|
||
|
if tt.shouldSend {
|
||
|
t.Error("UpdateNodeHealth() did not send any status to HealthCh when expected")
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestNodeHealth_Concurrency(t *testing.T) {
|
||
|
ctrl := gomock.NewController(t)
|
||
|
defer ctrl.Finish()
|
||
|
client := healthTesting.NewMockHealthClient(ctrl)
|
||
|
n := NewNodeHealthTracker(client)
|
||
|
var wg sync.WaitGroup
|
||
|
|
||
|
// Number of goroutines to spawn for both reading and writing
|
||
|
numGoroutines := 6
|
||
|
|
||
|
go func() {
|
||
|
for range n.HealthUpdates() {
|
||
|
// Consume values to avoid blocking on channel send.
|
||
|
}
|
||
|
}()
|
||
|
|
||
|
wg.Add(numGoroutines * 2) // for readers and writers
|
||
|
|
||
|
// Concurrently update health status
|
||
|
for i := 0; i < numGoroutines; i++ {
|
||
|
go func() {
|
||
|
defer wg.Done()
|
||
|
client.EXPECT().IsHealthy(gomock.Any()).Return(false)
|
||
|
n.CheckHealth(context.Background())
|
||
|
client.EXPECT().IsHealthy(gomock.Any()).Return(true)
|
||
|
n.CheckHealth(context.Background())
|
||
|
}()
|
||
|
}
|
||
|
|
||
|
// Concurrently read health status
|
||
|
for i := 0; i < numGoroutines; i++ {
|
||
|
go func() {
|
||
|
defer wg.Done()
|
||
|
_ = n.IsHealthy() // Just read the value
|
||
|
}()
|
||
|
}
|
||
|
|
||
|
wg.Wait() // Wait for all goroutines to finish
|
||
|
}
|