-
Notifications
You must be signed in to change notification settings - Fork 2
/
report.go
95 lines (82 loc) · 2.64 KB
/
report.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
package loaderbot
import (
"encoding/csv"
"fmt"
"os"
"path"
"strconv"
"time"
"github.com/google/uuid"
)
type Report struct {
runId string
runName string
requestsLogFilename string
percsReportFilename string
percLogFilename string
requestsLogFile *csv.Writer
percLogFile *csv.Writer
reportOptions *ReportOptions
L *Logger
}
func NewReport(cfg *RunnerConfig) *Report {
_ = os.Mkdir(cfg.ReportOptions.CSVDir, os.ModePerm)
_ = os.Mkdir(cfg.ReportOptions.HTMLDir, os.ModePerm)
tn := time.Now().Unix()
runId := uuid.New().String()
requestsLogFilename := fmt.Sprintf(MetricsLogFile, cfg.Name, runId, tn)
requestsLogFilename = path.Join(cfg.ReportOptions.CSVDir, requestsLogFilename)
percLogFilename := fmt.Sprintf(PercsLogFile, cfg.Name, runId, tn)
percLogFilename = path.Join(cfg.ReportOptions.CSVDir, percLogFilename)
percsReportFilename := fmt.Sprintf(ReportGraphFile, cfg.Name, runId, tn)
percsReportFilename = path.Join(cfg.ReportOptions.HTMLDir, percsReportFilename)
r := &Report{
runId: runId,
runName: cfg.Name,
requestsLogFilename: requestsLogFilename,
percsReportFilename: percsReportFilename,
percLogFilename: percLogFilename,
requestsLogFile: csv.NewWriter(CreateFileOrReplace(requestsLogFilename)),
percLogFile: csv.NewWriter(CreateFileOrReplace(percLogFilename)),
reportOptions: cfg.ReportOptions,
L: NewLogger(cfg).With("report", cfg.Name),
}
_ = r.requestsLogFile.Write(ResultsCsvHeader)
_ = r.percLogFile.Write(PercsCsvHeader)
return r
}
func (r *Report) plot() {
if r.reportOptions.PNG {
r.L.Infof("reporting graphs: %s", r.percLogFilename)
chart, err := PercsChart(r.percLogFilename, r.runName)
if err != nil {
r.L.Error(err)
return
}
RenderEChart(chart, r.percsReportFilename)
}
}
func (r *Report) flushLogs() {
r.percLogFile.Flush()
r.requestsLogFile.Flush()
}
func (r *Report) writeResultEntry(res AttackResult, errorMsg string) {
_ = r.requestsLogFile.Write([]string{
res.DoResult.RequestLabel,
strconv.Itoa(int(res.Begin.UnixNano())),
strconv.Itoa(int(res.End.UnixNano())),
res.Elapsed.String(),
string(res.DoResult.StatusCode),
errorMsg,
})
}
func (r *Report) writePercentilesEntry(res AttackResult, tickMetrics *Metrics) {
_ = r.percLogFile.Write([]string{
res.DoResult.RequestLabel,
strconv.Itoa(res.AttackToken.Tick),
strconv.Itoa(int(tickMetrics.Rate)),
strconv.Itoa(int(tickMetrics.Latencies.P50.Milliseconds())),
strconv.Itoa(int(tickMetrics.Latencies.P95.Milliseconds())),
strconv.Itoa(int(tickMetrics.Latencies.P99.Milliseconds())),
})
}