Skip to content

Commit

Permalink
Simplify current instance assertion and add test status utility
Browse files Browse the repository at this point in the history
Use `Progress` to assert the current instance and add utility to print
change across test F3 nodes useful for debugging.
  • Loading branch information
masih committed Jan 9, 2025
1 parent 3507ce2 commit 668558a
Showing 1 changed file with 67 additions and 7 deletions.
74 changes: 67 additions & 7 deletions f3_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,15 @@ package f3_test
import (
"context"
"fmt"
"os"
"path/filepath"
"reflect"
"sync/atomic"
"testing"
"time"

"github.com/filecoin-project/go-f3"
"github.com/filecoin-project/go-f3/certs"
"github.com/filecoin-project/go-f3/gpbft"
"github.com/filecoin-project/go-f3/internal/clock"
"github.com/filecoin-project/go-f3/internal/consensus"
Expand Down Expand Up @@ -89,14 +92,17 @@ func TestF3WithLookback(t *testing.T) {
func TestF3PauseResumeCatchup(t *testing.T) {
env := newTestEnvironment(t).withNodes(3).start()

if _, set := os.LookupEnv("GO_TEST_DEBUG"); set {
env.logStatus()
}

env.waitForInstanceNumber(1, 30*time.Second, true)

// Pausing two nodes should pause the network.
env.pauseNode(1)
env.pauseNode(2)

// Wait until node 0 stops receiving new instances
env.clock.Add(1 * time.Second)
env.waitForCondition(func() bool {
oldInstance := env.nodes[0].currentGpbftInstance()
env.clock.Add(10 * env.manifest.EC.Period)
Expand Down Expand Up @@ -324,12 +330,8 @@ type testNode struct {
}

func (n *testNode) currentGpbftInstance() uint64 {
c, err := n.f3.GetLatestCert(n.e.testCtx)
require.NoError(n.e.t, err)
if c == nil {
return n.e.manifest.InitialInstance
}
return c.GPBFTInstance + 1
require.NotNil(n.e.t, n.f3)
return n.f3.Progress().ID
}

func (n *testNode) init() *f3.F3 {
Expand Down Expand Up @@ -385,6 +387,43 @@ func (n *testNode) resume() {
require.NoError(n.e.t, n.f3.Resume(n.e.testCtx))
}

type testNodeStatus struct {
id int
initialised bool
running bool
progress gpbft.Instant
latestCert *certs.FinalityCertificate
}

func (s testNodeStatus) String() string {
switch {
case !s.initialised:
return fmt.Sprint("node ", s.id, ": uninitialised")
case !s.running:
return fmt.Sprint("node ", s.id, ": paused")
case s.latestCert == nil:
return fmt.Sprint("node ", s.id, ": at", s.progress, " with no finality certs")
default:
return fmt.Sprint("node ", s.id, ": at ", s.progress, ", finalized epoch ", s.latestCert.ECChain.Head().Epoch, " {", s.latestCert.GPBFTInstance, "}")
}
}

func (n *testNode) status() testNodeStatus {
var current testNodeStatus
current.id = n.id
if n.f3 != nil {
current.initialised = true
current.running = n.f3.IsRunning()
current.progress = n.f3.Progress()
if current.running {
cert, err := n.f3.GetLatestCert(n.e.testCtx)
require.NoError(n.e.t, err)
current.latestCert = cert
}
}
return current
}

type testEnv struct {
t *testing.T
errgrp *errgroup.Group
Expand Down Expand Up @@ -691,6 +730,27 @@ func (e *testEnv) injectDatastoreFailures(i int, fn func(op string) error) {
e.nodes[i].dsErrFunc = fn
}

func (e *testEnv) logStatus() {
e.t.Helper()

ctx, cancel := context.WithCancel(e.testCtx)
e.t.Cleanup(cancel)

go func() {
latest := make([]testNodeStatus, len(e.nodes))
for ctx.Err() == nil {
for i, n := range e.nodes {
current := n.status()
if !reflect.DeepEqual(current, latest[i]) {
latest[i] = current
e.t.Log(current.String())
}
}
time.Sleep(5 * time.Millisecond)
}
}()
}

// TODO: This code is copy-pasta from cmd/f3/run.go, consider taking it out into a shared testing lib.
// We could do the same to the F3 test instantiation
func runMessageSubscription(ctx context.Context, module *f3.F3, actorID gpbft.ActorID, signer gpbft.Signer) error {
Expand Down

0 comments on commit 668558a

Please sign in to comment.