Files
s-ui/service/stats.go
T
2026-03-08 01:50:49 +01:00

155 lines
3.8 KiB
Go

package service
import (
"sort"
"time"
"github.com/alireza0/s-ui/database"
"github.com/alireza0/s-ui/database/model"
"gorm.io/gorm"
)
type onlines struct {
Inbound []string `json:"inbound,omitempty"`
User []string `json:"user,omitempty"`
Outbound []string `json:"outbound,omitempty"`
}
var onlineResources = &onlines{}
type StatsService struct {
}
func (s *StatsService) SaveStats(enableTraffic bool) error {
if !corePtr.IsRunning() {
return nil
}
stats := corePtr.GetInstance().StatsTracker().GetStats()
// Reset onlines
onlineResources.Inbound = nil
onlineResources.Outbound = nil
onlineResources.User = nil
if len(*stats) == 0 {
return nil
}
var err error
db := database.GetDB()
tx := db.Begin()
defer func() {
if err == nil {
tx.Commit()
} else {
tx.Rollback()
}
}()
for _, stat := range *stats {
if stat.Resource == "user" {
if stat.Direction {
err = tx.Model(model.Client{}).Where("name = ?", stat.Tag).
UpdateColumn("up", gorm.Expr("up + ?", stat.Traffic)).Error
} else {
err = tx.Model(model.Client{}).Where("name = ?", stat.Tag).
UpdateColumn("down", gorm.Expr("down + ?", stat.Traffic)).Error
}
if err != nil {
return err
}
}
if stat.Direction {
switch stat.Resource {
case "inbound":
onlineResources.Inbound = append(onlineResources.Inbound, stat.Tag)
case "outbound":
onlineResources.Outbound = append(onlineResources.Outbound, stat.Tag)
case "user":
onlineResources.User = append(onlineResources.User, stat.Tag)
}
}
}
if !enableTraffic {
return nil
}
return tx.Create(&stats).Error
}
func (s *StatsService) GetStats(resource string, tag string, limit int) ([]model.Stats, error) {
var err error
var result []model.Stats
currentTime := time.Now().Unix()
timeDiff := currentTime - (int64(limit) * 3600)
db := database.GetDB()
resources := []string{resource}
if resource == "endpoint" {
resources = []string{"inbound", "outbound"}
}
err = db.Model(model.Stats{}).Where("resource in ? AND tag = ? AND date_time > ?", resources, tag, timeDiff).Scan(&result).Error
if err != nil {
return nil, err
}
result = s.downsampleStats(result, 60) // 60 rows for 30 buckets
return result, nil
}
// downsampleStats reduces stats to maxRows rows.
// Each bucket outputs two rows (direction false and true) with average Traffic.
func (s *StatsService) downsampleStats(stats []model.Stats, maxRows int) []model.Stats {
if len(stats) <= maxRows {
return stats
}
numBuckets := int(maxRows / 2)
sort.Slice(stats, func(i, j int) bool { return stats[i].DateTime < stats[j].DateTime })
timeMin, timeMax := stats[0].DateTime, stats[len(stats)-1].DateTime
bucketSpan := (timeMax - timeMin) / int64(numBuckets)
if bucketSpan == 0 {
bucketSpan = 1
}
downsampled := make([]model.Stats, 0, maxRows)
for i := 0; i < numBuckets; i++ {
bucketStart := timeMin + int64(i)*bucketSpan
bucketEnd := timeMin + int64(i+1)*bucketSpan
if i == numBuckets-1 {
bucketEnd = timeMax + 1
}
for _, dir := range []bool{false, true} {
var sum int64
var count int
for _, r := range stats {
if r.DateTime >= bucketStart && r.DateTime < bucketEnd && r.Direction == dir {
sum += r.Traffic
count++
}
}
avg := int64(0)
if count > 0 {
avg = sum / int64(count)
}
downsampled = append(downsampled, model.Stats{
DateTime: bucketStart,
Resource: stats[0].Resource,
Tag: stats[0].Tag,
Direction: dir,
Traffic: avg,
})
}
}
return downsampled
}
func (s *StatsService) GetOnlines() (onlines, error) {
return *onlineResources, nil
}
func (s *StatsService) DelOldStats(days int) error {
oldTime := time.Now().AddDate(0, 0, -(days)).Unix()
db := database.GetDB()
return db.Where("date_time < ?", oldTime).Delete(model.Stats{}).Error
}