mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2025-01-21 22:58:32 +00:00
82fd65665f
bidichk checks for dangerous unicode character sequences (https://github.com/golangci/golangci-lint/pull/2330)
116 lines
2.5 KiB
Go
116 lines
2.5 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
|
|
log logutils.Log
|
|
|
|
stages map[string]time.Duration
|
|
mu 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 := make([]stageDuration, 0, len(s.stages))
|
|
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 := make([]string, 0, len(stageDurations))
|
|
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.mu.Lock()
|
|
s.stages[name] += time.Since(startedAt)
|
|
s.mu.Unlock()
|
|
}
|