go-zero/core/prof/profilecenter.go

118 lines
2.1 KiB
Go
Raw Normal View History

2020-07-26 17:09:05 +08:00
package prof
import (
"bytes"
"strconv"
"sync"
"sync/atomic"
"time"
"github.com/olekukonko/tablewriter"
2020-08-08 16:40:10 +08:00
"github.com/tal-tech/go-zero/core/logx"
"github.com/tal-tech/go-zero/core/threading"
2020-07-26 17:09:05 +08:00
)
type (
2021-02-22 10:20:54 +08:00
profileSlot struct {
2020-07-26 17:09:05 +08:00
lifecount int64
lastcount int64
lifecycle int64
lastcycle int64
}
2021-02-22 10:20:54 +08:00
profileCenter struct {
2020-07-26 17:09:05 +08:00
lock sync.RWMutex
2021-02-22 10:20:54 +08:00
slots map[string]*profileSlot
2020-07-26 17:09:05 +08:00
}
)
const flushInterval = 5 * time.Minute
var (
2021-02-22 10:20:54 +08:00
pc = &profileCenter{
slots: make(map[string]*profileSlot),
2020-07-26 17:09:05 +08:00
}
once sync.Once
)
func report(name string, duration time.Duration) {
updated := func() bool {
2021-02-22 10:20:54 +08:00
pc.lock.RLock()
defer pc.lock.RUnlock()
2020-07-26 17:09:05 +08:00
2021-02-22 10:20:54 +08:00
slot, ok := pc.slots[name]
2020-07-26 17:09:05 +08:00
if ok {
atomic.AddInt64(&slot.lifecount, 1)
atomic.AddInt64(&slot.lastcount, 1)
atomic.AddInt64(&slot.lifecycle, int64(duration))
atomic.AddInt64(&slot.lastcycle, int64(duration))
}
return ok
}()
if !updated {
func() {
2021-02-22 10:20:54 +08:00
pc.lock.Lock()
defer pc.lock.Unlock()
2020-07-26 17:09:05 +08:00
2021-02-22 10:20:54 +08:00
pc.slots[name] = &profileSlot{
2020-07-26 17:09:05 +08:00
lifecount: 1,
lastcount: 1,
lifecycle: int64(duration),
lastcycle: int64(duration),
}
}()
}
once.Do(flushRepeatly)
}
func flushRepeatly() {
threading.GoSafe(func() {
for {
time.Sleep(flushInterval)
logx.Stat(generateReport())
}
})
}
func generateReport() string {
var buffer bytes.Buffer
buffer.WriteString("Profiling report\n")
var data [][]string
calcFn := func(total, count int64) string {
if count == 0 {
return "-"
}
2021-02-09 13:50:21 +08:00
return (time.Duration(total) / time.Duration(count)).String()
2020-07-26 17:09:05 +08:00
}
func() {
2021-02-22 10:20:54 +08:00
pc.lock.Lock()
defer pc.lock.Unlock()
2020-07-26 17:09:05 +08:00
2021-02-22 10:20:54 +08:00
for key, slot := range pc.slots {
2020-07-26 17:09:05 +08:00
data = append(data, []string{
key,
strconv.FormatInt(slot.lifecount, 10),
calcFn(slot.lifecycle, slot.lifecount),
strconv.FormatInt(slot.lastcount, 10),
calcFn(slot.lastcycle, slot.lastcount),
})
// reset the data for last cycle
slot.lastcount = 0
slot.lastcycle = 0
}
}()
table := tablewriter.NewWriter(&buffer)
table.SetHeader([]string{"QUEUE", "LIFECOUNT", "LIFECYCLE", "LASTCOUNT", "LASTCYCLE"})
table.SetBorder(false)
table.AppendBulk(data)
table.Render()
return buffer.String()
}