mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2025-01-05 23:28:42 +00:00
117 lines
2.4 KiB
Go
117 lines
2.4 KiB
Go
|
package timeutils
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"sort"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
"time"
|
||
|
|
||
|
"github.com/golangci/golangci-lint/pkg/logutils"
|
||
|
)
|
||
|
|
||
|
const noStagesText = "no stages"
|
||
|
|
||
|
type Stopwatch struct {
|
||
|
name string
|
||
|
startedAt time.Time
|
||
|
stages map[string]time.Duration
|
||
|
log logutils.Log
|
||
|
|
||
|
sync.Mutex
|
||
|
}
|
||
|
|
||
|
func NewStopwatch(name string, log logutils.Log) *Stopwatch {
|
||
|
return &Stopwatch{
|
||
|
name: name,
|
||
|
startedAt: time.Now(),
|
||
|
stages: map[string]time.Duration{},
|
||
|
log: log,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type stageDuration struct {
|
||
|
name string
|
||
|
d time.Duration
|
||
|
}
|
||
|
|
||
|
func (s *Stopwatch) stageDurationsSorted() []stageDuration {
|
||
|
stageDurations := []stageDuration{}
|
||
|
for n, d := range s.stages {
|
||
|
stageDurations = append(stageDurations, stageDuration{
|
||
|
name: n,
|
||
|
d: d,
|
||
|
})
|
||
|
}
|
||
|
sort.Slice(stageDurations, func(i, j int) bool {
|
||
|
return stageDurations[i].d > stageDurations[j].d
|
||
|
})
|
||
|
return stageDurations
|
||
|
}
|
||
|
|
||
|
func (s *Stopwatch) sprintStages() string {
|
||
|
if len(s.stages) == 0 {
|
||
|
return noStagesText
|
||
|
}
|
||
|
|
||
|
stageDurations := s.stageDurationsSorted()
|
||
|
|
||
|
stagesStrings := []string{}
|
||
|
for _, s := range stageDurations {
|
||
|
stagesStrings = append(stagesStrings, fmt.Sprintf("%s: %s", s.name, s.d))
|
||
|
}
|
||
|
|
||
|
return fmt.Sprintf("stages: %s", strings.Join(stagesStrings, ", "))
|
||
|
}
|
||
|
|
||
|
func (s *Stopwatch) sprintTopStages(n int) string {
|
||
|
if len(s.stages) == 0 {
|
||
|
return noStagesText
|
||
|
}
|
||
|
|
||
|
stageDurations := s.stageDurationsSorted()
|
||
|
|
||
|
stagesStrings := []string{}
|
||
|
for i := 0; i < len(stageDurations) && i < n; i++ {
|
||
|
s := stageDurations[i]
|
||
|
stagesStrings = append(stagesStrings, fmt.Sprintf("%s: %s", s.name, s.d))
|
||
|
}
|
||
|
|
||
|
return fmt.Sprintf("top %d stages: %s", n, strings.Join(stagesStrings, ", "))
|
||
|
}
|
||
|
|
||
|
func (s *Stopwatch) Print() {
|
||
|
p := fmt.Sprintf("%s took %s", s.name, time.Since(s.startedAt))
|
||
|
if len(s.stages) == 0 {
|
||
|
s.log.Infof("%s", p)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
s.log.Infof("%s with %s", p, s.sprintStages())
|
||
|
}
|
||
|
|
||
|
func (s *Stopwatch) PrintStages() {
|
||
|
var stagesDuration time.Duration
|
||
|
for _, s := range s.stages {
|
||
|
stagesDuration += s
|
||
|
}
|
||
|
s.log.Infof("%s took %s with %s", s.name, stagesDuration, s.sprintStages())
|
||
|
}
|
||
|
|
||
|
func (s *Stopwatch) PrintTopStages(n int) {
|
||
|
var stagesDuration time.Duration
|
||
|
for _, s := range s.stages {
|
||
|
stagesDuration += s
|
||
|
}
|
||
|
s.log.Infof("%s took %s with %s", s.name, stagesDuration, s.sprintTopStages(n))
|
||
|
}
|
||
|
|
||
|
func (s *Stopwatch) TrackStage(name string, f func()) {
|
||
|
startedAt := time.Now()
|
||
|
f()
|
||
|
|
||
|
s.Lock()
|
||
|
s.stages[name] += time.Since(startedAt)
|
||
|
s.Unlock()
|
||
|
}
|