2022-07-05 20:37:00 +08:00
|
|
|
|
// Copyright 2022 Liuxiangchao iwind.liu@gmail.com. All rights reserved. Official site: https://goedge.cn .
|
|
|
|
|
|
|
|
|
|
|
|
package stats
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
2023-04-02 17:24:55 +08:00
|
|
|
|
"encoding/json"
|
2022-10-11 18:57:35 +08:00
|
|
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/nodeconfigs"
|
2022-07-05 20:37:00 +08:00
|
|
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
2023-03-10 15:14:14 +08:00
|
|
|
|
teaconst "github.com/TeaOSLab/EdgeNode/internal/const"
|
2022-10-23 19:41:21 +08:00
|
|
|
|
"github.com/TeaOSLab/EdgeNode/internal/events"
|
|
|
|
|
|
"github.com/TeaOSLab/EdgeNode/internal/goman"
|
2022-07-05 20:37:00 +08:00
|
|
|
|
"github.com/TeaOSLab/EdgeNode/internal/remotelogs"
|
|
|
|
|
|
"github.com/TeaOSLab/EdgeNode/internal/rpc"
|
2023-04-08 12:47:04 +08:00
|
|
|
|
"github.com/TeaOSLab/EdgeNode/internal/utils/fasttime"
|
2023-04-02 17:24:55 +08:00
|
|
|
|
"github.com/iwind/TeaGo/Tea"
|
2022-07-05 20:37:00 +08:00
|
|
|
|
"github.com/iwind/TeaGo/logs"
|
|
|
|
|
|
"github.com/iwind/TeaGo/types"
|
|
|
|
|
|
timeutil "github.com/iwind/TeaGo/utils/time"
|
2023-04-02 17:24:55 +08:00
|
|
|
|
"os"
|
2022-07-05 20:37:00 +08:00
|
|
|
|
"sync"
|
|
|
|
|
|
"time"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
var SharedBandwidthStatManager = NewBandwidthStatManager()
|
|
|
|
|
|
|
2022-08-26 16:47:42 +08:00
|
|
|
|
const bandwidthTimestampDelim = 2 // N秒平均,更为精确
|
|
|
|
|
|
|
2022-10-23 19:41:21 +08:00
|
|
|
|
func init() {
|
2023-03-10 15:14:14 +08:00
|
|
|
|
if !teaconst.IsMain {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2022-10-23 19:41:21 +08:00
|
|
|
|
events.On(events.EventLoaded, func() {
|
|
|
|
|
|
goman.New(func() {
|
|
|
|
|
|
SharedBandwidthStatManager.Start()
|
|
|
|
|
|
})
|
|
|
|
|
|
})
|
2023-04-02 17:24:55 +08:00
|
|
|
|
|
2023-06-23 17:45:39 +08:00
|
|
|
|
events.OnClose(func() {
|
2023-04-05 11:34:15 +08:00
|
|
|
|
SharedBandwidthStatManager.Cancel()
|
|
|
|
|
|
|
2023-04-02 17:24:55 +08:00
|
|
|
|
err := SharedBandwidthStatManager.Save()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
remotelogs.Error("STAT", "save bandwidth stats failed: "+err.Error())
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
2022-10-23 19:41:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2022-07-05 20:37:00 +08:00
|
|
|
|
type BandwidthStat struct {
|
2023-04-02 17:24:55 +08:00
|
|
|
|
Day string `json:"day"`
|
|
|
|
|
|
TimeAt string `json:"timeAt"`
|
|
|
|
|
|
UserId int64 `json:"userId"`
|
|
|
|
|
|
ServerId int64 `json:"serverId"`
|
|
|
|
|
|
|
|
|
|
|
|
CurrentBytes int64 `json:"currentBytes"`
|
|
|
|
|
|
CurrentTimestamp int64 `json:"currentTimestamp"`
|
|
|
|
|
|
MaxBytes int64 `json:"maxBytes"`
|
|
|
|
|
|
TotalBytes int64 `json:"totalBytes"`
|
|
|
|
|
|
|
2023-12-20 11:43:00 +08:00
|
|
|
|
CachedBytes int64 `json:"cachedBytes"`
|
|
|
|
|
|
AttackBytes int64 `json:"attackBytes"`
|
|
|
|
|
|
CountRequests int64 `json:"countRequests"`
|
|
|
|
|
|
CountCachedRequests int64 `json:"countCachedRequests"`
|
|
|
|
|
|
CountAttackRequests int64 `json:"countAttackRequests"`
|
|
|
|
|
|
CountWebsocketConnections int64 `json:"countWebsocketConnections"`
|
|
|
|
|
|
UserPlanId int64 `json:"userPlanId"`
|
2022-07-05 20:37:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// BandwidthStatManager 服务带宽统计
|
|
|
|
|
|
type BandwidthStatManager struct {
|
2023-03-22 18:00:35 +08:00
|
|
|
|
m map[string]*BandwidthStat // serverId@day@time => *BandwidthStat
|
2022-07-05 20:37:00 +08:00
|
|
|
|
|
2022-10-23 19:41:21 +08:00
|
|
|
|
pbStats []*pb.ServerBandwidthStat
|
|
|
|
|
|
|
2022-07-05 20:37:00 +08:00
|
|
|
|
lastTime string // 上一次执行的时间
|
|
|
|
|
|
|
|
|
|
|
|
ticker *time.Ticker
|
|
|
|
|
|
locker sync.Mutex
|
2024-04-12 20:12:09 +08:00
|
|
|
|
|
|
|
|
|
|
cacheFile string
|
2022-07-05 20:37:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func NewBandwidthStatManager() *BandwidthStatManager {
|
|
|
|
|
|
return &BandwidthStatManager{
|
2024-04-12 20:12:09 +08:00
|
|
|
|
m: map[string]*BandwidthStat{},
|
|
|
|
|
|
ticker: time.NewTicker(1 * time.Minute), // 时间小于1分钟是为了更快速地上传结果
|
|
|
|
|
|
cacheFile: Tea.Root + "/data/stat_bandwidth.cache",
|
2022-07-05 20:37:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (this *BandwidthStatManager) Start() {
|
2023-04-02 17:24:55 +08:00
|
|
|
|
// 从上次数据中恢复
|
|
|
|
|
|
this.locker.Lock()
|
|
|
|
|
|
this.recover()
|
|
|
|
|
|
this.locker.Unlock()
|
|
|
|
|
|
|
|
|
|
|
|
// 循环上报数据
|
2022-07-05 20:37:00 +08:00
|
|
|
|
for range this.ticker.C {
|
|
|
|
|
|
err := this.Loop()
|
|
|
|
|
|
if err != nil && !rpc.IsConnError(err) {
|
|
|
|
|
|
remotelogs.Error("BANDWIDTH_STAT_MANAGER", err.Error())
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (this *BandwidthStatManager) Loop() error {
|
2022-10-11 18:57:35 +08:00
|
|
|
|
var regionId int64
|
|
|
|
|
|
nodeConfig, _ := nodeconfigs.SharedNodeConfig()
|
|
|
|
|
|
if nodeConfig != nil {
|
|
|
|
|
|
regionId = nodeConfig.RegionId
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2022-07-05 20:37:00 +08:00
|
|
|
|
var now = time.Now()
|
|
|
|
|
|
var day = timeutil.Format("Ymd", now)
|
2023-03-22 18:00:35 +08:00
|
|
|
|
var currentTime = timeutil.FormatTime("Hi", now.Unix()/300*300) // 300s = 5 minutes
|
2022-07-05 20:37:00 +08:00
|
|
|
|
|
|
|
|
|
|
if this.lastTime == currentTime {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
this.lastTime = currentTime
|
|
|
|
|
|
|
|
|
|
|
|
var pbStats = []*pb.ServerBandwidthStat{}
|
|
|
|
|
|
|
2022-10-23 19:41:21 +08:00
|
|
|
|
// 历史未提交记录
|
|
|
|
|
|
if len(this.pbStats) > 0 {
|
|
|
|
|
|
var expiredTime = timeutil.FormatTime("Hi", time.Now().Unix()-1200) // 只保留20分钟
|
|
|
|
|
|
|
|
|
|
|
|
for _, stat := range this.pbStats {
|
|
|
|
|
|
if stat.TimeAt > expiredTime {
|
|
|
|
|
|
pbStats = append(pbStats, stat)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
this.pbStats = nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-04-12 20:12:09 +08:00
|
|
|
|
var ipStatMap = SharedDAUManager.ReadStatMap()
|
|
|
|
|
|
|
2022-07-05 20:37:00 +08:00
|
|
|
|
this.locker.Lock()
|
|
|
|
|
|
for key, stat := range this.m {
|
|
|
|
|
|
if stat.Day < day || stat.TimeAt < currentTime {
|
2023-03-22 18:00:35 +08:00
|
|
|
|
// 防止数据出现错误
|
2023-07-30 14:49:16 +08:00
|
|
|
|
if stat.CachedBytes > stat.TotalBytes || stat.CountCachedRequests == stat.CountRequests {
|
2023-03-22 18:00:35 +08:00
|
|
|
|
stat.CachedBytes = stat.TotalBytes
|
|
|
|
|
|
}
|
2023-07-30 14:49:16 +08:00
|
|
|
|
|
2023-03-22 18:00:35 +08:00
|
|
|
|
if stat.AttackBytes > stat.TotalBytes {
|
|
|
|
|
|
stat.AttackBytes = stat.TotalBytes
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-04-12 20:12:09 +08:00
|
|
|
|
var ipKey = "server_" + stat.Day + "_" + types.String(stat.ServerId)
|
|
|
|
|
|
|
2022-07-05 20:37:00 +08:00
|
|
|
|
pbStats = append(pbStats, &pb.ServerBandwidthStat{
|
2023-12-20 11:43:00 +08:00
|
|
|
|
Id: 0,
|
|
|
|
|
|
UserId: stat.UserId,
|
|
|
|
|
|
ServerId: stat.ServerId,
|
|
|
|
|
|
Day: stat.Day,
|
|
|
|
|
|
TimeAt: stat.TimeAt,
|
|
|
|
|
|
Bytes: stat.MaxBytes / bandwidthTimestampDelim,
|
|
|
|
|
|
TotalBytes: stat.TotalBytes,
|
|
|
|
|
|
CachedBytes: stat.CachedBytes,
|
|
|
|
|
|
AttackBytes: stat.AttackBytes,
|
|
|
|
|
|
CountRequests: stat.CountRequests,
|
|
|
|
|
|
CountCachedRequests: stat.CountCachedRequests,
|
|
|
|
|
|
CountAttackRequests: stat.CountAttackRequests,
|
|
|
|
|
|
CountWebsocketConnections: stat.CountWebsocketConnections,
|
2024-04-12 20:12:09 +08:00
|
|
|
|
CountIPs: ipStatMap[ipKey],
|
2023-12-20 11:43:00 +08:00
|
|
|
|
UserPlanId: stat.UserPlanId,
|
|
|
|
|
|
NodeRegionId: regionId,
|
2022-07-05 20:37:00 +08:00
|
|
|
|
})
|
|
|
|
|
|
delete(this.m, key)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
this.locker.Unlock()
|
|
|
|
|
|
|
|
|
|
|
|
if len(pbStats) > 0 {
|
|
|
|
|
|
// 上传
|
|
|
|
|
|
rpcClient, err := rpc.SharedRPC()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
2022-08-24 20:04:46 +08:00
|
|
|
|
_, err = rpcClient.ServerBandwidthStatRPC.UploadServerBandwidthStats(rpcClient.Context(), &pb.UploadServerBandwidthStatsRequest{ServerBandwidthStats: pbStats})
|
2022-07-05 20:37:00 +08:00
|
|
|
|
if err != nil {
|
2022-10-23 19:41:21 +08:00
|
|
|
|
this.pbStats = pbStats
|
|
|
|
|
|
|
2022-07-05 20:37:00 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-03-22 18:00:35 +08:00
|
|
|
|
// AddBandwidth 添加带宽数据
|
2023-09-06 16:34:11 +08:00
|
|
|
|
func (this *BandwidthStatManager) AddBandwidth(userId int64, userPlanId int64, serverId int64, peekBytes int64, totalBytes int64) {
|
2023-02-27 10:48:16 +08:00
|
|
|
|
if serverId <= 0 || (peekBytes == 0 && totalBytes == 0) {
|
2022-07-05 20:37:00 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-04-08 12:47:04 +08:00
|
|
|
|
var now = fasttime.Now()
|
|
|
|
|
|
var timestamp = now.Unix() / bandwidthTimestampDelim * bandwidthTimestampDelim // 将时间戳均分成N等份
|
|
|
|
|
|
var day = now.Ymd()
|
|
|
|
|
|
var timeAt = now.Round5Hi()
|
2022-07-05 20:37:00 +08:00
|
|
|
|
var key = types.String(serverId) + "@" + day + "@" + timeAt
|
|
|
|
|
|
|
2022-08-26 16:47:42 +08:00
|
|
|
|
// 增加TCP Header尺寸,这里默认MTU为1500,且默认为IPv4
|
|
|
|
|
|
const mtu = 1500
|
|
|
|
|
|
const tcpHeaderSize = 20
|
2023-02-27 10:48:16 +08:00
|
|
|
|
if peekBytes > mtu {
|
|
|
|
|
|
peekBytes += peekBytes * tcpHeaderSize / mtu
|
2022-08-26 16:47:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2022-07-05 20:37:00 +08:00
|
|
|
|
this.locker.Lock()
|
|
|
|
|
|
stat, ok := this.m[key]
|
|
|
|
|
|
if ok {
|
2022-07-07 09:21:18 +08:00
|
|
|
|
// 此刻如果发生用户ID(userId)的变化也忽略,等N分钟后有新记录后再换
|
|
|
|
|
|
|
2022-07-05 20:37:00 +08:00
|
|
|
|
if stat.CurrentTimestamp == timestamp {
|
2023-02-27 10:48:16 +08:00
|
|
|
|
stat.CurrentBytes += peekBytes
|
2022-07-05 20:37:00 +08:00
|
|
|
|
} else {
|
2023-02-27 10:48:16 +08:00
|
|
|
|
stat.CurrentBytes = peekBytes
|
2022-07-05 20:37:00 +08:00
|
|
|
|
stat.CurrentTimestamp = timestamp
|
|
|
|
|
|
}
|
|
|
|
|
|
if stat.CurrentBytes > stat.MaxBytes {
|
|
|
|
|
|
stat.MaxBytes = stat.CurrentBytes
|
|
|
|
|
|
}
|
2023-02-27 10:48:16 +08:00
|
|
|
|
|
|
|
|
|
|
stat.TotalBytes += totalBytes
|
2022-07-05 20:37:00 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
this.m[key] = &BandwidthStat{
|
|
|
|
|
|
Day: day,
|
|
|
|
|
|
TimeAt: timeAt,
|
2022-07-07 09:21:18 +08:00
|
|
|
|
UserId: userId,
|
2023-09-06 16:34:11 +08:00
|
|
|
|
UserPlanId: userPlanId,
|
2022-07-05 20:37:00 +08:00
|
|
|
|
ServerId: serverId,
|
2023-02-27 10:48:16 +08:00
|
|
|
|
CurrentBytes: peekBytes,
|
|
|
|
|
|
MaxBytes: peekBytes,
|
|
|
|
|
|
TotalBytes: totalBytes,
|
2022-07-05 20:37:00 +08:00
|
|
|
|
CurrentTimestamp: timestamp,
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
this.locker.Unlock()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-03-22 18:00:35 +08:00
|
|
|
|
// AddTraffic 添加请求数据
|
2023-12-20 11:43:00 +08:00
|
|
|
|
func (this *BandwidthStatManager) AddTraffic(serverId int64, cachedBytes int64, countRequests int64, countCachedRequests int64, countAttacks int64, attackBytes int64, countWebsocketConnections int64) {
|
2023-04-08 12:47:04 +08:00
|
|
|
|
var now = fasttime.Now()
|
|
|
|
|
|
var day = now.Ymd()
|
|
|
|
|
|
var timeAt = now.Round5Hi()
|
2023-03-22 18:00:35 +08:00
|
|
|
|
var key = types.String(serverId) + "@" + day + "@" + timeAt
|
|
|
|
|
|
this.locker.Lock()
|
|
|
|
|
|
// 只有有记录了才会添加
|
|
|
|
|
|
stat, ok := this.m[key]
|
|
|
|
|
|
if ok {
|
|
|
|
|
|
stat.CachedBytes += cachedBytes
|
|
|
|
|
|
stat.CountRequests += countRequests
|
|
|
|
|
|
stat.CountCachedRequests += countCachedRequests
|
|
|
|
|
|
stat.CountAttackRequests += countAttacks
|
|
|
|
|
|
stat.AttackBytes += attackBytes
|
2023-12-20 11:43:00 +08:00
|
|
|
|
stat.CountWebsocketConnections += countWebsocketConnections
|
2023-03-22 18:00:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
this.locker.Unlock()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2022-07-05 20:37:00 +08:00
|
|
|
|
func (this *BandwidthStatManager) Inspect() {
|
|
|
|
|
|
this.locker.Lock()
|
|
|
|
|
|
logs.PrintAsJSON(this.m)
|
|
|
|
|
|
this.locker.Unlock()
|
|
|
|
|
|
}
|
2022-08-26 16:47:42 +08:00
|
|
|
|
|
|
|
|
|
|
func (this *BandwidthStatManager) Map() map[int64]int64 /** serverId => max bytes **/ {
|
|
|
|
|
|
this.locker.Lock()
|
|
|
|
|
|
defer this.locker.Unlock()
|
|
|
|
|
|
|
|
|
|
|
|
var m = map[int64]int64{}
|
|
|
|
|
|
for _, v := range this.m {
|
|
|
|
|
|
m[v.ServerId] = v.MaxBytes / bandwidthTimestampDelim
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return m
|
|
|
|
|
|
}
|
2023-04-02 17:24:55 +08:00
|
|
|
|
|
|
|
|
|
|
// Save 保存到本地磁盘
|
|
|
|
|
|
func (this *BandwidthStatManager) Save() error {
|
|
|
|
|
|
this.locker.Lock()
|
|
|
|
|
|
defer this.locker.Unlock()
|
|
|
|
|
|
|
2023-06-23 18:11:27 +08:00
|
|
|
|
if len(this.m) == 0 {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-04-02 17:24:55 +08:00
|
|
|
|
data, err := json.Marshal(this.m)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-04-12 20:12:09 +08:00
|
|
|
|
_ = os.Remove(this.cacheFile)
|
|
|
|
|
|
return os.WriteFile(this.cacheFile, data, 0666)
|
2023-04-02 17:24:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-04-05 11:34:15 +08:00
|
|
|
|
// Cancel 取消上传
|
|
|
|
|
|
func (this *BandwidthStatManager) Cancel() {
|
|
|
|
|
|
this.ticker.Stop()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-04-02 17:24:55 +08:00
|
|
|
|
// 从本地缓存文件中恢复数据
|
|
|
|
|
|
func (this *BandwidthStatManager) recover() {
|
2024-04-12 20:12:09 +08:00
|
|
|
|
cacheData, err := os.ReadFile(this.cacheFile)
|
2023-04-02 17:24:55 +08:00
|
|
|
|
if err == nil {
|
|
|
|
|
|
var m = map[string]*BandwidthStat{}
|
|
|
|
|
|
err = json.Unmarshal(cacheData, &m)
|
|
|
|
|
|
if err == nil && len(m) > 0 {
|
|
|
|
|
|
var lastTime = ""
|
|
|
|
|
|
for _, stat := range m {
|
2023-04-08 12:47:04 +08:00
|
|
|
|
if stat.Day != fasttime.Now().Ymd() {
|
2023-04-07 11:32:40 +08:00
|
|
|
|
continue
|
|
|
|
|
|
}
|
2023-04-02 17:24:55 +08:00
|
|
|
|
if len(lastTime) == 0 || stat.TimeAt > lastTime {
|
|
|
|
|
|
lastTime = stat.TimeAt
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if len(lastTime) > 0 {
|
|
|
|
|
|
var availableTime = timeutil.FormatTime("Hi", (time.Now().Unix()-300) /** 只保留5分钟的 **/ /300*300) // 300s = 5 minutes
|
|
|
|
|
|
if lastTime >= availableTime {
|
|
|
|
|
|
this.m = m
|
|
|
|
|
|
this.lastTime = lastTime
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-04-12 20:12:09 +08:00
|
|
|
|
_ = os.Remove(this.cacheFile)
|
2023-04-02 17:24:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|