itest: add icase to node log filename + restart nodes before each icase

This commit adds the icase name to the log filename, to make it simpler
to find problematic tests. Additionally after this commit we'll restart
Alice and Bob (the base harness nodes) before each icase to start with a
clean state.
This commit is contained in:
Andras Banki-Horvath 2020-10-29 15:37:17 +01:00
parent d6c2ee1bbf
commit 979c8497b2
No known key found for this signature in database
GPG Key ID: 80E5375C094198D8
4 changed files with 85 additions and 50 deletions

3
.gitignore vendored

@ -32,8 +32,7 @@ _testmain.go
/lncli-itest /lncli-itest
# Integration test log files # Integration test log files
lntest/itest/output*.log lntest/itest/*.log
lntest/itest/pprof*.log
lntest/itest/.backendlogs lntest/itest/.backendlogs
lntest/itest/.minerlogs lntest/itest/.minerlogs
lntest/itest/lnd-itest lntest/itest/lnd-itest

@ -36,6 +36,9 @@ const DefaultCSV = 4
type NetworkHarness struct { type NetworkHarness struct {
netParams *chaincfg.Params netParams *chaincfg.Params
// currentTestCase holds the name for the currently run test case.
currentTestCase string
// lndBinary is the full path to the lnd binary that was specifically // lndBinary is the full path to the lnd binary that was specifically
// compiled with all required itest flags. // compiled with all required itest flags.
lndBinary string lndBinary string
@ -131,10 +134,11 @@ func (f *fakeLogger) Println(args ...interface{}) {}
// rpc clients capable of communicating with the initial seeder nodes are // rpc clients capable of communicating with the initial seeder nodes are
// created. Nodes are initialized with the given extra command line flags, which // created. Nodes are initialized with the given extra command line flags, which
// should be formatted properly - "--arg=value". // should be formatted properly - "--arg=value".
func (n *NetworkHarness) SetUp(lndArgs []string) error { func (n *NetworkHarness) SetUp(testCase string, lndArgs []string) error {
// Swap out grpc's default logger with out fake logger which drops the // Swap out grpc's default logger with out fake logger which drops the
// statements on the floor. // statements on the floor.
grpclog.SetLogger(&fakeLogger{}) grpclog.SetLogger(&fakeLogger{})
n.currentTestCase = testCase
// Start the initial seeder nodes within the test network, then connect // Start the initial seeder nodes within the test network, then connect
// their respective RPC clients. // their respective RPC clients.
@ -241,21 +245,23 @@ out:
return nil return nil
} }
// TearDownAll tears down all active nodes within the test lightning network. // TearDown tears down all active nodes within the test lightning network.
func (n *NetworkHarness) TearDownAll() error { func (n *NetworkHarness) TearDown() error {
for _, node := range n.activeNodes { for _, node := range n.activeNodes {
if err := n.ShutdownNode(node); err != nil { if err := n.ShutdownNode(node); err != nil {
return err return err
} }
} }
return nil
}
// Stop stops the test harness.
func (n *NetworkHarness) Stop() {
close(n.lndErrorChan) close(n.lndErrorChan)
close(n.quit) close(n.quit)
n.feeService.stop() n.feeService.stop()
return nil
} }
// NewNode fully initializes a returns a new HarnessNode bound to the // NewNode fully initializes a returns a new HarnessNode bound to the
@ -358,17 +364,18 @@ func (n *NetworkHarness) RestoreNodeWithSeed(name string, extraArgs []string,
// wallet with or without a seed. If hasSeed is false, the returned harness node // wallet with or without a seed. If hasSeed is false, the returned harness node
// can be used immediately. Otherwise, the node will require an additional // can be used immediately. Otherwise, the node will require an additional
// initialization phase where the wallet is either created or restored. // initialization phase where the wallet is either created or restored.
func (n *NetworkHarness) newNode(name string, extraArgs []string, func (n *NetworkHarness) newNode(name string, extraArgs []string, hasSeed bool,
hasSeed bool, password []byte) (*HarnessNode, error) { password []byte) (*HarnessNode, error) {
node, err := newNode(NodeConfig{ node, err := newNode(NodeConfig{
Name: name, Name: name,
HasSeed: hasSeed, LogFilenamePrefix: n.currentTestCase,
Password: password, HasSeed: hasSeed,
BackendCfg: n.BackendCfg, Password: password,
NetParams: n.netParams, BackendCfg: n.BackendCfg,
ExtraArgs: extraArgs, NetParams: n.netParams,
FeeURL: n.feeService.url, ExtraArgs: extraArgs,
FeeURL: n.feeService.url,
}) })
if err != nil { if err != nil {
return nil, err return nil, err

@ -14232,7 +14232,7 @@ func TestLightningNetworkDaemon(t *testing.T) {
if err != nil { if err != nil {
ht.Fatalf("unable to create lightning network harness: %v", err) ht.Fatalf("unable to create lightning network harness: %v", err)
} }
defer lndHarness.TearDownAll() defer lndHarness.Stop()
// Spawn a new goroutine to watch for any fatal errors that any of the // Spawn a new goroutine to watch for any fatal errors that any of the
// running lnd processes encounter. If an error occurs, then the test // running lnd processes encounter. If an error occurs, then the test
@ -14265,34 +14265,52 @@ func TestLightningNetworkDaemon(t *testing.T) {
aliceBobArgs := []string{ aliceBobArgs := []string{
"--default-remote-max-htlcs=483", "--default-remote-max-htlcs=483",
} }
if err = lndHarness.SetUp(aliceBobArgs); err != nil {
ht.Fatalf("unable to set up test lightning network: %v", err)
}
// Run the subset of the test cases selected in this tranche. // Run the subset of the test cases selected in this tranche.
for idx, testCase := range testCases { for idx, testCase := range testCases {
testCase := testCase testCase := testCase
logLine := fmt.Sprintf("STARTING ============ %v ============\n",
testCase.name)
err := lndHarness.EnsureConnected(
context.Background(), lndHarness.Alice, lndHarness.Bob,
)
require.NoError(t, err, "unable to connect alice to bob")
err = lndHarness.Alice.AddToLog(logLine)
require.NoError(t, err, "unable to add to Alice's log")
err = lndHarness.Bob.AddToLog(logLine)
require.NoError(t, err, "unable to add to Bob's log")
// Start every test with the default static fee estimate.
lndHarness.SetFeeEstimate(12500)
name := fmt.Sprintf("%02d-of-%d/%s/%s", name := fmt.Sprintf("%02d-of-%d/%s/%s",
trancheOffset+uint(idx)+1, len(allTestCases), trancheOffset+uint(idx)+1, len(allTestCases),
chainBackend.Name(), testCase.name) chainBackend.Name(), testCase.name)
success := t.Run(name, func(t1 *testing.T) { success := t.Run(name, func(t1 *testing.T) {
cleanTestCaseName := strings.ReplaceAll(
testCase.name, " ", "_",
)
err = lndHarness.SetUp(cleanTestCaseName, aliceBobArgs)
require.NoError(t1,
err, "unable to set up test lightning network",
)
defer func() {
require.NoError(t1, lndHarness.TearDown())
}()
err = lndHarness.EnsureConnected(
context.Background(), lndHarness.Alice,
lndHarness.Bob,
)
require.NoError(t1,
err, "unable to connect alice to bob",
)
logLine := fmt.Sprintf(
"STARTING ============ %v ============\n",
testCase.name,
)
err = lndHarness.Alice.AddToLog(logLine)
require.NoError(t1, err, "unable to add to log")
err = lndHarness.Bob.AddToLog(logLine)
require.NoError(t1, err, "unable to add to log")
// Start every test with the default static fee estimate.
lndHarness.SetFeeEstimate(12500)
// Create a separate harness test for the testcase to
// avoid overwriting the external harness test that is
// tied to the parent test.
ht := newHarnessTest(t1, lndHarness) ht := newHarnessTest(t1, lndHarness)
ht.RunTestCase(testCase) ht.RunTestCase(testCase)
}) })

@ -153,7 +153,12 @@ type BackendConfig interface {
} }
type NodeConfig struct { type NodeConfig struct {
Name string Name string
// LogFilenamePrefix is is used to prefix node log files. Can be used
// to store the current test case for simpler postmortem debugging.
LogFilenamePrefix string
BackendCfg BackendConfig BackendCfg BackendConfig
NetParams *chaincfg.Params NetParams *chaincfg.Params
BaseDir string BaseDir string
@ -493,17 +498,21 @@ func (hn *HarnessNode) start(lndBinary string, lndError chan<- error) error {
// log files. // log files.
if *logOutput { if *logOutput {
dir := GetLogDir() dir := GetLogDir()
fileName := fmt.Sprintf("%s/output-%d-%s-%s.log", dir, hn.NodeID, fileName := fmt.Sprintf("%s/%d-%s-%s-%s.log", dir, hn.NodeID,
hn.Cfg.Name, hex.EncodeToString(hn.PubKey[:logPubKeyBytes])) hn.Cfg.LogFilenamePrefix, hn.Cfg.Name,
hex.EncodeToString(hn.PubKey[:logPubKeyBytes]))
// If the node's PubKey is not yet initialized, create a temporary // If the node's PubKey is not yet initialized, create a
// file name. Later, after the PubKey has been initialized, the // temporary file name. Later, after the PubKey has been
// file can be moved to its final name with the PubKey included. // initialized, the file can be moved to its final name with
// the PubKey included.
if bytes.Equal(hn.PubKey[:4], []byte{0, 0, 0, 0}) { if bytes.Equal(hn.PubKey[:4], []byte{0, 0, 0, 0}) {
fileName = fmt.Sprintf("%s/output-%d-%s-tmp__.log", fileName = fmt.Sprintf("%s/%d-%s-%s-tmp__.log", dir,
dir, hn.NodeID, hn.Cfg.Name) hn.NodeID, hn.Cfg.LogFilenamePrefix,
hn.Cfg.Name)
// Once the node has done its work, the log file can be renamed. // Once the node has done its work, the log file can be
// renamed.
finalizeLogfile = func() { finalizeLogfile = func() {
if hn.logFile != nil { if hn.logFile != nil {
hn.logFile.Close() hn.logFile.Close()
@ -511,8 +520,10 @@ func (hn *HarnessNode) start(lndBinary string, lndError chan<- error) error {
pubKeyHex := hex.EncodeToString( pubKeyHex := hex.EncodeToString(
hn.PubKey[:logPubKeyBytes], hn.PubKey[:logPubKeyBytes],
) )
newFileName := fmt.Sprintf("%s/output"+ newFileName := fmt.Sprintf("%s/"+
"-%d-%s-%s.log", dir, hn.NodeID, "%d-%s-%s-%s.log",
dir, hn.NodeID,
hn.Cfg.LogFilenamePrefix,
hn.Cfg.Name, pubKeyHex) hn.Cfg.Name, pubKeyHex)
err := os.Rename(fileName, newFileName) err := os.Rename(fileName, newFileName)
if err != nil { if err != nil {