fin-check-api/handlers/statistics.go

107 lines
3.4 KiB
Go
Raw Permalink Normal View History

2024-11-29 19:59:22 +01:00
package handlers
import (
"fmt"
"log"
"slices"
"git.qowevisa.me/Qowevisa/fin-check-api/db"
"git.qowevisa.me/Qowevisa/fin-check-api/types"
2024-12-01 11:17:08 +01:00
"git.qowevisa.me/Qowevisa/fin-check-api/utils"
2024-11-29 19:59:22 +01:00
"github.com/gin-gonic/gin"
)
// @Summary Get all statisticss for user
// @Description Get all statisticss for user
// @Tags type
// @Produce json
// @Param Authorization header string true "Bearer token"
// @Success 200 {object} []types.StatsTypeCurrencyChart
// @Failure 401 {object} types.ErrorResponse
// @Failure 500 {object} types.ErrorResponse
// @Security ApiKeyAuth
// @Router /statistics/type [get]
func StatisticsGetAllSpendingsForTypes(c *gin.Context) {
userID, err := GetUserID(c)
if err != nil {
c.JSON(500, types.ErrorResponse{Message: err.Error()})
return
}
dbc := db.Connect()
var settingsTypeFilter []*db.SettingsTypeFilter
if err := dbc.Find(&settingsTypeFilter, db.SettingsTypeFilter{UserID: userID}).Error; err != nil {
c.JSON(500, types.ErrorResponse{Message: err.Error()})
return
}
var filerTypeIDs []uint
for _, typeFilter := range settingsTypeFilter {
filerTypeIDs = append(filerTypeIDs, typeFilter.TypeID)
}
2024-11-29 19:59:22 +01:00
var userTypes []*db.Type
if err := dbc.Find(&userTypes, db.Type{UserID: userID}).Error; err != nil {
c.JSON(500, types.ErrorResponse{Message: err.Error()})
return
}
var userExpenses []*db.Expense
if err := dbc.Not(map[string]interface{}{"type_id": filerTypeIDs}).Preload("Card.Currency").Find(&userExpenses, db.Expense{UserID: userID}).Error; err != nil {
2024-11-29 19:59:22 +01:00
c.JSON(500, types.ErrorResponse{Message: err.Error()})
return
}
currToChart := make(map[uint][]*db.Expense)
for _, expense := range userExpenses {
if expense.Card == nil || expense.Card.Currency == nil {
log.Printf("ERROR: db.Preload DID NOT WORKED OUT!\n")
c.JSON(500, types.ErrorResponse{Message: "Internal error. E.S.T.1"})
return
}
if val, exists := currToChart[expense.Card.CurrencyID]; !exists {
currToChart[expense.Card.CurrencyID] = []*db.Expense{}
currToChart[expense.Card.CurrencyID] = append(currToChart[expense.Card.CurrencyID], expense)
} else {
currToChart[expense.Card.CurrencyID] = append(val, expense)
}
}
var ret []types.StatsTypeCurrencyChart
for _, expenseArray := range currToChart {
if expenseArray[0].Card == nil || expenseArray[0].Card.Currency == nil {
log.Printf("ERROR: db.Preload DID NOT WORKED OUT!\n")
c.JSON(500, types.ErrorResponse{Message: "Internal error. E.S.T.2"})
return
}
currency := expenseArray[0].Card.Currency
typeToValue := make(map[uint]types.StatsType)
for _, expense := range expenseArray {
if val, exists := typeToValue[expense.TypeID]; !exists {
idx := slices.IndexFunc(userTypes, func(t *db.Type) bool { return t.ID == expense.TypeID })
typeForChart := userTypes[idx]
typeToValue[expense.TypeID] = types.StatsType{
Value: expense.Value,
Name: typeForChart.Name,
Color: typeForChart.Color,
}
} else {
val.Value += expense.Value
typeToValue[expense.TypeID] = val
}
}
2024-12-01 11:17:08 +01:00
var sum uint64 = 0
2024-11-29 19:59:22 +01:00
var elements []types.StatsType
for _, val := range typeToValue {
elements = append(elements, val)
2024-12-01 11:17:08 +01:00
sum += val.Value
2024-11-29 19:59:22 +01:00
}
2024-12-01 11:17:08 +01:00
slices.SortFunc(elements, func(a, b types.StatsType) int {
return utils.DescendingSort(a.Value, b.Value)
})
2024-11-29 19:59:22 +01:00
ret = append(ret, types.StatsTypeCurrencyChart{
CurrencyLabel: fmt.Sprintf("%s (%s)", currency.Symbol, currency.ISOName),
Elements: elements,
2024-12-01 11:17:08 +01:00
Total: sum,
2024-11-29 19:59:22 +01:00
})
}
c.JSON(200, ret)
}