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

     1  // Package common has helper / utility functions used by all levels of flaky test monitor.
     2  package common
     3  
     4  import (
     5  	"encoding/json"
     6  	"errors"
     7  	"fmt"
     8  	"io"
     9  	"io/fs"
    10  	"os"
    11  	"reflect"
    12  	"strconv"
    13  	"time"
    14  )
    15  
    16  // AssertNoError checks that the passed in error is nil and panics
    17  // with the supplied message if that's not the case.
    18  // Useful helper to eliminate the need to keep checking for errors.
    19  func AssertNoError(err error, panicMessage string) {
    20  	if err != nil {
    21  		panic(panicMessage + ": " + err.Error())
    22  	}
    23  }
    24  
    25  // ConvertToNDecimalPlaces2 converts the supplied numerator and denominator fraction into
    26  // a decimal with n decimal places. Works the same way as ConvertToNDecimalPlaces()
    27  // but has a float for the numerator.
    28  func ConvertToNDecimalPlaces2(n int, numerator float64, denominator int) float64 {
    29  	return convertToNDecimalPlacesInternal(n, numerator, float64(denominator))
    30  }
    31  
    32  // ConvertToNDecimalPlaces converts the supplied numerator and denominator fraction into
    33  // a decimal with n decimal places.
    34  func ConvertToNDecimalPlaces(n int, numerator, denominator int) float64 {
    35  	return convertToNDecimalPlacesInternal(n, float64(numerator), float64(denominator))
    36  }
    37  
    38  func convertToNDecimalPlacesInternal(n int, numerator, denominator float64) float64 {
    39  	if numerator == 0 || denominator == 0 {
    40  		return 0
    41  	}
    42  	formatSpecifier := "%." + fmt.Sprint(n) + "f"
    43  	ratioString := fmt.Sprintf(formatSpecifier, numerator/denominator)
    44  	ratioFloat, err := strconv.ParseFloat(ratioString, 32)
    45  	AssertNoError(err, "failure parsing string to float")
    46  	return float64(ratioFloat)
    47  }
    48  
    49  func GetCommitSha() string {
    50  	commitSha := os.Getenv("COMMIT_SHA")
    51  	if commitSha == "" {
    52  		panic("COMMIT_SHA can't be empty")
    53  	}
    54  	return commitSha
    55  }
    56  
    57  func GetRunID() string {
    58  	runID := os.Getenv("RUN_ID")
    59  	if runID == "" {
    60  		panic("RUN_ID can't be empty")
    61  	}
    62  	return runID
    63  }
    64  
    65  func GetCommitDate() time.Time {
    66  	commitDate, err := time.Parse(time.RFC3339, os.Getenv("COMMIT_DATE"))
    67  	AssertNoError(err, "error parsing COMMIT_DATE")
    68  	return commitDate
    69  }
    70  
    71  func GetJobRunDate() time.Time {
    72  	jobStarted, err := time.Parse(time.RFC3339, os.Getenv("JOB_STARTED"))
    73  	AssertNoError(err, "error parsing JOB_STARTED")
    74  	return jobStarted
    75  }
    76  
    77  // IsDirEmpty checks if directory is empty (has no files) and return true if it's empty, false otherwise.
    78  // Useful for determining whether to delete the failures / exceptions directories
    79  // for cases when there were no failures / exceptions.
    80  // From https://stackoverflow.com/a/30708914/5719544.
    81  func IsDirEmpty(name string) bool {
    82  	f, err := os.Open(name)
    83  	AssertNoError(err, "error reading directory")
    84  
    85  	defer f.Close()
    86  
    87  	_, err = f.Readdirnames(1)
    88  	if err == io.EOF {
    89  		return true
    90  	}
    91  	AssertNoError(err, "error reading dir contents")
    92  	return false
    93  }
    94  
    95  // DirExists checks if directory exists and return true if it does, false otherwise.
    96  func DirExists(path string) bool {
    97  	_, err := os.Stat(path)
    98  
    99  	// directory exists if there is no error
   100  	if err == nil {
   101  		return true
   102  	}
   103  
   104  	// directory doesn't exist if error is of specific type
   105  	if errors.Is(err, fs.ErrNotExist) {
   106  		return false
   107  	}
   108  
   109  	// should never get to here
   110  	panic("error checking if directory exists:" + err.Error())
   111  }
   112  
   113  // SaveToFile save test run/summary to local JSON file.
   114  func SaveToFile(fileName string, testSummary interface{}) {
   115  	testSummaryBytes, err := json.MarshalIndent(testSummary, "", "  ")
   116  	AssertNoError(err, "error marshalling json")
   117  
   118  	file, err := os.Create(fileName)
   119  	AssertNoError(err, "error creating "+fileName)
   120  	defer file.Close()
   121  
   122  	_, err = file.Write(testSummaryBytes)
   123  	AssertNoError(err, "error saving test summary to file")
   124  }
   125  
   126  func SaveLinesToFile(fileName string, list interface{}) {
   127  	sliceType := reflect.TypeOf(list)
   128  	if sliceType.Kind() != reflect.Slice && sliceType.Kind() != reflect.Array {
   129  		panic("argument must be an array or slice")
   130  	}
   131  
   132  	file, err := os.Create(fileName)
   133  	AssertNoError(err, "error creating "+fileName)
   134  	defer file.Close()
   135  
   136  	l := reflect.ValueOf(list)
   137  	for i := 0; i < l.Len(); i++ {
   138  		b, err := json.Marshal(l.Index(i).Interface())
   139  		AssertNoError(err, "error marshalling json")
   140  
   141  		_, err = file.Write(b)
   142  		AssertNoError(err, "error writing line to file")
   143  
   144  		_, err = file.Write([]byte("\n"))
   145  		AssertNoError(err, "error writing newline")
   146  	}
   147  }