2020-12-06 13:03:40 -08:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2020-12-10 07:02:45 -08:00
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
|
|
|
|
lang "golang.org/x/text/language"
|
|
|
|
"golang.org/x/text/message"
|
2020-12-06 13:03:40 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
type executiontime struct {
|
|
|
|
props *properties
|
|
|
|
env environmentInfo
|
|
|
|
output string
|
|
|
|
}
|
|
|
|
|
2020-12-12 03:53:53 -08:00
|
|
|
// DurationStyle how to display the time
|
2020-12-10 07:02:45 -08:00
|
|
|
type DurationStyle string
|
|
|
|
|
2020-12-06 13:03:40 -08:00
|
|
|
const (
|
|
|
|
// ThresholdProperty represents minimum duration (milliseconds) required to enable this segment
|
|
|
|
ThresholdProperty Property = "threshold"
|
2020-12-12 03:53:53 -08:00
|
|
|
// Austin milliseconds short
|
|
|
|
Austin DurationStyle = "austin"
|
|
|
|
// Roundrock milliseconds long
|
2020-12-10 07:02:45 -08:00
|
|
|
Roundrock DurationStyle = "roundrock"
|
2020-12-12 03:53:53 -08:00
|
|
|
// Dallas milliseconds full
|
|
|
|
Dallas DurationStyle = "dallas"
|
|
|
|
// Galveston hour
|
2020-12-10 07:02:45 -08:00
|
|
|
Galveston DurationStyle = "galveston"
|
2020-12-12 03:53:53 -08:00
|
|
|
// Houston hour and milliseconds
|
|
|
|
Houston DurationStyle = "houston"
|
|
|
|
// Amarillo seconds
|
|
|
|
Amarillo DurationStyle = "amarillo"
|
2021-05-20 10:53:42 -07:00
|
|
|
// Round will round the output of the format
|
|
|
|
Round DurationStyle = "round"
|
2020-12-10 07:02:45 -08:00
|
|
|
|
|
|
|
second = 1000
|
|
|
|
minute = 60000
|
|
|
|
hour = 3600000
|
|
|
|
day = 86400000
|
|
|
|
secondsPerMinute = 60
|
|
|
|
minutesPerHour = 60
|
|
|
|
hoursPerDay = 24
|
2020-12-06 13:03:40 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
func (t *executiontime) enabled() bool {
|
2020-12-12 03:53:06 -08:00
|
|
|
alwaysEnabled := t.props.getBool(AlwaysEnabled, false)
|
2020-12-06 13:03:40 -08:00
|
|
|
executionTimeMs := t.env.executionTime()
|
|
|
|
thresholdMs := t.props.getFloat64(ThresholdProperty, float64(500))
|
2020-12-12 03:53:06 -08:00
|
|
|
if !alwaysEnabled && executionTimeMs < thresholdMs {
|
2020-12-06 13:03:40 -08:00
|
|
|
return false
|
|
|
|
}
|
2020-12-10 07:02:45 -08:00
|
|
|
style := DurationStyle(t.props.getString(Style, string(Austin)))
|
|
|
|
t.output = t.formatDuration(int64(executionTimeMs), style)
|
2020-12-06 13:03:40 -08:00
|
|
|
|
|
|
|
return t.output != ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *executiontime) string() string {
|
|
|
|
return t.output
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *executiontime) init(props *properties, env environmentInfo) {
|
|
|
|
t.props = props
|
|
|
|
t.env = env
|
|
|
|
}
|
2020-12-10 07:02:45 -08:00
|
|
|
|
|
|
|
func (t *executiontime) formatDuration(ms int64, style DurationStyle) string {
|
|
|
|
switch style {
|
|
|
|
case Austin:
|
|
|
|
return t.formatDurationAustin(ms)
|
|
|
|
case Roundrock:
|
|
|
|
return t.formatDurationRoundrock(ms)
|
|
|
|
case Dallas:
|
|
|
|
return t.formatDurationDallas(ms)
|
|
|
|
case Galveston:
|
|
|
|
return t.formatDurationGalveston(ms)
|
|
|
|
case Houston:
|
|
|
|
return t.formatDurationHouston(ms)
|
|
|
|
case Amarillo:
|
|
|
|
return t.formatDurationAmarillo(ms)
|
2021-05-20 10:53:42 -07:00
|
|
|
case Round:
|
|
|
|
return t.formatDurationRound(ms)
|
2020-12-10 07:02:45 -08:00
|
|
|
default:
|
|
|
|
return fmt.Sprintf("Style: %s is not available", style)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *executiontime) formatDurationAustin(ms int64) string {
|
|
|
|
if ms < second {
|
|
|
|
return fmt.Sprintf("%dms", ms%second)
|
|
|
|
}
|
|
|
|
|
|
|
|
seconds := float64(ms%minute) / second
|
|
|
|
result := strconv.FormatFloat(seconds, 'f', -1, 64) + "s"
|
|
|
|
|
|
|
|
if ms >= minute {
|
|
|
|
result = fmt.Sprintf("%dm %s", ms/minute%secondsPerMinute, result)
|
|
|
|
}
|
|
|
|
if ms >= hour {
|
|
|
|
result = fmt.Sprintf("%dh %s", ms/hour%hoursPerDay, result)
|
|
|
|
}
|
|
|
|
if ms >= day {
|
|
|
|
result = fmt.Sprintf("%dd %s", ms/day, result)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *executiontime) formatDurationRoundrock(ms int64) string {
|
|
|
|
result := fmt.Sprintf("%dms", ms%second)
|
|
|
|
if ms >= second {
|
|
|
|
result = fmt.Sprintf("%ds %s", ms/second%secondsPerMinute, result)
|
|
|
|
}
|
|
|
|
if ms >= minute {
|
|
|
|
result = fmt.Sprintf("%dm %s", ms/minute%minutesPerHour, result)
|
|
|
|
}
|
|
|
|
if ms >= hour {
|
|
|
|
result = fmt.Sprintf("%dh %s", ms/hour%hoursPerDay, result)
|
|
|
|
}
|
|
|
|
if ms >= day {
|
|
|
|
result = fmt.Sprintf("%dd %s", ms/day, result)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *executiontime) formatDurationDallas(ms int64) string {
|
|
|
|
seconds := float64(ms%minute) / second
|
|
|
|
result := strconv.FormatFloat(seconds, 'f', -1, 64)
|
|
|
|
|
|
|
|
if ms >= minute {
|
|
|
|
result = fmt.Sprintf("%d:%s", ms/minute%minutesPerHour, result)
|
|
|
|
}
|
|
|
|
if ms >= hour {
|
|
|
|
result = fmt.Sprintf("%d:%s", ms/hour%hoursPerDay, result)
|
|
|
|
}
|
|
|
|
if ms >= day {
|
|
|
|
result = fmt.Sprintf("%d:%s", ms/day, result)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *executiontime) formatDurationGalveston(ms int64) string {
|
|
|
|
result := fmt.Sprintf("%02d:%02d:%02d", ms/hour, ms/minute%minutesPerHour, ms%minute/second)
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *executiontime) formatDurationHouston(ms int64) string {
|
|
|
|
milliseconds := ".0"
|
|
|
|
if ms%second > 0 {
|
|
|
|
// format milliseconds as a string with truncated trailing zeros
|
|
|
|
milliseconds = strconv.FormatFloat(float64(ms%second)/second, 'f', -1, 64)
|
|
|
|
// at this point milliseconds looks like "0.5". remove the leading "0"
|
|
|
|
milliseconds = milliseconds[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
result := fmt.Sprintf("%02d:%02d:%02d%s", ms/hour, ms/minute%minutesPerHour, ms%minute/second, milliseconds)
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *executiontime) formatDurationAmarillo(ms int64) string {
|
|
|
|
// wholeNumber represents the value to the left of the decimal point (seconds)
|
|
|
|
wholeNumber := ms / second
|
|
|
|
// decimalNumber represents the value to the right of the decimal point (milliseconds)
|
|
|
|
decimalNumber := float64(ms%second) / second
|
|
|
|
|
|
|
|
// format wholeNumber as a string with thousands separators
|
|
|
|
printer := message.NewPrinter(lang.English)
|
|
|
|
result := printer.Sprintf("%d", wholeNumber)
|
|
|
|
|
|
|
|
if decimalNumber > 0 {
|
|
|
|
// format decimalNumber as a string with truncated trailing zeros
|
|
|
|
decimalResult := strconv.FormatFloat(decimalNumber, 'f', -1, 64)
|
|
|
|
// at this point decimalResult looks like "0.5"
|
|
|
|
// remove the leading "0" and append
|
|
|
|
result += decimalResult[1:]
|
|
|
|
}
|
|
|
|
result += "s"
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
2021-05-20 10:53:42 -07:00
|
|
|
|
|
|
|
func (t *executiontime) formatDurationRound(ms int64) string {
|
2021-05-20 23:14:02 -07:00
|
|
|
toRoundString := func(one, two int64, oneText, twoText string) string {
|
|
|
|
if two == 0 {
|
|
|
|
return fmt.Sprintf("%d%s", one, oneText)
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%d%s %d%s", one, oneText, two, twoText)
|
|
|
|
}
|
|
|
|
hours := ms / hour % hoursPerDay
|
2021-05-20 10:53:42 -07:00
|
|
|
if ms >= day {
|
2021-05-20 23:14:02 -07:00
|
|
|
return toRoundString(ms/day, hours, "d", "h")
|
2021-05-20 10:53:42 -07:00
|
|
|
}
|
2021-05-20 23:14:02 -07:00
|
|
|
minutes := ms / minute % secondsPerMinute
|
2021-05-20 10:53:42 -07:00
|
|
|
if ms >= hour {
|
2021-05-20 23:14:02 -07:00
|
|
|
return toRoundString(hours, minutes, "h", "m")
|
2021-05-20 10:53:42 -07:00
|
|
|
}
|
2021-05-20 23:14:02 -07:00
|
|
|
seconds := (ms % minute) / second
|
2021-05-20 10:53:42 -07:00
|
|
|
if ms >= minute {
|
2021-05-20 23:14:02 -07:00
|
|
|
return toRoundString(minutes, seconds, "m", "s")
|
2021-05-20 10:53:42 -07:00
|
|
|
}
|
|
|
|
if ms >= second {
|
2021-05-20 23:14:02 -07:00
|
|
|
return fmt.Sprintf("%ds", seconds)
|
2021-05-20 10:53:42 -07:00
|
|
|
}
|
|
|
|
return fmt.Sprintf("%dms", ms%second)
|
|
|
|
}
|