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 == nil || !corePtr.IsRunning() { return nil } box := corePtr.GetInstance() if box == nil { return nil } st := box.StatsTracker() if st == nil { return nil } stats := st.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 } err = tx.Create(&stats).Error return err } 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 }