-
Notifications
You must be signed in to change notification settings - Fork 179
/
process_summary2_results.go
190 lines (156 loc) · 7.6 KB
/
process_summary2_results.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
package main
import (
"encoding/json"
"fmt"
"os"
"path/filepath"
"strings"
"github.com/onflow/flow-go/tools/flaky_test_monitor/common"
)
const failuresDir = "./failures/"
const noResultsDir = "./no-results/"
// process level 1 summary files in a single directory and output level 2 summary
func processSummary2TestRun(level1Directory string) common.TestSummary2 {
dirEntries, err := os.ReadDir(filepath.Join(level1Directory))
common.AssertNoError(err, "error reading level 1 directory")
// create directory to store failure messages
err = os.Mkdir(failuresDir, 0755)
common.AssertNoError(err, "error creating failures directory")
// create directory to store no-results messages
err = os.Mkdir(noResultsDir, 0755)
common.AssertNoError(err, "error creating no-results directory")
testSummary2 := common.TestSummary2{}
testSummary2.TestResults = make(map[string]*common.TestResultSummary)
// go through all level 1 summaries in a folder to create level 2 summary
for i := 0; i < len(dirEntries); i++ {
// read in each level 1 summary
var level1TestRun common.TestRun
level1JsonBytes, err := os.ReadFile(filepath.Join(level1Directory, dirEntries[i].Name()))
common.AssertNoError(err, "error reading level 1 json")
err = json.Unmarshal(level1JsonBytes, &level1TestRun)
common.AssertNoError(err, "error unmarshalling level 1 test run")
// go through each level 1 summary and update level 2 summary
for _, packageResult := range level1TestRun.PackageResults {
for _, testResult := range packageResult.Tests {
// check if already started collecting summary for this test
mapKey := testResult.Package + "/" + testResult.Test
testResultSummary, testResultSummaryExists := testSummary2.TestResults[mapKey]
// this test doesn't have a summary so create it
// no need to specify other fields explicitly - default values will suffice
if !testResultSummaryExists {
testResultSummary = &common.TestResultSummary{
Test: testResult.Test,
Package: testResult.Package,
}
}
// keep track of each duration so can later calculate average duration
testResultSummary.Durations = append(testResultSummary.Durations, testResult.Elapsed)
// increment # of passes, fails, skips or no-results for this test
testResultSummary.Runs++
switch testResult.Result {
case "pass":
testResultSummary.Passed++
case "fail":
testResultSummary.Failed++
saveFailureMessage(testResult)
case "skip":
testResultSummary.Skipped++
// don't count skip as a run
testResultSummary.Runs--
// truncate last duration - don't count durations of skips
testResultSummary.Durations = testResultSummary.Durations[:len(testResultSummary.Durations)-1]
case "":
testResultSummary.NoResult++
// don't count no result as a run
testResultSummary.Runs--
// truncate last duration - don't count durations of no-results
testResultSummary.Durations = testResultSummary.Durations[:len(testResultSummary.Durations)-1]
saveNoResultMessage(testResult)
default:
panic(fmt.Sprintf("unexpected test result: %s", testResult.Result))
}
testSummary2.TestResults[mapKey] = testResultSummary
}
}
}
// calculate averages and other calculations that can only be completed after all test runs have been read
postProcessTestSummary2(testSummary2)
return testSummary2
}
func saveFailureMessage(testResult common.TestResult) {
saveMessageHelper(testResult, "fail", failuresDir, "failure")
}
func saveNoResultMessage(testResult common.TestResult) {
saveMessageHelper(testResult, "", noResultsDir, "no-result")
}
// for each failed / no-result test, we want to save the raw output message as a text file
// there could be multiple failures / no-results of the same test so we want to save each failed / no-result message in a separate text file
// each test with failures / no-results will have a uniquely named (based on test name and package) sub-directory where failed / no-result messages are saved
// e.g. "failures/TestSanitySha3_256+github.com-onflow-flow-go-crypto-hash" will store failed messages text files
// from test TestSanitySha3_256 from the "github.com/onflow/flow-go/crypto/hash" package
// failure and no-result messages are saved in a similar way so this helper function
// handles saving both types of messages
func saveMessageHelper(testResult common.TestResult, expectedResult string, messagesDir string, messageFileStem string) {
if testResult.Result != expectedResult {
panic(fmt.Sprintf("unexpected test result: " + testResult.Result))
}
// each sub directory corresponds to a failed / no-result test name and package name
messagesDirFullPath := messagesDir + testResult.Test + "+" + strings.ReplaceAll(testResult.Package, "/", "-") + "/"
// there could already be previous failures / no-results for this test, so it's important
// to check if failed test / no-result folder exists
if !common.DirExists(messagesDirFullPath) {
err := os.Mkdir(messagesDirFullPath, 0755)
common.AssertNoError(err, "error creating sub-dir under failures / no-results dir")
}
// under each sub-directory, there should be 1 or more text files
// (failure1.txt / no-result1.txt, failure2.txt / no-result2.txt, etc)
// that holds the raw failure / no-result message for that test
dirEntries, err := os.ReadDir(messagesDirFullPath)
common.AssertNoError(err, "error reading sub-dir entries under failures / no-results dir")
// failure text files will be named "failure1.txt", "failure2.txt", etc
// no-result text files will be named "no-result1.txt", "no-result2.txt", etc
// need to know how many failure / no-result text files already exist in the sub-directory before creating the next one
messageFile, err := os.Create(messagesDirFullPath + fmt.Sprintf(messageFileStem+"%d.txt", len(dirEntries)+1))
common.AssertNoError(err, "error creating failure / no-result file")
defer messageFile.Close()
for _, output := range testResult.Output {
_, err = messageFile.WriteString(output)
common.AssertNoError(err, "error writing to failure / no-result file")
}
}
func postProcessTestSummary2(testSummary2 common.TestSummary2) {
for _, testResultSummary := range testSummary2.TestResults {
// calculate average duration for each test summary
var durationSum float32 = 0
for _, duration := range testResultSummary.Durations {
durationSum += duration
}
testResultSummary.AverageDuration = common.ConvertToNDecimalPlaces2(2, durationSum, testResultSummary.Runs)
// calculate failure rate for each test summary
testResultSummary.FailureRate = common.ConvertToNDecimalPlaces(2, testResultSummary.Failed, testResultSummary.Runs)
}
// check if there are no failures so can delete failures sub-directory
if common.IsDirEmpty(failuresDir) {
err := os.RemoveAll(failuresDir)
common.AssertNoError(err, "error removing failures directory")
}
// check if there are no no-result tests so can delete no-results sub-directory
if common.IsDirEmpty(noResultsDir) {
err := os.RemoveAll(noResultsDir)
common.AssertNoError(err, "error removing no-results directory")
}
}
// level 2 flaky test summary processor
// input: command line argument of directory where level 1 summary files exist
// output: level 2 summary json file that will be used as input for level 3 summary processor
func main() {
// need to pass in single argument of where level 1 summary files exist
if len(os.Args[1:]) != 1 {
panic("wrong number of arguments - expected single argument with directory of level 1 files")
}
if !common.DirExists(os.Args[1]) {
panic("directory doesn't exist")
}
testSummary2 := processSummary2TestRun(os.Args[1])
common.SaveToFile("level2-summary.json", testSummary2)
}