From 29bf1427a0fc73e59eb5e40ffb5a15ecb9aedc02 Mon Sep 17 00:00:00 2001 From: qowevisa Date: Fri, 29 Nov 2024 20:59:22 +0200 Subject: [PATCH] Add statistics handler and endpoint --- cmd/http-server/main.go | 4 ++ handlers/statistics.go | 90 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 94 insertions(+) create mode 100644 handlers/statistics.go diff --git a/cmd/http-server/main.go b/cmd/http-server/main.go index ea96186..5119184 100644 --- a/cmd/http-server/main.go +++ b/cmd/http-server/main.go @@ -133,6 +133,10 @@ func main() { { currencyRoutes.GET("/all", handlers.CurrencyGetAll) } + statisticRoute := api.Group("/statistics", middleware.AuthMiddleware()) + { + statisticRoute.GET("/type", handlers.StatisticsGetAllSpendingsForTypes) + } } r.GET("/swagger/*any", ginSwagger.WrapHandler(swaggerFiles.Handler)) diff --git a/handlers/statistics.go b/handlers/statistics.go new file mode 100644 index 0000000..640debc --- /dev/null +++ b/handlers/statistics.go @@ -0,0 +1,90 @@ +package handlers + +import ( + "fmt" + "log" + "slices" + + "git.qowevisa.me/Qowevisa/fin-check-api/db" + "git.qowevisa.me/Qowevisa/fin-check-api/types" + "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 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.Preload("Card.Currency").Find(&userExpenses, db.Expense{UserID: userID}).Error; err != nil { + 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 + } + } + var elements []types.StatsType + for _, val := range typeToValue { + elements = append(elements, val) + } + + ret = append(ret, types.StatsTypeCurrencyChart{ + CurrencyLabel: fmt.Sprintf("%s (%s)", currency.Symbol, currency.ISOName), + Elements: elements, + }) + } + c.JSON(200, ret) +}