mirror of
https://gitlab.com/mbugroup/lti-api.git
synced 2026-05-24 15:25:43 +00:00
FEAT[BE]: implement expense report retrieval with filtering options
This commit is contained in:
@@ -28,10 +28,7 @@ type SalesDTO struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type PenjualanRealisasiResponseDTO struct {
|
type PenjualanRealisasiResponseDTO struct {
|
||||||
ProjectType string `json:"project_type"`
|
Sales []SalesDTO `json:"sales"`
|
||||||
FlockId uint `json:"flock_id"`
|
|
||||||
Period int `json:"period"`
|
|
||||||
Sales []SalesDTO `json:"sales"`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// === Mapper Functions ===
|
// === Mapper Functions ===
|
||||||
@@ -87,12 +84,10 @@ func ToSalesDTOs(e []entity.MarketingDeliveryProduct) []SalesDTO {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func ToPenjualanRealisasiResponseDTO(projectType string, projectFlockID uint, e []entity.MarketingDeliveryProduct) PenjualanRealisasiResponseDTO {
|
func ToPenjualanRealisasiResponseDTO(projectType string, projectFlockID uint, e []entity.MarketingDeliveryProduct) PenjualanRealisasiResponseDTO {
|
||||||
period := extractPeriodFromRealisasi(e)
|
|
||||||
return PenjualanRealisasiResponseDTO{
|
return PenjualanRealisasiResponseDTO{
|
||||||
ProjectType: projectType,
|
|
||||||
FlockId: projectFlockID,
|
Sales: ToSalesDTOs(e),
|
||||||
Period: period,
|
|
||||||
Sales: ToSalesDTOs(e),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,8 @@ import (
|
|||||||
|
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
"gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
||||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||||
|
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/validations"
|
||||||
|
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -13,6 +15,7 @@ type ExpenseRealizationRepository interface {
|
|||||||
IdExists(ctx context.Context, id uint64) (bool, error)
|
IdExists(ctx context.Context, id uint64) (bool, error)
|
||||||
GetByExpenseNonstockID(ctx context.Context, expenseNonstockID uint64) (*entity.ExpenseRealization, error)
|
GetByExpenseNonstockID(ctx context.Context, expenseNonstockID uint64) (*entity.ExpenseRealization, error)
|
||||||
GetByProjectFlockID(ctx context.Context, projectFlockID uint) ([]entity.ExpenseRealization, error)
|
GetByProjectFlockID(ctx context.Context, projectFlockID uint) ([]entity.ExpenseRealization, error)
|
||||||
|
GetAllWithFilters(ctx context.Context, offset, limit int, filters *validation.Query) ([]entity.ExpenseRealization, int64, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type ExpenseRealizationRepositoryImpl struct {
|
type ExpenseRealizationRepositoryImpl struct {
|
||||||
@@ -50,3 +53,83 @@ func (r *ExpenseRealizationRepositoryImpl) GetByProjectFlockID(ctx context.Conte
|
|||||||
Find(&realizations).Error
|
Find(&realizations).Error
|
||||||
return realizations, err
|
return realizations, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *ExpenseRealizationRepositoryImpl) GetAllWithFilters(ctx context.Context, offset, limit int, filters *validation.Query) ([]entity.ExpenseRealization, int64, error) {
|
||||||
|
var realizations []entity.ExpenseRealization
|
||||||
|
var total int64
|
||||||
|
|
||||||
|
db := r.DB().WithContext(ctx).
|
||||||
|
Model(&entity.ExpenseRealization{}).
|
||||||
|
Preload("ExpenseNonstock", func(db *gorm.DB) *gorm.DB {
|
||||||
|
return db.
|
||||||
|
Preload("Expense").
|
||||||
|
Preload("Expense.Supplier").
|
||||||
|
Preload("Kandang").
|
||||||
|
Preload("Kandang.Location").
|
||||||
|
Preload("Nonstock")
|
||||||
|
}).
|
||||||
|
Joins("JOIN expense_nonstocks ON expense_nonstocks.id = expense_realizations.expense_nonstock_id").
|
||||||
|
Joins("JOIN expenses ON expenses.id = expense_nonstocks.expense_id").
|
||||||
|
Joins("LEFT JOIN suppliers ON suppliers.id = expenses.supplier_id")
|
||||||
|
|
||||||
|
if filters.Search != "" {
|
||||||
|
db = db.Where("expenses.category LIKE ? OR expenses.reference_number LIKE ? OR expenses.po_number LIKE ? OR expenses.notes LIKE ? OR suppliers.name LIKE ?",
|
||||||
|
"%"+filters.Search+"%", "%"+filters.Search+"%", "%"+filters.Search+"%", "%"+filters.Search+"%", "%"+filters.Search+"%")
|
||||||
|
}
|
||||||
|
|
||||||
|
if filters.Category != "" {
|
||||||
|
db = db.Where("expenses.category = ?", filters.Category)
|
||||||
|
}
|
||||||
|
|
||||||
|
if filters.SupplierId > 0 {
|
||||||
|
db = db.Where("expenses.supplier_id = ?", filters.SupplierId)
|
||||||
|
}
|
||||||
|
|
||||||
|
if filters.KandangId > 0 {
|
||||||
|
db = db.Where("expense_nonstocks.kandang_id = ?", filters.KandangId)
|
||||||
|
}
|
||||||
|
|
||||||
|
if filters.ProjectFlockKandangId > 0 {
|
||||||
|
db = db.Where("expense_nonstocks.project_flock_kandang_id = ?", filters.ProjectFlockKandangId)
|
||||||
|
}
|
||||||
|
|
||||||
|
if filters.NonstockId > 0 {
|
||||||
|
db = db.Where("expense_nonstocks.nonstock_id = ?", filters.NonstockId)
|
||||||
|
}
|
||||||
|
|
||||||
|
locationID := filters.LocationId
|
||||||
|
areaID := filters.AreaId
|
||||||
|
|
||||||
|
if locationID > 0 || areaID > 0 {
|
||||||
|
db = db.Joins("JOIN kandangs ON kandangs.id = expense_nonstocks.kandang_id")
|
||||||
|
|
||||||
|
if locationID > 0 {
|
||||||
|
db = db.Where("kandangs.location_id = ?", uint(locationID))
|
||||||
|
}
|
||||||
|
|
||||||
|
if areaID > 0 {
|
||||||
|
db = db.Joins("JOIN locations ON locations.id = kandangs.location_id").
|
||||||
|
Where("locations.area_id = ?", uint(areaID))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if filters.RealizationDate != "" {
|
||||||
|
if realizationDate, err := utils.ParseDateString(filters.RealizationDate); err == nil {
|
||||||
|
db = db.Where("DATE(expenses.realization_date) = ?", realizationDate)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := db.Count(&total).Error; err != nil {
|
||||||
|
return nil, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := db.
|
||||||
|
Offset(offset).
|
||||||
|
Limit(limit).
|
||||||
|
Order("expense_realizations.created_at DESC").
|
||||||
|
Find(&realizations).Error; err != nil {
|
||||||
|
return nil, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return realizations, total, nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -229,7 +229,6 @@ func (s *adjustmentService) AdjustmentHistory(c *fiber.Ctx, query *validation.Qu
|
|||||||
|
|
||||||
isWarehousesExist, err := s.WarehouseRepo.IdExists(c.Context(), uint(query.WarehouseID))
|
isWarehousesExist, err := s.WarehouseRepo.IdExists(c.Context(), uint(query.WarehouseID))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.Log.Errorf("Failed to check warehouse existence: %+v", err)
|
|
||||||
return nil, 0, fiber.NewError(fiber.StatusInternalServerError, "Failed to validate warehouse")
|
return nil, 0, fiber.NewError(fiber.StatusInternalServerError, "Failed to validate warehouse")
|
||||||
}
|
}
|
||||||
if query.WarehouseID > 0 && !isWarehousesExist {
|
if query.WarehouseID > 0 && !isWarehousesExist {
|
||||||
|
|||||||
+1
-1
@@ -115,7 +115,7 @@ func (r *ProductWarehouseRepositoryImpl) GetLatestByCategoryCodeAndWarehouseID(c
|
|||||||
Joins("JOIN products ON products.id = product_warehouses.product_id").
|
Joins("JOIN products ON products.id = product_warehouses.product_id").
|
||||||
Joins("JOIN product_categories ON product_categories.id = products.product_category_id").
|
Joins("JOIN product_categories ON product_categories.id = products.product_category_id").
|
||||||
Where("product_categories.code = ? AND product_warehouses.warehouse_id = ?", categoryCode, warehouseId).
|
Where("product_categories.code = ? AND product_warehouses.warehouse_id = ?", categoryCode, warehouseId).
|
||||||
Order("product_warehouses.id DESC").
|
Order("product_warehouses.created_at DESC").
|
||||||
Preload("Product").Preload("Warehouse").
|
Preload("Product").Preload("Warehouse").
|
||||||
First(&productWarehouse).Error
|
First(&productWarehouse).Error
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -281,7 +281,6 @@ func (u *ProjectflockController) LookupProjectFlockKandang(c *fiber.Ctx) error {
|
|||||||
dtoResult := dto.ToProjectFlockKandangDTO(*result)
|
dtoResult := dto.ToProjectFlockKandangDTO(*result)
|
||||||
dtoResult.AvailableQuantity = float64(availableStock)
|
dtoResult.AvailableQuantity = float64(availableStock)
|
||||||
|
|
||||||
// populate available quantity for each kandang inside project_flock
|
|
||||||
if dtoResult.ProjectFlock != nil {
|
if dtoResult.ProjectFlock != nil {
|
||||||
for i := range dtoResult.ProjectFlock.Kandangs {
|
for i := range dtoResult.ProjectFlock.Kandangs {
|
||||||
kand := &dtoResult.ProjectFlock.Kandangs[i]
|
kand := &dtoResult.ProjectFlock.Kandangs[i]
|
||||||
@@ -292,7 +291,7 @@ func (u *ProjectflockController) LookupProjectFlockKandang(c *fiber.Ctx) error {
|
|||||||
kand.AvailableQuantity = q
|
kand.AvailableQuantity = q
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// remove inner kandangs from project_flock to avoid duplication
|
|
||||||
dtoResult.ProjectFlock.Kandangs = nil
|
dtoResult.ProjectFlock.Kandangs = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ package controller
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"math"
|
"math"
|
||||||
"strconv"
|
|
||||||
|
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/modules/repports/dto"
|
"gitlab.com/mbugroup/lti-api.git/internal/modules/repports/dto"
|
||||||
service "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/services"
|
service "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/services"
|
||||||
@@ -22,27 +21,35 @@ func NewRepportController(repportService service.RepportService) *RepportControl
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *RepportController) GetAll(ctx *fiber.Ctx) error {
|
func (c *RepportController) GetExpense(ctx *fiber.Ctx) error {
|
||||||
query := &validation.Query{
|
query := &validation.Query{
|
||||||
Page: ctx.QueryInt("page", 1),
|
Page: ctx.QueryInt("page", 1),
|
||||||
Limit: ctx.QueryInt("limit", 10),
|
Limit: ctx.QueryInt("limit", 10),
|
||||||
Search: ctx.Query("search", ""),
|
Search: ctx.Query("search", ""),
|
||||||
|
Category: ctx.Query("category", ""),
|
||||||
|
SupplierId: int64(ctx.QueryInt("supplier_id", 0)),
|
||||||
|
KandangId: int64(ctx.QueryInt("kandang_id", 0)),
|
||||||
|
ProjectFlockKandangId: int64(ctx.QueryInt("project_flock_kandang_id", 0)),
|
||||||
|
NonstockId: int64(ctx.QueryInt("nonstock_id", 0)),
|
||||||
|
AreaId: int64(ctx.QueryInt("area_id", 0)),
|
||||||
|
LocationId: int64(ctx.QueryInt("location_id", 0)),
|
||||||
|
RealizationDate: ctx.Query("realization_date", ""),
|
||||||
}
|
}
|
||||||
|
|
||||||
if query.Page < 1 || query.Limit < 1 {
|
if query.Page < 1 || query.Limit < 1 {
|
||||||
return fiber.NewError(fiber.StatusBadRequest, "page and limit must be greater than 0")
|
return fiber.NewError(fiber.StatusBadRequest, "page and limit must be greater than 0")
|
||||||
}
|
}
|
||||||
|
|
||||||
result, totalResults, err := c.RepportService.GetAll(ctx, query)
|
result, totalResults, err := c.RepportService.GetExpense(ctx, query)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
return ctx.Status(fiber.StatusOK).
|
return ctx.Status(fiber.StatusOK).
|
||||||
JSON(response.SuccessWithPaginate[dto.RepportListDTO]{
|
JSON(response.SuccessWithPaginate[dto.RepportExpenseListDTO]{
|
||||||
Code: fiber.StatusOK,
|
Code: fiber.StatusOK,
|
||||||
Status: "success",
|
Status: "success",
|
||||||
Message: "Get all reports successfully",
|
Message: "Get expense report successfully",
|
||||||
Meta: response.Meta{
|
Meta: response.Meta{
|
||||||
Page: query.Page,
|
Page: query.Page,
|
||||||
Limit: query.Limit,
|
Limit: query.Limit,
|
||||||
@@ -52,47 +59,3 @@ func (c *RepportController) GetAll(ctx *fiber.Ctx) error {
|
|||||||
Data: result,
|
Data: result,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *RepportController) GetOne(ctx *fiber.Ctx) error {
|
|
||||||
param := ctx.Params("id")
|
|
||||||
|
|
||||||
id, err := strconv.Atoi(param)
|
|
||||||
if err != nil {
|
|
||||||
return fiber.NewError(fiber.StatusBadRequest, "Invalid Id")
|
|
||||||
}
|
|
||||||
|
|
||||||
result, err := c.RepportService.GetOne(ctx, uint(id))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return ctx.Status(fiber.StatusOK).
|
|
||||||
JSON(response.Success{
|
|
||||||
Code: fiber.StatusOK,
|
|
||||||
Status: "success",
|
|
||||||
Message: "Get report successfully",
|
|
||||||
Data: result,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *RepportController) GetExpense(ctx *fiber.Ctx) error {
|
|
||||||
param := ctx.Params("id")
|
|
||||||
|
|
||||||
id, err := strconv.Atoi(param)
|
|
||||||
if err != nil {
|
|
||||||
return fiber.NewError(fiber.StatusBadRequest, "Invalid Id")
|
|
||||||
}
|
|
||||||
|
|
||||||
result, err := c.RepportService.GetOne(ctx, uint(id))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return ctx.Status(fiber.StatusOK).
|
|
||||||
JSON(response.Success{
|
|
||||||
Code: fiber.StatusOK,
|
|
||||||
Status: "success",
|
|
||||||
Message: "Get report successfully",
|
|
||||||
Data: result,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,16 +0,0 @@
|
|||||||
package dto
|
|
||||||
|
|
||||||
import "time"
|
|
||||||
|
|
||||||
// === DTO Structs ===
|
|
||||||
|
|
||||||
type RepportListDTO struct {
|
|
||||||
Id uint `json:"id"`
|
|
||||||
Name string `json:"name"`
|
|
||||||
CreatedAt time.Time `json:"created_at"`
|
|
||||||
UpdatedAt time.Time `json:"updated_at"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type RepportDetailDTO struct {
|
|
||||||
RepportListDTO
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,173 @@
|
|||||||
|
package dto
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||||
|
approvalDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/approvals/dto"
|
||||||
|
kandangDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/kandangs/dto"
|
||||||
|
nonstockDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/nonstocks/dto"
|
||||||
|
supplierDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/suppliers/dto"
|
||||||
|
)
|
||||||
|
|
||||||
|
// === DTO Structs ===
|
||||||
|
|
||||||
|
type RepportExpenseBaseDTO struct {
|
||||||
|
Id uint64 `json:"id"`
|
||||||
|
ReferenceNumber string `json:"reference_number"`
|
||||||
|
PoNumber string `json:"po_number"`
|
||||||
|
Category string `json:"category"`
|
||||||
|
Supplier *supplierDTO.SupplierRelationDTO `json:"supplier,omitempty"`
|
||||||
|
RealizationDate *time.Time `json:"realization_date,omitempty"`
|
||||||
|
TransactionDate time.Time `json:"transaction_date"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type RepportExpensePengajuanDTO struct {
|
||||||
|
Id uint64 `json:"id"`
|
||||||
|
ExpenseId *uint64 `json:"expense_id,omitempty"`
|
||||||
|
ProjectFlockKandangId *uint64 `json:"project_flock_kandang_id,omitempty"`
|
||||||
|
Qty float64 `json:"qty"`
|
||||||
|
Price float64 `json:"price"`
|
||||||
|
Notes string `json:"notes"`
|
||||||
|
Nonstock *nonstockDTO.NonstockRelationDTO `json:"nonstock,omitempty"`
|
||||||
|
Kandang *kandangDTO.KandangRelationDTO `json:"kandang,omitempty"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type RepportExpenseRealisasiDTO struct {
|
||||||
|
Id *uint64 `json:"id,omitempty"`
|
||||||
|
ExpenseNonstockId *uint64 `json:"expense_nonstock_id,omitempty"`
|
||||||
|
Qty float64 `json:"qty"`
|
||||||
|
Price float64 `json:"price"`
|
||||||
|
Notes string `json:"notes"`
|
||||||
|
Nonstock *nonstockDTO.NonstockRelationDTO `json:"nonstock,omitempty"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type RepportExpenseListDTO struct {
|
||||||
|
RepportExpenseBaseDTO
|
||||||
|
Pengajuan RepportExpensePengajuanDTO `json:"pengajuan"`
|
||||||
|
Realisasi RepportExpenseRealisasiDTO `json:"realisasi"`
|
||||||
|
TotalPengajuan float64 `json:"total_pengajuan"`
|
||||||
|
TotalRealisasi float64 `json:"total_realisasi"`
|
||||||
|
LatestApproval *approvalDTO.ApprovalRelationDTO `json:"latest_approval,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// === MAPPERS ===
|
||||||
|
|
||||||
|
func ToRepportExpenseBaseDTO(e *entity.Expense) RepportExpenseBaseDTO {
|
||||||
|
var realizationDate *time.Time
|
||||||
|
if !e.RealizationDate.IsZero() {
|
||||||
|
realizationDate = &e.RealizationDate
|
||||||
|
}
|
||||||
|
|
||||||
|
var supplier *supplierDTO.SupplierRelationDTO
|
||||||
|
if e.Supplier != nil && e.Supplier.Id != 0 {
|
||||||
|
mapped := supplierDTO.ToSupplierRelationDTO(*e.Supplier)
|
||||||
|
supplier = &mapped
|
||||||
|
}
|
||||||
|
|
||||||
|
return RepportExpenseBaseDTO{
|
||||||
|
Id: e.Id,
|
||||||
|
ReferenceNumber: e.ReferenceNumber,
|
||||||
|
PoNumber: e.PoNumber,
|
||||||
|
Category: e.Category,
|
||||||
|
Supplier: supplier,
|
||||||
|
RealizationDate: realizationDate,
|
||||||
|
TransactionDate: e.TransactionDate,
|
||||||
|
CreatedAt: e.CreatedAt,
|
||||||
|
UpdatedAt: e.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ToRepportExpensePengajuanDTO(ns *entity.ExpenseNonstock) RepportExpensePengajuanDTO {
|
||||||
|
var nonstock *nonstockDTO.NonstockRelationDTO
|
||||||
|
if ns.Nonstock != nil && ns.Nonstock.Id != 0 {
|
||||||
|
mapped := nonstockDTO.ToNonstockRelationDTO(*ns.Nonstock)
|
||||||
|
nonstock = &mapped
|
||||||
|
}
|
||||||
|
|
||||||
|
var kandang *kandangDTO.KandangRelationDTO
|
||||||
|
if ns.Kandang != nil && ns.Kandang.Id != 0 {
|
||||||
|
mapped := kandangDTO.ToKandangRelationDTO(*ns.Kandang)
|
||||||
|
kandang = &mapped
|
||||||
|
}
|
||||||
|
|
||||||
|
return RepportExpensePengajuanDTO{
|
||||||
|
Id: ns.Id,
|
||||||
|
ExpenseId: ns.ExpenseId,
|
||||||
|
ProjectFlockKandangId: ns.ProjectFlockKandangId,
|
||||||
|
Qty: ns.Qty,
|
||||||
|
Price: ns.Price,
|
||||||
|
Notes: ns.Notes,
|
||||||
|
Nonstock: nonstock,
|
||||||
|
Kandang: kandang,
|
||||||
|
CreatedAt: ns.CreatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ToRepportExpenseRealisasiDTO(r *entity.ExpenseRealization) RepportExpenseRealisasiDTO {
|
||||||
|
var nonstock *nonstockDTO.NonstockRelationDTO
|
||||||
|
if r.ExpenseNonstock != nil && r.ExpenseNonstock.Nonstock != nil && r.ExpenseNonstock.Nonstock.Id != 0 {
|
||||||
|
mapped := nonstockDTO.ToNonstockRelationDTO(*r.ExpenseNonstock.Nonstock)
|
||||||
|
nonstock = &mapped
|
||||||
|
}
|
||||||
|
|
||||||
|
return RepportExpenseRealisasiDTO{
|
||||||
|
Id: r.ExpenseNonstockId,
|
||||||
|
ExpenseNonstockId: r.ExpenseNonstockId,
|
||||||
|
Qty: r.Qty,
|
||||||
|
Price: r.Price,
|
||||||
|
Notes: r.Notes,
|
||||||
|
Nonstock: nonstock,
|
||||||
|
CreatedAt: r.CreatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ToRepportExpenseListDTO(baseDTO RepportExpenseBaseDTO, ns *entity.ExpenseNonstock, latestApproval *approvalDTO.ApprovalRelationDTO) RepportExpenseListDTO {
|
||||||
|
var realisasi RepportExpenseRealisasiDTO
|
||||||
|
if ns.Realization != nil {
|
||||||
|
realisasi = ToRepportExpenseRealisasiDTO(ns.Realization)
|
||||||
|
}
|
||||||
|
|
||||||
|
totalPengajuan := ns.Qty * ns.Price
|
||||||
|
totalRealisasi := float64(0)
|
||||||
|
if ns.Realization != nil {
|
||||||
|
totalRealisasi = ns.Realization.Qty * ns.Realization.Price
|
||||||
|
}
|
||||||
|
|
||||||
|
return RepportExpenseListDTO{
|
||||||
|
RepportExpenseBaseDTO: baseDTO,
|
||||||
|
Pengajuan: ToRepportExpensePengajuanDTO(ns),
|
||||||
|
Realisasi: realisasi,
|
||||||
|
TotalPengajuan: totalPengajuan,
|
||||||
|
TotalRealisasi: totalRealisasi,
|
||||||
|
LatestApproval: latestApproval,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ToRepportExpenseListDTOs(realizations []entity.ExpenseRealization) []RepportExpenseListDTO {
|
||||||
|
result := make([]RepportExpenseListDTO, 0, len(realizations))
|
||||||
|
|
||||||
|
for _, realization := range realizations {
|
||||||
|
if realization.ExpenseNonstock == nil || realization.ExpenseNonstock.Expense == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
expense := realization.ExpenseNonstock.Expense
|
||||||
|
baseDTO := ToRepportExpenseBaseDTO(expense)
|
||||||
|
|
||||||
|
var latestApproval *approvalDTO.ApprovalRelationDTO
|
||||||
|
if expense.LatestApproval != nil {
|
||||||
|
mapped := approvalDTO.ToApprovalDTO(*expense.LatestApproval)
|
||||||
|
latestApproval = &mapped
|
||||||
|
}
|
||||||
|
|
||||||
|
dto := ToRepportExpenseListDTO(baseDTO, realization.ExpenseNonstock, latestApproval)
|
||||||
|
result = append(result, dto)
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
|
}
|
||||||
@@ -5,6 +5,8 @@ import (
|
|||||||
"github.com/gofiber/fiber/v2"
|
"github.com/gofiber/fiber/v2"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
|
|
||||||
|
commonRepo "gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
||||||
|
approvalService "gitlab.com/mbugroup/lti-api.git/internal/common/service"
|
||||||
sRepport "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/services"
|
sRepport "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/services"
|
||||||
|
|
||||||
expenseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/expenses/repositories"
|
expenseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/expenses/repositories"
|
||||||
@@ -13,11 +15,12 @@ import (
|
|||||||
type RepportModule struct{}
|
type RepportModule struct{}
|
||||||
|
|
||||||
func (RepportModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate *validator.Validate) {
|
func (RepportModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate *validator.Validate) {
|
||||||
// Initialize expense realization repository
|
|
||||||
expRealizationRepo := expenseRepo.NewExpenseRealizationRepository(db)
|
|
||||||
|
|
||||||
// Initialize report service with expense realization repo
|
expenseRealizationRepository := expenseRepo.NewExpenseRealizationRepository(db)
|
||||||
repportService := sRepport.NewRepportService(validate, expRealizationRepo)
|
approvalRepository := commonRepo.NewApprovalRepository(db)
|
||||||
|
|
||||||
|
approvalSvc := approvalService.NewApprovalService(approvalRepository)
|
||||||
|
repportService := sRepport.NewRepportService(validate, expenseRealizationRepository, approvalSvc)
|
||||||
|
|
||||||
RepportRoutes(router, repportService)
|
RepportRoutes(router, repportService)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
package repports
|
package repports
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
|
||||||
controller "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/controllers"
|
controller "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/controllers"
|
||||||
repport "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/services"
|
repport "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/services"
|
||||||
|
|
||||||
@@ -13,8 +12,5 @@ func RepportRoutes(v1 fiber.Router, s repport.RepportService) {
|
|||||||
|
|
||||||
route := v1.Group("/repports")
|
route := v1.Group("/repports")
|
||||||
|
|
||||||
route.Get("/", ctrl.GetAll)
|
|
||||||
route.Get("/:id", ctrl.GetOne)
|
|
||||||
|
|
||||||
route.Get("expense", ctrl.GetExpense)
|
route.Get("expense", ctrl.GetExpense)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,106 +1,74 @@
|
|||||||
package service
|
package service
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/modules/repports/dto"
|
"gitlab.com/mbugroup/lti-api.git/internal/modules/repports/dto"
|
||||||
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/validations"
|
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/repports/validations"
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
||||||
|
|
||||||
|
approvalService "gitlab.com/mbugroup/lti-api.git/internal/common/service"
|
||||||
|
approvalDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/approvals/dto"
|
||||||
expenseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/expenses/repositories"
|
expenseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/expenses/repositories"
|
||||||
|
|
||||||
"github.com/go-playground/validator/v10"
|
"github.com/go-playground/validator/v10"
|
||||||
"github.com/gofiber/fiber/v2"
|
"github.com/gofiber/fiber/v2"
|
||||||
"github.com/sirupsen/logrus"
|
"github.com/sirupsen/logrus"
|
||||||
|
"gorm.io/gorm"
|
||||||
)
|
)
|
||||||
|
|
||||||
type RepportService interface {
|
type RepportService interface {
|
||||||
GetAll(ctx *fiber.Ctx, params *validation.Query) ([]dto.RepportListDTO, int64, error)
|
GetExpense(ctx *fiber.Ctx, params *validation.Query) ([]dto.RepportExpenseListDTO, int64, error)
|
||||||
GetOne(ctx *fiber.Ctx, id uint) (*dto.RepportListDTO, error)
|
|
||||||
GetExpense(ctx *fiber.Ctx, id uint) (*dto.RepportListDTO, error)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type repportService struct {
|
type repportService struct {
|
||||||
Log *logrus.Logger
|
Log *logrus.Logger
|
||||||
Validate *validator.Validate
|
Validate *validator.Validate
|
||||||
dummyData map[uint]dto.RepportListDTO
|
|
||||||
ExpenseRealizationRepo expenseRepo.ExpenseRealizationRepository
|
ExpenseRealizationRepo expenseRepo.ExpenseRealizationRepository
|
||||||
|
ApprovalSvc approvalService.ApprovalService
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewRepportService(validate *validator.Validate, expenseRealizationRepo expenseRepo.ExpenseRealizationRepository) RepportService {
|
func NewRepportService(validate *validator.Validate, expenseRealizationRepo expenseRepo.ExpenseRealizationRepository, approvalSvc approvalService.ApprovalService) RepportService {
|
||||||
// Initialize with dummy data
|
|
||||||
now := time.Now()
|
|
||||||
dummyData := map[uint]dto.RepportListDTO{
|
|
||||||
1: {
|
|
||||||
Id: 1,
|
|
||||||
Name: "Sales Report",
|
|
||||||
CreatedAt: now,
|
|
||||||
UpdatedAt: now,
|
|
||||||
},
|
|
||||||
2: {
|
|
||||||
Id: 2,
|
|
||||||
Name: "Inventory Report",
|
|
||||||
CreatedAt: now,
|
|
||||||
UpdatedAt: now,
|
|
||||||
},
|
|
||||||
3: {
|
|
||||||
Id: 3,
|
|
||||||
Name: "Production Report",
|
|
||||||
CreatedAt: now,
|
|
||||||
UpdatedAt: now,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
return &repportService{
|
return &repportService{
|
||||||
Log: utils.Log,
|
Log: utils.Log,
|
||||||
Validate: validate,
|
Validate: validate,
|
||||||
dummyData: dummyData,
|
|
||||||
ExpenseRealizationRepo: expenseRealizationRepo,
|
ExpenseRealizationRepo: expenseRealizationRepo,
|
||||||
|
ApprovalSvc: approvalSvc,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *repportService) GetAll(c *fiber.Ctx, params *validation.Query) ([]dto.RepportListDTO, int64, error) {
|
func (s *repportService) GetExpense(c *fiber.Ctx, params *validation.Query) ([]dto.RepportExpenseListDTO, int64, error) {
|
||||||
if err := s.Validate.Struct(params); err != nil {
|
if err := s.Validate.Struct(params); err != nil {
|
||||||
return nil, 0, err
|
return nil, 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert map to slice
|
|
||||||
var results []dto.RepportListDTO
|
|
||||||
for _, v := range s.dummyData {
|
|
||||||
// Apply search filter if provided
|
|
||||||
if params.Search != "" && !strings.Contains(strings.ToLower(v.Name), strings.ToLower(params.Search)) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
results = append(results, v)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Apply pagination
|
|
||||||
total := int64(len(results))
|
|
||||||
offset := (params.Page - 1) * params.Limit
|
offset := (params.Page - 1) * params.Limit
|
||||||
|
|
||||||
if offset >= int(total) {
|
realizations, total, err := s.ExpenseRealizationRepo.GetAllWithFilters(c.Context(), offset, params.Limit, params)
|
||||||
return []dto.RepportListDTO{}, total, nil
|
if err != nil {
|
||||||
|
s.Log.Errorf("GetAllWithFilters error: %v", err)
|
||||||
|
return nil, 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
end := offset + params.Limit
|
result := dto.ToRepportExpenseListDTOs(realizations)
|
||||||
if end > int(total) {
|
|
||||||
end = int(total)
|
expenseIDs := make([]uint, 0, len(result))
|
||||||
|
for i := range result {
|
||||||
|
expenseIDs = append(expenseIDs, uint(result[i].Id))
|
||||||
}
|
}
|
||||||
|
|
||||||
return results[offset:end], total, nil
|
approvals, err := s.ApprovalSvc.LatestByTargets(c.Context(), utils.ApprovalWorkflowExpense, expenseIDs, func(db *gorm.DB) *gorm.DB {
|
||||||
}
|
return db.Preload("ActionUser")
|
||||||
|
})
|
||||||
func (s *repportService) GetOne(c *fiber.Ctx, id uint) (*dto.RepportListDTO, error) {
|
if err != nil {
|
||||||
if data, ok := s.dummyData[id]; ok {
|
s.Log.Warnf("LatestByTargets error: %v", err)
|
||||||
return &data, nil
|
}
|
||||||
}
|
|
||||||
return nil, fiber.NewError(fiber.StatusNotFound, "Report not found")
|
for i := range result {
|
||||||
}
|
expenseIDAsUint := uint(result[i].Id)
|
||||||
|
if approval, exists := approvals[expenseIDAsUint]; exists && approval != nil {
|
||||||
func (s *repportService) GetExpense(c *fiber.Ctx, id uint) (*dto.RepportListDTO, error) {
|
mapped := approvalDTO.ToApprovalDTO(*approval)
|
||||||
if data, ok := s.dummyData[id]; ok {
|
result[i].LatestApproval = &mapped
|
||||||
return &data, nil
|
}
|
||||||
}
|
}
|
||||||
return nil, fiber.NewError(fiber.StatusNotFound, "Report not found")
|
|
||||||
|
return result, total, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,16 @@
|
|||||||
package validation
|
package validation
|
||||||
|
|
||||||
type Query struct {
|
type Query struct {
|
||||||
Page int `query:"page" validate:"omitempty,number,min=1,gt=0"`
|
Page int `query:"page" validate:"omitempty,min=1,gt=0"`
|
||||||
Limit int `query:"limit" validate:"omitempty,number,min=1,max=100,gt=0"`
|
Limit int `query:"limit" validate:"omitempty,min=1,max=100,gt=0"`
|
||||||
Search string `query:"search" validate:"omitempty,max=50"`
|
Search string `query:"search" validate:"omitempty,max=100"`
|
||||||
|
Category string `query:"category" validate:"omitempty,oneof=BOP NON-BOP"`
|
||||||
|
SupplierId int64 `query:"supplier_id" validate:"omitempty"`
|
||||||
|
KandangId int64 `query:"kandang_id" validate:"omitempty"`
|
||||||
|
ProjectFlockKandangId int64 `query:"project_flock_kandang_id" validate:"omitempty"`
|
||||||
|
ProjectFlockId int64 `query:"project_flock_id" validate:"omitempty"`
|
||||||
|
NonstockId int64 `query:"nonstock_id" validate:"omitempty"`
|
||||||
|
AreaId int64 `query:"area_id" validate:"omitempty"`
|
||||||
|
LocationId int64 `query:"location_id" validate:"omitempty"`
|
||||||
|
RealizationDate string `query:"realization_date" validate:"omitempty"`
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user