package main import ( "encoding/json" "fmt" "log" "math" "reflect" "time" "github.com/pocketbase/dbx" "github.com/pocketbase/pocketbase/models" "github.com/pocketbase/pocketbase/tools/types" ) func createLongerRecords(collectionName string, shorterRecord *models.Record) { shorterRecordType := shorterRecord.Get("type").(string) systemId := shorterRecord.Get("system").(string) // fmt.Println("create longer records", "recordType", shorterRecordType, "systemId", systemId) var longerRecordType string var timeAgo time.Duration var expectedShorterRecords int switch shorterRecordType { case "1m": longerRecordType = "10m" timeAgo = -10 * time.Minute expectedShorterRecords = 10 case "10m": longerRecordType = "20m" timeAgo = -20 * time.Minute expectedShorterRecords = 2 case "20m": longerRecordType = "120m" timeAgo = -120 * time.Minute expectedShorterRecords = 6 default: longerRecordType = "480m" timeAgo = -480 * time.Minute expectedShorterRecords = 4 } longerRecordPeriod := time.Now().UTC().Add(timeAgo + 10*time.Second).Format("2006-01-02 15:04:05") // check creation time of last 10m record lastLongerRecord, err := app.Dao().FindFirstRecordByFilter( collectionName, "type = {:type} && system = {:system} && created > {:created}", dbx.Params{"type": longerRecordType, "system": systemId, "created": longerRecordPeriod}, ) // return if longer record exists if err == nil || lastLongerRecord != nil { // log.Println("longer record found. returning") return } // get shorter records from the past x minutes // shorterRecordPeriod := time.Now().UTC().Add(timeAgo + time.Second).Format("2006-01-02 15:04:05") allShorterRecords, err := app.Dao().FindRecordsByFilter( collectionName, "type = {:type} && system = {:system} && created > {:created}", "-created", -1, 0, dbx.Params{"type": shorterRecordType, "system": systemId, "created": longerRecordPeriod}, ) // return if not enough shorter records if err != nil || len(allShorterRecords) < expectedShorterRecords { // log.Println("not enough shorter records. returning") return } // average the shorter records and create longer record var stats interface{} switch collectionName { case "system_stats": stats = averageSystemStats(allShorterRecords) case "container_stats": stats = averageContainerStats(allShorterRecords) } collection, _ := app.Dao().FindCollectionByNameOrId(collectionName) tenMinRecord := models.NewRecord(collection) tenMinRecord.Set("system", systemId) tenMinRecord.Set("stats", stats) tenMinRecord.Set("type", longerRecordType) if err := app.Dao().SaveRecord(tenMinRecord); err != nil { fmt.Println("failed to save longer record", "err", err.Error()) } } // calculate the average stats of a list of system_stats records func averageSystemStats(records []*models.Record) SystemStats { count := float64(len(records)) sum := reflect.New(reflect.TypeOf(SystemStats{})).Elem() for _, record := range records { var stats SystemStats json.Unmarshal([]byte(record.Get("stats").(types.JsonRaw)), &stats) statValue := reflect.ValueOf(stats) for i := 0; i < statValue.NumField(); i++ { field := sum.Field(i) field.SetFloat(field.Float() + statValue.Field(i).Float()) } } average := reflect.New(reflect.TypeOf(SystemStats{})).Elem() for i := 0; i < sum.NumField(); i++ { average.Field(i).SetFloat(twoDecimals(sum.Field(i).Float() / count)) } return average.Interface().(SystemStats) } // calculate the average stats of a list of container_stats records func averageContainerStats(records []*models.Record) (stats []ContainerStats) { sums := make(map[string]*ContainerStats) count := float64(len(records)) for _, record := range records { var stats []ContainerStats json.Unmarshal([]byte(record.Get("stats").(types.JsonRaw)), &stats) for _, stat := range stats { if _, ok := sums[stat.Name]; !ok { sums[stat.Name] = &ContainerStats{Name: stat.Name, Cpu: 0, Mem: 0} } sums[stat.Name].Cpu += stat.Cpu sums[stat.Name].Mem += stat.Mem } } for _, value := range sums { stats = append(stats, ContainerStats{ Name: value.Name, Cpu: twoDecimals(value.Cpu / count), Mem: twoDecimals(value.Mem / count), }) } return stats } /* Round float to two decimals */ func twoDecimals(value float64) float64 { return math.Round(value*100) / 100 } /* Delete records of specified collection and type that are older than timeLimit */ func deleteOldRecords(collection string, recordType string, timeLimit time.Duration) { timeLimitStamp := time.Now().UTC().Add(-timeLimit).Format("2006-01-02 15:04:05") records, _ := app.Dao().FindRecordsByExpr(collection, dbx.NewExp("type = {:type}", dbx.Params{"type": recordType}), dbx.NewExp("created < {:created}", dbx.Params{"created": timeLimitStamp}), ) for _, record := range records { if err := app.Dao().DeleteRecord(record); err != nil { log.Fatal(err) } } }