数据埋点实现

This commit is contained in:
kercylan98 2023-06-01 11:35:01 +08:00
parent 6def300d4a
commit 3d9d6a9da9
4 changed files with 121 additions and 0 deletions

42
report/data_buried.go Normal file
View File

@ -0,0 +1,42 @@
package report
import (
"github.com/kercylan98/minotaur/utils/asynchronization"
"github.com/kercylan98/minotaur/utils/hash"
)
// NewDataBuried 创建一个数据埋点
func NewDataBuried[DataID comparable, Data any](hitLogic HitLogic[Data], options ...DataBuriedOption[DataID, Data]) *DataBuried[DataID, Data] {
buried := &DataBuried[DataID, Data]{
data: asynchronization.NewMap[DataID, Data](),
hitLogic: hitLogic,
}
for _, option := range options {
option(buried)
}
return buried
}
// DataBuried 数据埋点
// - 数据埋点通常用于统计不同类型的数据,例如用户数据、商城数据等
type DataBuried[DataID comparable, Data any] struct {
data hash.Map[DataID, Data]
hitLogic HitLogic[Data]
}
// Hit 命中数据埋点
func (slf *DataBuried[DataID, Data]) Hit(id DataID, data Data) {
slf.data.Atom(func(m hash.Map[DataID, Data]) {
m.Set(id, slf.hitLogic(m.Get(id), data))
})
}
// GetData 获取数据
func (slf *DataBuried[DataID, Data]) GetData(id DataID) Data {
return slf.data.Get(id)
}
// GetSize 获取已触发该埋点的id数量
func (slf *DataBuried[DataID, Data]) GetSize() int {
return slf.data.Size()
}

View File

@ -0,0 +1,12 @@
package report
import "github.com/kercylan98/minotaur/utils/synchronization"
type DataBuriedOption[DataID comparable, Data any] func(buried *DataBuried[DataID, Data])
// WithDataBuriedSync 通过并发安全的方式创建数据埋点
func WithDataBuriedSync[DataId comparable, Data any]() DataBuriedOption[DataId, Data] {
return func(buried *DataBuried[DataId, Data]) {
buried.data = synchronization.NewMap[DataId, Data]()
}
}

35
report/global_buried.go Normal file
View File

@ -0,0 +1,35 @@
package report
import (
"sync"
)
// NewGlobalBuried 创建一个全局埋点
func NewGlobalBuried[Data any](hitLogic HitLogic[Data]) *GlobalBuried[Data] {
return &GlobalBuried[Data]{
hitLogic: hitLogic,
}
}
// GlobalBuried 全局埋点
// - 天然并发安全
// - 全局埋点适用于活跃用户数等统计
type GlobalBuried[Data any] struct {
rw sync.RWMutex
data Data
hitLogic HitLogic[Data]
}
// Hit 命中数据埋点
func (slf *GlobalBuried[Data]) Hit(data Data) {
slf.rw.Lock()
defer slf.rw.Unlock()
slf.data = slf.hitLogic(slf.data, data)
}
// GetData 获取数据
func (slf *GlobalBuried[Data]) GetData() Data {
slf.rw.RLock()
defer slf.rw.RUnlock()
return slf.data
}

32
report/hit_logic.go Normal file
View File

@ -0,0 +1,32 @@
package report
import "github.com/kercylan98/minotaur/utils/generic"
// HitLogic 埋点命中逻辑
// - data: 当前数据
// - input: 新输入的数据
// - return: 设置当前数据
type HitLogic[Data any] func(data Data, input Data) Data
// WithHitLogicCustomize 通过自定义命中处理逻辑处理命中事件
// - 通过对旧数据和新数据的处理,将返回的值作为新的数据值
func WithHitLogicCustomize[Data any](logic func(data Data, input Data) Data) HitLogic[Data] {
return func(data Data, input Data) Data {
return logic(data, input)
}
}
// WithHitLogicCover 通过覆盖原数值的方式处理命中事件
func WithHitLogicCover[Data any]() HitLogic[Data] {
return func(data Data, input Data) Data {
return input
}
}
// WithHitLogicOverlay 通过数值叠加的方式处理命中事件
// - 例如原始值为100新输入为200最终结果为300
func WithHitLogicOverlay[Data generic.Number]() HitLogic[Data] {
return func(data Data, input Data) Data {
return data + input
}
}