2020-04-29 21:32:39 +00:00
|
|
|
// Package helpers defines helper functions to peer into
|
|
|
|
// end to end processes and kill processes as needed.
|
2020-03-22 23:04:23 +00:00
|
|
|
package helpers
|
2020-01-28 19:16:00 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
2020-03-22 23:04:23 +00:00
|
|
|
|
|
|
|
e2e "github.com/prysmaticlabs/prysm/endtoend/params"
|
|
|
|
"github.com/prysmaticlabs/prysm/endtoend/types"
|
2020-01-28 19:16:00 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2020-05-26 19:04:42 +00:00
|
|
|
maxPollingWaitTime = 60 * time.Second // A minute so timing out doesn't take very long.
|
2020-03-21 06:42:51 +00:00
|
|
|
filePollingInterval = 500 * time.Millisecond
|
2020-01-28 19:16:00 +00:00
|
|
|
)
|
|
|
|
|
2020-03-22 23:04:23 +00:00
|
|
|
// KillProcesses finds the passed in process IDs and kills the process.
|
|
|
|
func KillProcesses(t *testing.T, pIDs []int) {
|
2020-01-28 19:16:00 +00:00
|
|
|
for _, id := range pIDs {
|
|
|
|
process, err := os.FindProcess(id)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not find process %d: %v", id, err)
|
|
|
|
}
|
|
|
|
if err := process.Kill(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2020-03-31 15:15:33 +00:00
|
|
|
if _, err := process.Wait(); err != nil {
|
2020-01-28 19:16:00 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-22 23:04:23 +00:00
|
|
|
// DeleteAndCreateFile checks if the file path given exists, if it does, it deletes it and creates a new file.
|
|
|
|
// If not, it just creates the requested file.
|
|
|
|
func DeleteAndCreateFile(tmpPath string, fileName string) (*os.File, error) {
|
2020-01-31 00:16:36 +00:00
|
|
|
filePath := path.Join(tmpPath, fileName)
|
|
|
|
if _, err := os.Stat(filePath); os.IsExist(err) {
|
|
|
|
if err := os.Remove(filePath); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
newFile, err := os.Create(path.Join(tmpPath, fileName))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return newFile, nil
|
|
|
|
}
|
|
|
|
|
2020-03-22 23:04:23 +00:00
|
|
|
// WaitForTextInFile checks a file every polling interval for the text requested.
|
|
|
|
func WaitForTextInFile(file *os.File, text string) error {
|
2020-01-28 19:16:00 +00:00
|
|
|
d := time.Now().Add(maxPollingWaitTime)
|
|
|
|
ctx, cancel := context.WithDeadline(context.Background(), d)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
// Use a ticker with a deadline to poll a given file.
|
|
|
|
ticker := time.NewTicker(filePollingInterval)
|
|
|
|
defer ticker.Stop()
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
contents, err := ioutil.ReadAll(file)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-31 00:16:36 +00:00
|
|
|
return fmt.Errorf("could not find requested text \"%s\" in logs:\n%s", text, contents)
|
2020-01-28 19:16:00 +00:00
|
|
|
case <-ticker.C:
|
|
|
|
fileScanner := bufio.NewScanner(file)
|
|
|
|
for fileScanner.Scan() {
|
|
|
|
scanned := fileScanner.Text()
|
|
|
|
if strings.Contains(scanned, text) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err := fileScanner.Err(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-31 00:16:36 +00:00
|
|
|
_, err := file.Seek(0, io.SeekStart)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-28 19:16:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-22 23:04:23 +00:00
|
|
|
// LogOutput logs the output of all log files made.
|
|
|
|
func LogOutput(t *testing.T, config *types.E2EConfig) {
|
2020-01-28 19:16:00 +00:00
|
|
|
// Log out errors from beacon chain nodes.
|
2020-03-22 23:04:23 +00:00
|
|
|
for i := 0; i < e2e.TestParams.BeaconNodeCount; i++ {
|
|
|
|
beaconLogFile, err := os.Open(path.Join(e2e.TestParams.LogPath, fmt.Sprintf(e2e.BeaconNodeLogFileName, i)))
|
2020-01-28 19:16:00 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2020-03-22 23:04:23 +00:00
|
|
|
LogErrorOutput(t, beaconLogFile, "beacon chain node", i)
|
2020-01-28 19:16:00 +00:00
|
|
|
|
2020-03-22 23:04:23 +00:00
|
|
|
validatorLogFile, err := os.Open(path.Join(e2e.TestParams.LogPath, fmt.Sprintf(e2e.ValidatorLogFileName, i)))
|
2020-01-28 19:16:00 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2020-03-22 23:04:23 +00:00
|
|
|
LogErrorOutput(t, validatorLogFile, "validator client", i)
|
2020-03-15 05:09:23 +00:00
|
|
|
|
2020-03-22 23:04:23 +00:00
|
|
|
if config.TestSlasher {
|
|
|
|
slasherLogFile, err := os.Open(path.Join(e2e.TestParams.LogPath, fmt.Sprintf(e2e.SlasherLogFileName, i)))
|
2020-03-21 06:42:51 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2020-03-22 23:04:23 +00:00
|
|
|
LogErrorOutput(t, slasherLogFile, "slasher client", i)
|
2020-03-15 05:09:23 +00:00
|
|
|
}
|
2020-01-28 19:16:00 +00:00
|
|
|
}
|
|
|
|
t.Logf("Ending time: %s\n", time.Now().String())
|
|
|
|
}
|
|
|
|
|
2020-03-22 23:04:23 +00:00
|
|
|
// LogErrorOutput logs the output of a specific file.
|
|
|
|
func LogErrorOutput(t *testing.T, file io.Reader, title string, index int) {
|
2020-01-28 19:16:00 +00:00
|
|
|
var errorLines []string
|
|
|
|
|
|
|
|
scanner := bufio.NewScanner(file)
|
|
|
|
for scanner.Scan() {
|
|
|
|
currentLine := scanner.Text()
|
|
|
|
if strings.Contains(currentLine, "level=error") {
|
|
|
|
errorLines = append(errorLines, currentLine)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(errorLines) < 1 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-03-15 05:09:23 +00:00
|
|
|
t.Logf("==================== Start of %s %d error output ==================\n", title, index)
|
|
|
|
var lines uint64
|
2020-01-28 19:16:00 +00:00
|
|
|
for _, err := range errorLines {
|
2020-03-15 05:09:23 +00:00
|
|
|
lines++
|
|
|
|
if lines >= 10 {
|
|
|
|
break
|
|
|
|
}
|
2020-01-28 19:16:00 +00:00
|
|
|
t.Log(err)
|
|
|
|
}
|
|
|
|
}
|