// Package helpers defines helper functions to peer into // end to end processes and kill processes as needed. package helpers import ( "bufio" "context" "fmt" "io" "io/ioutil" "net/http" "os" "path" "path/filepath" "strings" "testing" "time" e2e "github.com/prysmaticlabs/prysm/endtoend/params" "github.com/prysmaticlabs/prysm/endtoend/types" ) const ( maxPollingWaitTime = 60 * time.Second // A minute so timing out doesn't take very long. filePollingInterval = 500 * time.Millisecond memoryHeapFileName = "node_heap_%d.pb.gz" cpuProfileFileName = "node_cpu_profile_%d.pb.gz" fileBufferSize = 64 * 1024 maxFileBufferSize = 1024 * 1024 ) // KillProcesses finds the passed in process IDs and kills the process. func KillProcesses(t *testing.T, pIDs []int) { 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) } if _, err := process.Wait(); err != nil { t.Fatal(err) } } time.Sleep(5 * time.Second) } // 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) { 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 } // WaitForTextInFile checks a file every polling interval for the text requested. func WaitForTextInFile(file *os.File, text string) error { 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 } return fmt.Errorf("could not find requested text \"%s\" in logs:\n%s", text, contents) case <-ticker.C: fileScanner := bufio.NewScanner(file) buf := make([]byte, 0, fileBufferSize) fileScanner.Buffer(buf, maxFileBufferSize) for fileScanner.Scan() { scanned := fileScanner.Text() if strings.Contains(scanned, text) { return nil } } if err := fileScanner.Err(); err != nil { return err } _, err := file.Seek(0, io.SeekStart) if err != nil { return err } } } } // LogOutput logs the output of all log files made. func LogOutput(t *testing.T, config *types.E2EConfig) { // Log out errors from beacon chain nodes. for i := 0; i < e2e.TestParams.BeaconNodeCount; i++ { beaconLogFile, err := os.Open(path.Join(e2e.TestParams.LogPath, fmt.Sprintf(e2e.BeaconNodeLogFileName, i))) if err != nil { t.Fatal(err) } LogErrorOutput(t, beaconLogFile, "beacon chain node", i) validatorLogFile, err := os.Open(path.Join(e2e.TestParams.LogPath, fmt.Sprintf(e2e.ValidatorLogFileName, i))) if err != nil { t.Fatal(err) } LogErrorOutput(t, validatorLogFile, "validator client", i) if config.TestSlasher { slasherLogFile, err := os.Open(path.Join(e2e.TestParams.LogPath, fmt.Sprintf(e2e.SlasherLogFileName, i))) if err != nil { t.Fatal(err) } LogErrorOutput(t, slasherLogFile, "slasher client", i) } } t.Logf("Ending time: %s\n", time.Now().String()) } // LogErrorOutput logs the output of a specific file. func LogErrorOutput(t *testing.T, file io.Reader, title string, index int) { 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 } t.Logf("==================== Start of %s %d error output ==================\n", title, index) for _, err := range errorLines { t.Log(err) } } // WritePprofFiles writes the memory heap and cpu profile files to the test path. func WritePprofFiles(testDir string, index int) error { if err := os.MkdirAll(filepath.Join(testDir), os.ModePerm); err != nil { return err } url := fmt.Sprintf("http://127.0.0.1:%d/debug/pprof/heap", e2e.TestParams.BeaconNodeRPCPort+50+index) filePath := filepath.Join(testDir, fmt.Sprintf(memoryHeapFileName, index)) if err := writeURLRespAtPath(url, filePath); err != nil { return err } url = fmt.Sprintf("http://127.0.0.1:%d/debug/pprof/profile", e2e.TestParams.BeaconNodeRPCPort+50+index) filePath = filepath.Join(testDir, fmt.Sprintf(cpuProfileFileName, index)) if err := writeURLRespAtPath(url, filePath); err != nil { return err } return nil } func writeURLRespAtPath(url string, filePath string) error { resp, err := http.Get(url) if err != nil { return err } defer func() { if err := resp.Body.Close(); err != nil { return } }() body, err := ioutil.ReadAll(resp.Body) if err != nil { return err } file, err := os.Create(filePath) if err != nil { return err } if _, err := file.Write(body); err != nil { return err } return nil }