155 lines
3.8 KiB
Go
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
|
|
}
|