2023-10-22 01:28:28 +02:00
|
|
|
// Exports test metrics to DD so we can monitor for flaky tests over time
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2023-12-22 17:04:10 +01:00
|
|
|
"bufio"
|
2023-10-22 01:28:28 +02:00
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"os"
|
2023-12-22 17:04:10 +01:00
|
|
|
"path"
|
2024-02-11 20:54:27 +01:00
|
|
|
"path/filepath"
|
2023-10-22 01:28:28 +02:00
|
|
|
"runtime"
|
2023-12-22 17:04:10 +01:00
|
|
|
"slices"
|
|
|
|
"strings"
|
2023-10-22 01:28:28 +02:00
|
|
|
|
|
|
|
"github.com/DataDog/datadog-go/statsd"
|
|
|
|
"gotest.tools/gotestsum/testjson"
|
|
|
|
)
|
|
|
|
|
2023-10-22 21:01:51 +02:00
|
|
|
var GLOBAL_STATSD *statsd.Client = nil
|
2023-10-22 01:28:28 +02:00
|
|
|
|
|
|
|
var NUM_TEST_RETRIES map[string]int
|
|
|
|
|
2024-08-11 21:19:41 +02:00
|
|
|
var UNUSED_GOLDENS []string = []string{
|
|
|
|
"testCustomColumns-query-isAction=false", "testCustomColumns-tquery-bash",
|
|
|
|
"testCustomColumns-tquery-zsh",
|
|
|
|
}
|
2023-12-22 17:04:10 +01:00
|
|
|
|
2023-10-22 01:28:28 +02:00
|
|
|
func main() {
|
2024-02-11 20:54:27 +01:00
|
|
|
if os.Args[1] == "export" {
|
|
|
|
exportMetrics()
|
|
|
|
}
|
|
|
|
if os.Args[1] == "check-goldens" {
|
|
|
|
checkGoldensUsed()
|
|
|
|
}
|
2023-12-22 17:04:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func checkGoldensUsed() {
|
|
|
|
if os.Getenv("HISHTORY_FILTERED_TEST") != "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// Read the goldens that were used
|
|
|
|
usedGoldens := make([]string, 0)
|
2024-02-11 20:54:27 +01:00
|
|
|
filenames, err := filepath.Glob("*/goldens-used.txt")
|
2023-12-22 17:04:10 +01:00
|
|
|
if err != nil {
|
2024-02-11 20:54:27 +01:00
|
|
|
log.Fatalf("failed to list golden files: %v", err)
|
2023-12-22 17:04:10 +01:00
|
|
|
}
|
2024-02-11 20:54:27 +01:00
|
|
|
fmt.Printf("Found used goldens in %#v\n", filenames)
|
|
|
|
for _, filename := range filenames {
|
|
|
|
usedGoldensFile, err := os.Open(filename)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("failed to open %s: %v", filename, err)
|
|
|
|
}
|
|
|
|
defer usedGoldensFile.Close()
|
|
|
|
scanner := bufio.NewScanner(usedGoldensFile)
|
|
|
|
for scanner.Scan() {
|
|
|
|
usedGoldens = append(usedGoldens, strings.TrimSpace(scanner.Text()))
|
|
|
|
}
|
|
|
|
if err := scanner.Err(); err != nil {
|
|
|
|
log.Fatalf("failed to read lines from /tmp/goldens-used.txt: %v", err)
|
|
|
|
}
|
2023-12-22 17:04:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// List all the goldens that exist
|
|
|
|
goldensDir := "client/testdata/"
|
|
|
|
files, err := os.ReadDir(goldensDir)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Errorf("failed to list files in %s: %w", goldensDir, err))
|
|
|
|
}
|
|
|
|
|
|
|
|
// And check for mismatches
|
2023-12-29 10:16:33 +01:00
|
|
|
var unusedGoldenErr error = nil
|
2023-12-22 17:04:10 +01:00
|
|
|
for _, f := range files {
|
|
|
|
goldenName := path.Base(f.Name())
|
2023-12-29 04:52:39 +01:00
|
|
|
if !slices.Contains(usedGoldens, goldenName) {
|
|
|
|
if slices.Contains(UNUSED_GOLDENS, goldenName) {
|
|
|
|
// It is allowlisted to not be used
|
|
|
|
continue
|
|
|
|
}
|
2023-12-29 10:16:33 +01:00
|
|
|
unusedGoldenErr = fmt.Errorf("golden file %v was never used", goldenName)
|
|
|
|
fmt.Println(unusedGoldenErr)
|
2023-12-22 17:04:10 +01:00
|
|
|
}
|
|
|
|
}
|
2023-12-29 10:16:33 +01:00
|
|
|
if unusedGoldenErr != nil {
|
|
|
|
log.Fatalf("%v", unusedGoldenErr)
|
|
|
|
}
|
2023-12-29 04:45:59 +01:00
|
|
|
|
|
|
|
// And print out anything that is in UNUSED_GOLDENS that was actually used, so we
|
|
|
|
// can manually trim UNUSED_GOLDENS
|
|
|
|
for _, g := range UNUSED_GOLDENS {
|
|
|
|
if slices.Contains(usedGoldens, g) {
|
|
|
|
fmt.Printf("Golden %s is in UNUSED_GOLDENS, but was actually used\n", g)
|
|
|
|
}
|
|
|
|
}
|
2023-12-22 17:04:10 +01:00
|
|
|
fmt.Println("Validated that all goldens in testdata/ were referenced!")
|
|
|
|
}
|
|
|
|
|
|
|
|
func exportMetrics() {
|
2023-10-22 01:28:28 +02:00
|
|
|
// Configure Datadog
|
2023-10-22 21:01:51 +02:00
|
|
|
if _, has_dd_api_key := os.LookupEnv("DD_API_KEY"); has_dd_api_key {
|
|
|
|
ddStats, err := statsd.New("localhost:8125")
|
|
|
|
if err != nil {
|
|
|
|
err := fmt.Errorf("failed to start DataDog statsd: %w", err)
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
fmt.Printf("failed to init datadog: %v", err)
|
|
|
|
os.Exit(0)
|
|
|
|
} else {
|
|
|
|
log.Fatalf("failed to init datadog: %v", err)
|
|
|
|
}
|
2023-10-22 01:28:28 +02:00
|
|
|
}
|
2023-10-24 04:26:25 +02:00
|
|
|
defer ddStats.Close()
|
2023-10-22 21:01:51 +02:00
|
|
|
GLOBAL_STATSD = ddStats
|
|
|
|
} else {
|
|
|
|
fmt.Printf("Skipping exporting test stats to datadog\n")
|
2023-10-22 01:28:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Parse the test output
|
|
|
|
NUM_TEST_RETRIES = make(map[string]int)
|
|
|
|
inputFile, err := os.Open("/tmp/testrun.json")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("failed to open test input file: %v", err)
|
|
|
|
}
|
|
|
|
_, err = testjson.ScanTestOutput(testjson.ScanConfig{
|
|
|
|
Stdout: inputFile,
|
|
|
|
Handler: eventHandler{},
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("failed to scan testjson: %v", err)
|
|
|
|
}
|
|
|
|
for testId, count := range NUM_TEST_RETRIES {
|
|
|
|
GLOBAL_STATSD.Distribution("test_retry_count", float64(count), []string{"test:" + testId, "os:" + runtime.GOOS}, 1.0)
|
|
|
|
}
|
2023-10-22 20:14:25 +02:00
|
|
|
if GLOBAL_STATSD == nil {
|
|
|
|
fmt.Printf("Skipped uploading data about %d tests to datadog because GLOBAL_STATSD==nil\n", len(NUM_TEST_RETRIES))
|
|
|
|
} else {
|
2023-10-23 03:10:49 +02:00
|
|
|
err := GLOBAL_STATSD.Flush()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("failed to flush metrics: %v", err)
|
|
|
|
}
|
2023-10-22 20:14:25 +02:00
|
|
|
fmt.Printf("Uploaded data about %d tests to datadog\n", len(NUM_TEST_RETRIES))
|
|
|
|
}
|
2023-10-22 01:28:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type eventHandler struct{}
|
|
|
|
|
|
|
|
func (eventHandler) Event(event testjson.TestEvent, execution *testjson.Execution) error {
|
2023-10-23 05:57:21 +02:00
|
|
|
testIdentifier := event.Test
|
2023-10-22 01:28:28 +02:00
|
|
|
if event.Action == testjson.ActionFail {
|
2023-11-19 10:25:48 +01:00
|
|
|
fmt.Println("Recorded failure for " + testIdentifier)
|
2023-10-22 01:28:28 +02:00
|
|
|
GLOBAL_STATSD.Incr("test_status", []string{"result:failed", "test:" + testIdentifier, "os:" + runtime.GOOS}, 1.0)
|
|
|
|
NUM_TEST_RETRIES[testIdentifier] += 1
|
|
|
|
}
|
|
|
|
if event.Action == testjson.ActionPass {
|
2023-10-22 01:42:41 +02:00
|
|
|
GLOBAL_STATSD.Distribution("test_runtime", event.Elapsed, []string{"test:" + testIdentifier, "os:" + runtime.GOOS}, 1.0)
|
2023-10-22 01:28:28 +02:00
|
|
|
GLOBAL_STATSD.Incr("test_status", []string{"result:passed", "test:" + testIdentifier, "os:" + runtime.GOOS}, 1.0)
|
|
|
|
NUM_TEST_RETRIES[testIdentifier] += 1
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (eventHandler) Err(text string) error {
|
|
|
|
return fmt.Errorf("unexpected error when parsing test output: %v", text)
|
|
|
|
}
|