package models import ( "time" ) const ( NSummaryTypes uint8 = 99 SummaryProject uint8 = 0 SummaryLanguage uint8 = 1 SummaryEditor uint8 = 2 SummaryOS uint8 = 3 SummaryMachine uint8 = 4 ) const ( IntervalToday string = "today" IntervalYesterday string = "day" IntervalThisWeek string = "week" IntervalThisMonth string = "month" IntervalThisYear string = "year" IntervalPast7Days string = "7_days" IntervalPast30Days string = "30_days" IntervalPast12Months string = "12_months" IntervalAny string = "any" ) func Intervals() []string { return []string{ IntervalToday, IntervalYesterday, IntervalThisWeek, IntervalThisMonth, IntervalThisYear, IntervalPast7Days, IntervalPast30Days, IntervalPast12Months, IntervalAny, } } const UnknownSummaryKey = "unknown" type Summary struct { ID uint `json:"-" gorm:"primary_key"` User *User `json:"-" gorm:"not null"` UserID string `json:"user_id" gorm:"not null; index:idx_time_summary_user"` FromTime CustomTime `json:"from" gorm:"not null; type:timestamp; default:CURRENT_TIMESTAMP; index:idx_time_summary_user"` ToTime CustomTime `json:"to" gorm:"not null; type:timestamp; default:CURRENT_TIMESTAMP; index:idx_time_summary_user"` Projects []*SummaryItem `json:"projects"` Languages []*SummaryItem `json:"languages"` Editors []*SummaryItem `json:"editors"` OperatingSystems []*SummaryItem `json:"operating_systems"` Machines []*SummaryItem `json:"machines"` } type SummaryItem struct { ID uint `json:"-" gorm:"primary_key"` Summary *Summary `json:"-" gorm:"not null"` SummaryID uint `json:"-"` Type uint8 `json:"-"` Key string `json:"key"` Total time.Duration `json:"total"` } type SummaryItemContainer struct { Type uint8 Items []*SummaryItem } type SummaryViewModel struct { *Summary LanguageColors map[string]string Error string Success string ApiKey string } type SummaryParams struct { From time.Time To time.Time User *User Recompute bool } func SummaryTypes() []uint8 { return []uint8{SummaryProject, SummaryLanguage, SummaryEditor, SummaryOS, SummaryMachine} } func (s *Summary) Types() []uint8 { return SummaryTypes() } func (s *Summary) MappedItems() map[uint8]*[]*SummaryItem { return map[uint8]*[]*SummaryItem{ SummaryProject: &s.Projects, SummaryLanguage: &s.Languages, SummaryEditor: &s.Editors, SummaryOS: &s.OperatingSystems, SummaryMachine: &s.Machines, } } /* Augments the summary in a way that at least one item is present for every type. If a summary has zero items for a given type, but one or more for any of the other types, the total summary duration can be derived from those and inserted as a dummy-item with key "unknown" for the missing type. For instance, the machine type was introduced post hoc. Accordingly, no "machine"-information is present in the data for old heartbeats and summaries. If a user has two years of data without machine information and one day with such, a "machine"-chart plotted from that data will reference a way smaller absolute total amount of time than the other ones. To avoid having to modify persisted data retrospectively, i.e. inserting a dummy SummaryItem for the new type, such is generated dynamically here, considering the "machine" for all old heartbeats "unknown". */ func (s *Summary) FillUnknown() { types := s.Types() typeItems := s.MappedItems() missingTypes := make([]uint8, 0) for _, t := range types { if len(*typeItems[t]) == 0 { missingTypes = append(missingTypes, t) } } // can't proceed if entire summary is empty if len(missingTypes) == len(types) { return } timeSum := s.TotalTime() // construct dummy item for all missing types for _, t := range missingTypes { *typeItems[t] = append(*typeItems[t], &SummaryItem{ Type: t, Key: UnknownSummaryKey, Total: timeSum, }) } } func (s *Summary) TotalTime() time.Duration { var timeSum time.Duration mappedItems := s.MappedItems() // calculate total duration from any of the present sets of items for _, t := range s.Types() { if items := mappedItems[t]; len(*items) > 0 { for _, item := range *items { timeSum += item.Total } break } } return timeSum * time.Second } func (s *Summary) TotalTimeBy(entityType uint8) (timeSum time.Duration) { mappedItems := s.MappedItems() if items := mappedItems[entityType]; len(*items) > 0 { for _, item := range *items { timeSum = timeSum + item.Total*time.Second } } return timeSum } func (s *Summary) TotalTimeByKey(entityType uint8, key string) (timeSum time.Duration) { mappedItems := s.MappedItems() if items := mappedItems[entityType]; len(*items) > 0 { for _, item := range *items { if item.Key != key { continue } timeSum = timeSum + item.Total*time.Second } } return timeSum } func (s *Summary) TotalTimeByFilters(filter *Filters) (timeSum time.Duration) { for _, f := range filter.All() { timeSum += s.TotalTimeByKey(f.Type, f.Key) } return timeSum }