mirror of
https://gitlab.com/mbugroup/lti-api.git
synced 2026-06-09 15:07:49 +00:00
ini api per farm
This commit is contained in:
@@ -49,6 +49,7 @@ type RepportService interface {
|
||||
GetPurchaseSupplier(ctx *fiber.Ctx, params *validation.PurchaseSupplierQuery) ([]dto.PurchaseSupplierDTO, int64, error)
|
||||
GetDebtSupplier(ctx *fiber.Ctx, params *validation.DebtSupplierQuery) ([]dto.DebtSupplierDTO, int64, error)
|
||||
GetHppPerKandang(ctx *fiber.Ctx) (*dto.HppPerKandangResponseData, *dto.HppPerKandangMetaDTO, error)
|
||||
GetHppPerFarm(ctx *fiber.Ctx) (*dto.HppPerFarmResponseData, *dto.HppPerFarmMetaDTO, error)
|
||||
GetHppV2Breakdown(ctx *fiber.Ctx, params *validation.HppV2BreakdownQuery) (*approvalService.HppV2Breakdown, error)
|
||||
GetProductionResult(ctx *fiber.Ctx, params *validation.ProductionResultQuery) ([]dto.ProductionResultDTO, int64, error)
|
||||
GetCustomerPayment(ctx *fiber.Ctx, params *validation.CustomerPaymentQuery) ([]dto.CustomerPaymentReportItem, int64, error)
|
||||
@@ -73,6 +74,7 @@ type repportService struct {
|
||||
PurchaseSupplierRepo repportRepo.PurchaseSupplierRepository
|
||||
DebtSupplierRepo repportRepo.DebtSupplierRepository
|
||||
HppPerKandangRepo repportRepo.HppPerKandangRepository
|
||||
HppPerFarmRepo repportRepo.HppPerFarmRepository
|
||||
ProductionResultRepo repportRepo.ProductionResultRepository
|
||||
CustomerPaymentRepo repportRepo.CustomerPaymentRepository
|
||||
BalanceMonitoringRepo repportRepo.BalanceMonitoringRepository
|
||||
@@ -106,6 +108,7 @@ func NewRepportService(
|
||||
purchaseSupplierRepo repportRepo.PurchaseSupplierRepository,
|
||||
debtSupplierRepo repportRepo.DebtSupplierRepository,
|
||||
hppPerKandangRepo repportRepo.HppPerKandangRepository,
|
||||
hppPerFarmRepo repportRepo.HppPerFarmRepository,
|
||||
productionResultRepo repportRepo.ProductionResultRepository,
|
||||
customerPaymentRepo repportRepo.CustomerPaymentRepository,
|
||||
balanceMonitoringRepo repportRepo.BalanceMonitoringRepository,
|
||||
@@ -130,6 +133,7 @@ func NewRepportService(
|
||||
PurchaseSupplierRepo: purchaseSupplierRepo,
|
||||
DebtSupplierRepo: debtSupplierRepo,
|
||||
HppPerKandangRepo: hppPerKandangRepo,
|
||||
HppPerFarmRepo: hppPerFarmRepo,
|
||||
ProductionResultRepo: productionResultRepo,
|
||||
CustomerPaymentRepo: customerPaymentRepo,
|
||||
BalanceMonitoringRepo: balanceMonitoringRepo,
|
||||
@@ -2945,6 +2949,480 @@ func (s *repportService) parseHppPerKandangQuery(ctx *fiber.Ctx) (*validation.Hp
|
||||
return params, filters, nil
|
||||
}
|
||||
|
||||
const (
|
||||
hppPerFarmProductionScope = "production_cost"
|
||||
hppPerFarmComponentDepreciation = "DEPRECIATION"
|
||||
hppPerFarmComponentPakan = "PAKAN"
|
||||
hppPerFarmComponentOvk = "OVK"
|
||||
hppPerFarmComponentBopRegular = "BOP_REGULAR"
|
||||
hppPerFarmComponentBopEkspedisi = "BOP_EKSPEDISI"
|
||||
hppPerFarmMaxRangeDays = 366
|
||||
)
|
||||
|
||||
// GetHppPerFarm builds the HPP-per-farm report: it groups all LAYING project
|
||||
// flocks by location/farm over [start_date, end_date] and reports, per farm,
|
||||
// the total cost (pakan + ovk + bop + depreciation) and two cost-per-kg figures
|
||||
// — one against egg weight produced (recording_eggs) and one against egg weight
|
||||
// sold/delivered (marketing delivery orders). DOC/pullet cost is informational
|
||||
// only (it is expensed through depreciation, so it is NOT added to total cost).
|
||||
func (s *repportService) GetHppPerFarm(ctx *fiber.Ctx) (*dto.HppPerFarmResponseData, *dto.HppPerFarmMetaDTO, error) {
|
||||
params, filters, err := s.parseHppPerFarmQuery(ctx)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
if err := s.Validate.Struct(params); err != nil {
|
||||
return nil, nil, fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||
}
|
||||
if s.HppPerFarmRepo == nil {
|
||||
return nil, nil, fiber.NewError(fiber.StatusInternalServerError, "hpp per farm repository is not configured")
|
||||
}
|
||||
|
||||
location, err := time.LoadLocation("Asia/Jakarta")
|
||||
if err != nil {
|
||||
return nil, nil, fiber.NewError(fiber.StatusInternalServerError, "failed to load timezone configuration")
|
||||
}
|
||||
startDate, err := time.ParseInLocation("2006-01-02", params.StartDate, location)
|
||||
if err != nil {
|
||||
return nil, nil, fiber.NewError(fiber.StatusBadRequest, "start_date must follow format YYYY-MM-DD")
|
||||
}
|
||||
endDate, err := time.ParseInLocation("2006-01-02", params.EndDate, location)
|
||||
if err != nil {
|
||||
return nil, nil, fiber.NewError(fiber.StatusBadRequest, "end_date must follow format YYYY-MM-DD")
|
||||
}
|
||||
if endDate.Before(startDate) {
|
||||
return nil, nil, fiber.NewError(fiber.StatusBadRequest, "end_date must be greater than or equal to start_date")
|
||||
}
|
||||
rangeDays := int(endDate.Sub(startDate).Hours()/24) + 1
|
||||
if rangeDays > hppPerFarmMaxRangeDays {
|
||||
return nil, nil, fiber.NewError(fiber.StatusBadRequest, "date range must not exceed 366 days")
|
||||
}
|
||||
|
||||
startOfRange := time.Date(startDate.Year(), startDate.Month(), startDate.Day(), 0, 0, 0, 0, location)
|
||||
endBreakdownDate := time.Date(endDate.Year(), endDate.Month(), endDate.Day(), 0, 0, 0, 0, location)
|
||||
endExclusive := endBreakdownDate.Add(24 * time.Hour)
|
||||
startBreakdownDate := startOfRange.AddDate(0, 0, -1)
|
||||
|
||||
limit := params.Limit
|
||||
if limit <= 0 {
|
||||
limit = 10
|
||||
}
|
||||
|
||||
flockRows, err := s.HppPerFarmRepo.GetCandidateFlocks(ctx.Context(), startOfRange, params.AreaIDs, params.LocationIDs)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
if len(flockRows) == 0 {
|
||||
meta := &dto.HppPerFarmMetaDTO{
|
||||
Page: params.Page,
|
||||
Limit: limit,
|
||||
TotalPages: 1,
|
||||
TotalResults: 0,
|
||||
Filters: filters,
|
||||
}
|
||||
data := &dto.HppPerFarmResponseData{
|
||||
StartDate: params.StartDate,
|
||||
EndDate: params.EndDate,
|
||||
Rows: []dto.HppPerFarmRowDTO{},
|
||||
Summary: dto.HppPerFarmSummaryDTO{},
|
||||
}
|
||||
return data, meta, nil
|
||||
}
|
||||
|
||||
flockIDs := make([]uint, 0, len(flockRows))
|
||||
for _, row := range flockRows {
|
||||
flockIDs = append(flockIDs, row.ProjectFlockID)
|
||||
}
|
||||
|
||||
depByFlock, err := s.sumHppPerFarmDepreciationOverRange(ctx.Context(), startOfRange, endBreakdownDate, flockIDs)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
recWeightByFlock, err := s.HppPerFarmRepo.SumRecordingEggWeightByFlock(ctx.Context(), startOfRange, endExclusive, flockIDs)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
doWeightByFlock, err := s.HppPerFarmRepo.SumMarketingDoTelurWeightByFlock(ctx.Context(), startOfRange, endExclusive, flockIDs)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
docByFlock, err := s.HppPerFarmRepo.GetDocCostByFlock(ctx.Context(), flockIDs)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
type hppPerFarmAggregate struct {
|
||||
locationID uint
|
||||
locationName string
|
||||
totalCost float64
|
||||
feed float64
|
||||
ovk float64
|
||||
bop float64
|
||||
depreciation float64
|
||||
other float64
|
||||
recWeight float64
|
||||
doWeight float64
|
||||
docCost float64
|
||||
docQty float64
|
||||
flocks []dto.HppPerFarmFlockDTO
|
||||
}
|
||||
|
||||
farmOrder := make([]uint, 0)
|
||||
farms := make(map[uint]*hppPerFarmAggregate)
|
||||
|
||||
for _, flock := range flockRows {
|
||||
flockID := flock.ProjectFlockID
|
||||
|
||||
codeTotals, err := s.hppPerFarmFlockCostRange(ctx.Context(), flockID, startBreakdownDate, endBreakdownDate)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
feed := codeTotals[hppPerFarmComponentPakan]
|
||||
ovk := codeTotals[hppPerFarmComponentOvk]
|
||||
bop := codeTotals[hppPerFarmComponentBopRegular] + codeTotals[hppPerFarmComponentBopEkspedisi]
|
||||
nonDepreciation := 0.0
|
||||
for _, value := range codeTotals {
|
||||
nonDepreciation += value
|
||||
}
|
||||
other := nonDepreciation - feed - ovk - bop
|
||||
depreciation := depByFlock[flockID]
|
||||
totalCost := nonDepreciation + depreciation
|
||||
|
||||
recWeight := recWeightByFlock[flockID]
|
||||
doWeight := doWeightByFlock[flockID]
|
||||
|
||||
averageDocPrice := int64(0)
|
||||
if doc, ok := docByFlock[flockID]; ok && doc.DocQty > 0 {
|
||||
averageDocPrice = int64(math.Round(doc.DocCost / doc.DocQty))
|
||||
}
|
||||
|
||||
flockDTO := dto.HppPerFarmFlockDTO{
|
||||
ProjectFlockID: int64(flockID),
|
||||
FlockName: flock.FlockName,
|
||||
TotalCostRp: totalCost,
|
||||
FeedCostRp: feed,
|
||||
OvkCostRp: ovk,
|
||||
BopCostRp: bop,
|
||||
DepreciationRp: depreciation,
|
||||
OtherCostRp: other,
|
||||
EggWeightRecordingKg: recWeight,
|
||||
EggWeightDoKg: doWeight,
|
||||
HppPerKgProduction: hppPerFarmSafeDiv(totalCost, recWeight),
|
||||
HppPerKgSales: hppPerFarmSafeDiv(totalCost, doWeight),
|
||||
AverageDocPriceRp: averageDocPrice,
|
||||
}
|
||||
|
||||
farm, ok := farms[flock.LocationID]
|
||||
if !ok {
|
||||
farm = &hppPerFarmAggregate{
|
||||
locationID: flock.LocationID,
|
||||
locationName: flock.LocationName,
|
||||
flocks: make([]dto.HppPerFarmFlockDTO, 0, 1),
|
||||
}
|
||||
farms[flock.LocationID] = farm
|
||||
farmOrder = append(farmOrder, flock.LocationID)
|
||||
}
|
||||
farm.flocks = append(farm.flocks, flockDTO)
|
||||
farm.totalCost += totalCost
|
||||
farm.feed += feed
|
||||
farm.ovk += ovk
|
||||
farm.bop += bop
|
||||
farm.depreciation += depreciation
|
||||
farm.other += other
|
||||
farm.recWeight += recWeight
|
||||
farm.doWeight += doWeight
|
||||
if doc, ok := docByFlock[flockID]; ok {
|
||||
farm.docCost += doc.DocCost
|
||||
farm.docQty += doc.DocQty
|
||||
}
|
||||
}
|
||||
|
||||
rows := make([]dto.HppPerFarmRowDTO, 0, len(farmOrder))
|
||||
summary := dto.HppPerFarmSummaryDTO{}
|
||||
for _, locID := range farmOrder {
|
||||
farm := farms[locID]
|
||||
averageDocPrice := int64(0)
|
||||
if farm.docQty > 0 {
|
||||
averageDocPrice = int64(math.Round(farm.docCost / farm.docQty))
|
||||
}
|
||||
rows = append(rows, dto.HppPerFarmRowDTO{
|
||||
Location: dto.HppPerKandangLocationDTO{ID: int64(farm.locationID), Name: farm.locationName},
|
||||
TotalCostRp: farm.totalCost,
|
||||
FeedCostRp: farm.feed,
|
||||
OvkCostRp: farm.ovk,
|
||||
BopCostRp: farm.bop,
|
||||
DepreciationRp: farm.depreciation,
|
||||
OtherCostRp: farm.other,
|
||||
EggWeightRecordingKg: farm.recWeight,
|
||||
EggWeightDoKg: farm.doWeight,
|
||||
HppPerKgProduction: hppPerFarmSafeDiv(farm.totalCost, farm.recWeight),
|
||||
HppPerKgSales: hppPerFarmSafeDiv(farm.totalCost, farm.doWeight),
|
||||
AverageDocPriceRp: averageDocPrice,
|
||||
Flocks: farm.flocks,
|
||||
})
|
||||
summary.TotalCostRp += farm.totalCost
|
||||
summary.TotalEggWeightRecordingKg += farm.recWeight
|
||||
summary.TotalEggWeightDoKg += farm.doWeight
|
||||
}
|
||||
summary.AverageHppPerKgProduction = hppPerFarmSafeDiv(summary.TotalCostRp, summary.TotalEggWeightRecordingKg)
|
||||
summary.AverageHppPerKgSales = hppPerFarmSafeDiv(summary.TotalCostRp, summary.TotalEggWeightDoKg)
|
||||
|
||||
totalResults := int64(len(rows))
|
||||
totalPages := int64(1)
|
||||
if totalResults > 0 {
|
||||
totalPages = int64(math.Ceil(float64(totalResults) / float64(limit)))
|
||||
}
|
||||
|
||||
offset := (params.Page - 1) * limit
|
||||
if offset < 0 {
|
||||
offset = 0
|
||||
}
|
||||
if offset > len(rows) {
|
||||
offset = len(rows)
|
||||
}
|
||||
end := offset + limit
|
||||
if end > len(rows) {
|
||||
end = len(rows)
|
||||
}
|
||||
|
||||
meta := &dto.HppPerFarmMetaDTO{
|
||||
Page: params.Page,
|
||||
Limit: limit,
|
||||
TotalPages: totalPages,
|
||||
TotalResults: totalResults,
|
||||
Filters: filters,
|
||||
}
|
||||
data := &dto.HppPerFarmResponseData{
|
||||
StartDate: params.StartDate,
|
||||
EndDate: params.EndDate,
|
||||
Rows: rows[offset:end],
|
||||
Summary: summary,
|
||||
}
|
||||
return data, meta, nil
|
||||
}
|
||||
|
||||
// hppPerFarmFlockCostRange returns the range-scoped production cost per component
|
||||
// code for a project flock, EXCLUDING depreciation (which is summed separately
|
||||
// from daily snapshots). Each non-depreciation production component is cumulative
|
||||
// up to a date in the HPP v2 engine, so the range value is the difference between
|
||||
// the cumulative breakdown at end and at the day before the range start.
|
||||
func (s *repportService) hppPerFarmFlockCostRange(ctx context.Context, projectFlockID uint, startBreakdownDate, endBreakdownDate time.Time) (map[string]float64, error) {
|
||||
if s.HppCostRepo == nil {
|
||||
return nil, errors.New("hpp cost repository is not configured")
|
||||
}
|
||||
if s.HppV2Svc == nil {
|
||||
return nil, errors.New("hpp v2 service is not configured")
|
||||
}
|
||||
|
||||
codeTotals := make(map[string]float64)
|
||||
pfkIDs, err := s.HppCostRepo.GetProjectFlockKandangIDs(ctx, projectFlockID)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
for _, pfkID := range pfkIDs {
|
||||
endBreakdown, err := s.HppV2Svc.CalculateHppBreakdown(pfkID, &endBreakdownDate)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
startBreakdown, err := s.HppV2Svc.CalculateHppBreakdown(pfkID, &startBreakdownDate)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
endMap := hppPerFarmProductionScopeTotalsByCode(endBreakdown)
|
||||
startMap := hppPerFarmProductionScopeTotalsByCode(startBreakdown)
|
||||
|
||||
seen := make(map[string]bool, len(endMap)+len(startMap))
|
||||
for code := range endMap {
|
||||
seen[code] = true
|
||||
}
|
||||
for code := range startMap {
|
||||
seen[code] = true
|
||||
}
|
||||
for code := range seen {
|
||||
if code == hppPerFarmComponentDepreciation {
|
||||
continue
|
||||
}
|
||||
codeTotals[code] += endMap[code] - startMap[code]
|
||||
}
|
||||
}
|
||||
|
||||
return codeTotals, nil
|
||||
}
|
||||
|
||||
// sumHppPerFarmDepreciationOverRange sums the daily depreciation_value from
|
||||
// farm_depreciation_snapshots across [startDate, endDate] per project flock,
|
||||
// computing (and persisting) any missing daily snapshot on demand — same lazy
|
||||
// compute path the single-day depreciation report uses.
|
||||
func (s *repportService) sumHppPerFarmDepreciationOverRange(ctx context.Context, startDate, endDate time.Time, projectFlockIDs []uint) (map[uint]float64, error) {
|
||||
acc := make(map[uint]float64, len(projectFlockIDs))
|
||||
if len(projectFlockIDs) == 0 {
|
||||
return acc, nil
|
||||
}
|
||||
if s.ExpenseDepreciationRepo == nil {
|
||||
return nil, fiber.NewError(fiber.StatusInternalServerError, "expense depreciation repository is not configured")
|
||||
}
|
||||
|
||||
for day := startDate; !day.After(endDate); day = day.AddDate(0, 0, 1) {
|
||||
snapshots, err := s.ExpenseDepreciationRepo.GetSnapshotsByPeriodAndFarmIDs(ctx, day, projectFlockIDs)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
byID := make(map[uint]entity.FarmDepreciationSnapshot, len(snapshots))
|
||||
for _, snapshot := range snapshots {
|
||||
byID[snapshot.ProjectFlockId] = snapshot
|
||||
}
|
||||
|
||||
missing := make([]uint, 0)
|
||||
for _, id := range projectFlockIDs {
|
||||
if _, ok := byID[id]; !ok {
|
||||
missing = append(missing, id)
|
||||
}
|
||||
}
|
||||
if len(missing) > 0 {
|
||||
computed, err := s.computeExpenseDepreciationSnapshots(ctx, day, missing, nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if len(computed) > 0 {
|
||||
if err := s.ExpenseDepreciationRepo.UpsertSnapshots(ctx, computed); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
for _, snapshot := range computed {
|
||||
byID[snapshot.ProjectFlockId] = snapshot
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for id, snapshot := range byID {
|
||||
acc[id] += snapshot.DepreciationValue
|
||||
}
|
||||
}
|
||||
|
||||
return acc, nil
|
||||
}
|
||||
|
||||
func hppPerFarmProductionScopeTotalsByCode(breakdown *approvalService.HppV2Breakdown) map[string]float64 {
|
||||
out := make(map[string]float64)
|
||||
if breakdown == nil {
|
||||
return out
|
||||
}
|
||||
for i := range breakdown.Components {
|
||||
comp := &breakdown.Components[i]
|
||||
out[comp.Code] += hppPerFarmProductionScopeTotal(comp)
|
||||
}
|
||||
return out
|
||||
}
|
||||
|
||||
// hppPerFarmProductionScopeTotal mirrors the engine's componentScopeTotal for the
|
||||
// production_cost scope (that helper is unexported in the common service package).
|
||||
func hppPerFarmProductionScopeTotal(component *approvalService.HppV2Component) float64 {
|
||||
if component == nil {
|
||||
return 0
|
||||
}
|
||||
total := 0.0
|
||||
hasPartScopes := false
|
||||
for i := range component.Parts {
|
||||
part := &component.Parts[i]
|
||||
if len(part.Scopes) == 0 {
|
||||
continue
|
||||
}
|
||||
hasPartScopes = true
|
||||
for _, scope := range part.Scopes {
|
||||
if scope == hppPerFarmProductionScope {
|
||||
total += part.Total
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
if hasPartScopes {
|
||||
return total
|
||||
}
|
||||
for _, scope := range component.Scopes {
|
||||
if scope == hppPerFarmProductionScope {
|
||||
return component.Total
|
||||
}
|
||||
}
|
||||
return 0
|
||||
}
|
||||
|
||||
func hppPerFarmSafeDiv(numerator, denominator float64) float64 {
|
||||
if denominator <= 0 {
|
||||
return 0
|
||||
}
|
||||
value := numerator / denominator
|
||||
if math.IsNaN(value) || math.IsInf(value, 0) {
|
||||
return 0
|
||||
}
|
||||
return value
|
||||
}
|
||||
|
||||
func (s *repportService) parseHppPerFarmQuery(ctx *fiber.Ctx) (*validation.HppPerFarmQuery, dto.HppPerFarmFiltersDTO, error) {
|
||||
page := ctx.QueryInt("page", 1)
|
||||
if page < 1 {
|
||||
page = 1
|
||||
}
|
||||
limit := ctx.QueryInt("limit", 10)
|
||||
if limit < 1 {
|
||||
limit = 10
|
||||
}
|
||||
|
||||
rawArea := ctx.Query("area_id", "")
|
||||
rawLocation := ctx.Query("location_id", "")
|
||||
startDate := ctx.Query("start_date", "")
|
||||
endDate := ctx.Query("end_date", "")
|
||||
|
||||
areaIDs, err := parseCommaSeparatedInt64s(rawArea)
|
||||
if err != nil {
|
||||
return nil, dto.HppPerFarmFiltersDTO{}, fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||
}
|
||||
locationIDs, err := parseCommaSeparatedInt64s(rawLocation)
|
||||
if err != nil {
|
||||
return nil, dto.HppPerFarmFiltersDTO{}, fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||
}
|
||||
|
||||
locationScope, err := m.ResolveLocationScope(ctx, s.ExpenseRealizationRepo.DB())
|
||||
if err != nil {
|
||||
return nil, dto.HppPerFarmFiltersDTO{}, err
|
||||
}
|
||||
areaScope, err := m.ResolveAreaScope(ctx, s.ExpenseRealizationRepo.DB())
|
||||
if err != nil {
|
||||
return nil, dto.HppPerFarmFiltersDTO{}, err
|
||||
}
|
||||
if locationScope.Restrict {
|
||||
allowed := toInt64Slice(locationScope.IDs)
|
||||
if len(allowed) == 0 {
|
||||
locationIDs = []int64{-1}
|
||||
} else if len(locationIDs) > 0 {
|
||||
locationIDs = intersectInt64(locationIDs, allowed)
|
||||
} else {
|
||||
locationIDs = allowed
|
||||
}
|
||||
}
|
||||
if areaScope.Restrict {
|
||||
allowed := toInt64Slice(areaScope.IDs)
|
||||
if len(allowed) == 0 {
|
||||
areaIDs = []int64{-1}
|
||||
} else if len(areaIDs) > 0 {
|
||||
areaIDs = intersectInt64(areaIDs, allowed)
|
||||
} else {
|
||||
areaIDs = allowed
|
||||
}
|
||||
}
|
||||
|
||||
params := &validation.HppPerFarmQuery{
|
||||
Page: page,
|
||||
Limit: limit,
|
||||
StartDate: startDate,
|
||||
EndDate: endDate,
|
||||
AreaIDs: areaIDs,
|
||||
LocationIDs: locationIDs,
|
||||
}
|
||||
filters := dto.NewHppPerFarmFiltersDTO(rawArea, rawLocation, startDate, endDate)
|
||||
return params, filters, nil
|
||||
}
|
||||
|
||||
func (s *repportService) parseExpenseDepreciationQuery(ctx *fiber.Ctx) (*validation.ExpenseDepreciationQuery, dto.ExpenseDepreciationFiltersDTO, error) {
|
||||
page := ctx.QueryInt("page", 1)
|
||||
if page < 1 {
|
||||
|
||||
Reference in New Issue
Block a user