github.com/onflow/flow-go@v0.35.7-crescendo-preview.23-atree-inlining/tools/test_monitor/level1/process_summary1_results.go (about)

     1  package main
     2  
     3  import (
     4  	"bufio"
     5  	"encoding/json"
     6  	"fmt"
     7  	"os"
     8  
     9  	"github.com/onflow/flow-go/tools/test_monitor/common"
    10  	"github.com/onflow/flow-go/utils/unittest"
    11  )
    12  
    13  // ResultReader gives us the flexibility to read test results in multiple ways - from stdin (for production) and from a local file (for unit testing)
    14  type ResultReader interface {
    15  	getReader() *os.File
    16  	close()
    17  
    18  	// where to save results - will be different for unit tests vs production
    19  	getResultsFileName() string
    20  }
    21  
    22  type StdinResultReader struct {
    23  }
    24  
    25  // return reader for reading from stdin - for production
    26  func (stdinResultReader StdinResultReader) getReader() *os.File {
    27  	return os.Stdin
    28  }
    29  
    30  // nothing to close when reading from stdin
    31  func (stdinResultReader StdinResultReader) close() {
    32  }
    33  
    34  func (stdinResultReader StdinResultReader) getResultsFileName() string {
    35  	return os.Getenv("RESULTS_FILE")
    36  }
    37  
    38  func generateLevel1Summary(resultReader ResultReader) (common.Level1Summary, map[string]*common.SkippedTestEntry) {
    39  	reader := resultReader.getReader()
    40  	scanner := bufio.NewScanner(reader)
    41  
    42  	defer resultReader.close()
    43  
    44  	testResultMap := processTestRunLineByLine(scanner)
    45  
    46  	err := scanner.Err()
    47  	common.AssertNoError(err, "error returning EOF for scanner")
    48  
    49  	testRun, skippedTests := finalizeLevel1Summary(testResultMap)
    50  
    51  	return testRun, skippedTests
    52  }
    53  
    54  // Raw JSON result step from `go test -json` execution
    55  // Sequence of result steps (specified by Action value) per test:
    56  // 1. run (once)
    57  // 2. output (one to many)
    58  // 3. pause (zero or once) - for tests using t.Parallel()
    59  // 4. cont (zero or once) - for tests using t.Parallel()
    60  // 5. pass OR fail OR skip (once)
    61  func processTestRunLineByLine(scanner *bufio.Scanner) map[string][]*common.Level1TestResult {
    62  	// test map holds all the tests
    63  	testResultMap := make(map[string][]*common.Level1TestResult)
    64  
    65  	for scanner.Scan() {
    66  		var rawTestStep common.RawTestStep
    67  		err := json.Unmarshal(scanner.Bytes(), &rawTestStep)
    68  		common.AssertNoError(err, "error unmarshalling raw test step")
    69  
    70  		// each test name needs to be unique, so we add package name in case there are
    71  		// tests with the same name across different packages
    72  		testResultMapKey := rawTestStep.Package + "/" + rawTestStep.Test
    73  
    74  		// most raw test steps will have Test value - only package specific steps won't have a Test value
    75  		// we're not storing package specific data
    76  		if rawTestStep.Test != "" {
    77  
    78  			// "run" is the very first test step and it needs special treatment - to create all the data structures that will be used by subsequent test steps for the same test
    79  			if rawTestStep.Action == "run" {
    80  				var newTestResult common.Level1TestResult
    81  				newTestResult.Test = rawTestStep.Test
    82  				newTestResult.Package = rawTestStep.Package
    83  
    84  				// each test holds specific data, irrespective of what the test result
    85  				newTestResult.CommitDate = common.GetCommitDate()
    86  				newTestResult.JobRunDate = common.GetJobRunDate()
    87  				newTestResult.CommitSha = common.GetCommitSha()
    88  				newTestResult.RunID = common.GetRunID()
    89  
    90  				// store outputs as a slice of strings - that's how "go test -json" outputs each output string on a separate line
    91  				// for passing tests, there are usually 2 outputs for a passing test and more outputs for a failing test
    92  				newTestResult.Output = make([]string, 0)
    93  
    94  				// append to test result slice, whether it's the first or subsequent test result
    95  				testResultMap[testResultMapKey] = append(testResultMap[testResultMapKey], &newTestResult)
    96  				continue
    97  			}
    98  
    99  			lastTestResultIndex := len(testResultMap[testResultMapKey]) - 1
   100  			if lastTestResultIndex < 0 {
   101  				lastTestResultIndex = 0
   102  			}
   103  
   104  			testResults, ok := testResultMap[testResultMapKey]
   105  			if !ok {
   106  				panic(fmt.Sprintf("no test result for test %s", rawTestStep.Test))
   107  			}
   108  			lastTestResultPointer := testResults[lastTestResultIndex]
   109  
   110  			// subsequent raw json outputs will have different data about the test - whether it passed/failed, what the test output was, etc
   111  			switch rawTestStep.Action {
   112  			case "output":
   113  				// keep appending output to the test
   114  				lastTestResultPointer.Output = append(lastTestResultPointer.Output, rawTestStep.Output)
   115  
   116  			// we need to convert pass / fail result into a numerical value so it can averaged and tracked on a graph
   117  			// pass is counted as 1, fail is counted as a 0
   118  			case "pass":
   119  				lastTestResultPointer.Pass = 1
   120  				lastTestResultPointer.Action = "pass"
   121  				lastTestResultPointer.Elapsed = rawTestStep.Elapsed
   122  				lastTestResultPointer.Exception = 0
   123  
   124  			case "fail":
   125  				lastTestResultPointer.Pass = 0
   126  				lastTestResultPointer.Action = "fail"
   127  				lastTestResultPointer.Elapsed = rawTestStep.Elapsed
   128  				lastTestResultPointer.Exception = 0
   129  
   130  			// skipped tests will be removed after all test results are gathered,
   131  			// since it would be more complicated to remove it here
   132  			case "skip":
   133  				lastTestResultPointer.Action = "skip"
   134  				lastTestResultPointer.Elapsed = rawTestStep.Elapsed
   135  
   136  			case "pause", "cont":
   137  				// tests using t.Parallel() will have these values
   138  				// nothing to do - test will continue to run normally and have a pass/fail result at the end
   139  
   140  			default:
   141  				panic(fmt.Sprintf("unexpected action: %s", rawTestStep.Action))
   142  			}
   143  		}
   144  	}
   145  	return testResultMap
   146  }
   147  
   148  func parseSkipReason(output []string) (unittest.SkipReason, bool) {
   149  	// skip reason is usually in the last output line, except when there is
   150  	// output from a test suite tear down function
   151  	for i := len(output) - 2; i >= 0; i-- {
   152  		skipReason, ok := unittest.ParseSkipReason(output[i])
   153  		if ok {
   154  			return skipReason, true
   155  		}
   156  	}
   157  	return 0, false
   158  }
   159  
   160  func finalizeLevel1Summary(testResultMap map[string][]*common.Level1TestResult) (common.Level1Summary, map[string]*common.SkippedTestEntry) {
   161  	var level1Summary common.Level1Summary
   162  	skippedTests := make(map[string]*common.SkippedTestEntry)
   163  	trackSkippedTests := getSkippedTestFile() != ""
   164  	testCategory := getTestCategory()
   165  
   166  	for _, testResults := range testResultMap {
   167  		for _, testResult := range testResults {
   168  			if trackSkippedTests {
   169  				skippedTestEntry := &common.SkippedTestEntry{
   170  					Test:       testResult.Test,
   171  					Package:    testResult.Package,
   172  					CommitDate: testResult.CommitDate,
   173  					CommitSHA:  testResult.CommitSha,
   174  					Category:   testCategory,
   175  				}
   176  				skippedTests[testResult.Test] = skippedTestEntry
   177  
   178  				if testResult.Action == "skip" {
   179  					skipReason, ok := parseSkipReason(testResult.Output)
   180  					if ok {
   181  						skippedTestEntry.SkipReason = skipReason
   182  					} else {
   183  						panic("could not parse Skip Reason from output for test: " + testResult.Test)
   184  					}
   185  				}
   186  			}
   187  
   188  			if testResult.Action == "skip" {
   189  				// don't add skipped tests to summary since they can't be used to compute an average pass rate
   190  				continue
   191  			}
   192  
   193  			// for tests that don't have a result generated (e.g. using fmt.Printf() with no newline in a test)
   194  			// we want to highlight these tests in Grafana
   195  			// we do this by setting the Exception field to true, so we can filter by that field in Grafana
   196  			if testResult.Action == "" {
   197  				// count exception as a failure
   198  				testResult.Pass = 0
   199  				testResult.Exception = 1
   200  			}
   201  
   202  			testResult.Action = ""
   203  
   204  			// only save output for failed tests
   205  			if testResult.Pass == 1 {
   206  				testResult.Output = nil
   207  			}
   208  
   209  			// only include passed or failed tests - don't include skipped tests
   210  			// this is needed to have accurate Grafana metrics for average pass rate
   211  			level1Summary = append(level1Summary, *testResult)
   212  		}
   213  	}
   214  
   215  	return level1Summary, skippedTests
   216  }
   217  
   218  func getSkippedTestFile() string {
   219  	return os.Getenv("SKIPPED_TESTS_FILE")
   220  }
   221  
   222  func getTestCategory() string {
   223  	return os.Getenv("TEST_CATEGORY")
   224  }
   225  
   226  // level 1 flaky test summary processor
   227  // input: json formatted test results from `go test -json` from console (not file)
   228  // output: level 1 summary json file that will be used as input for level 2 summary processor
   229  func main() {
   230  	resultReader := StdinResultReader{}
   231  
   232  	testRun, skippedTestMap := generateLevel1Summary(resultReader)
   233  
   234  	resultsFile := resultReader.getResultsFileName()
   235  	if resultsFile != "" {
   236  		common.SaveLinesToFile(resultsFile, testRun)
   237  	}
   238  
   239  	skippedTestsFile := getSkippedTestFile()
   240  	if skippedTestsFile != "" {
   241  		var skippedTests []*common.SkippedTestEntry
   242  		for _, skippedTestEntry := range skippedTestMap {
   243  			skippedTests = append(skippedTests, skippedTestEntry)
   244  		}
   245  		common.SaveLinesToFile(skippedTestsFile, skippedTests)
   246  	}
   247  }