mirror of
https://gitlab.com/mbugroup/lti-api.git
synced 2026-06-09 15:07:49 +00:00
Compare commits
27 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| aa3e655a67 | |||
| 98bfdac3c5 | |||
| a98d026ccb | |||
| c3eab60f49 | |||
| e455889dae | |||
| be8b99e7e8 | |||
| 5760bb6de8 | |||
| 1e8651b8f2 | |||
| efe9f0ce3c | |||
| 1ef32407f1 | |||
| 1cd72e5598 | |||
| 7f701511d3 | |||
| 9405c9d64b | |||
| b179ed2bc9 | |||
| 255e6a16d3 | |||
| 93ed89b4ef | |||
| b9201c2a4f | |||
| f443686505 | |||
| 9d8d54bd3c | |||
| 791c5880fd | |||
| 0581bf4a17 | |||
| 1a5dfbb162 | |||
| b28ffdf9c6 | |||
| 90a921ff46 | |||
| 2c9ae1d5ab | |||
| bf93770798 | |||
| 98d031cc18 |
@@ -114,6 +114,12 @@ type HppV2CostRepository interface {
|
|||||||
GetChickinPopulationByPFKForFarm(ctx context.Context, projectFlockID uint) (map[uint]float64, error)
|
GetChickinPopulationByPFKForFarm(ctx context.Context, projectFlockID uint) (map[uint]float64, error)
|
||||||
GetMultiplicationPercentages(ctx context.Context, houseTypes []string, maxDay int, projectFlockID uint) (map[string]map[int]float64, map[string]*time.Time, error)
|
GetMultiplicationPercentages(ctx context.Context, houseTypes []string, maxDay int, projectFlockID uint) (map[string]map[int]float64, map[string]*time.Time, error)
|
||||||
ListUsageCostRowsByProductFlags(ctx context.Context, projectFlockKandangIDs []uint, flagNames []string, date *time.Time) ([]HppV2UsageCostRow, error)
|
ListUsageCostRowsByProductFlags(ctx context.Context, projectFlockKandangIDs []uint, flagNames []string, date *time.Time) ([]HppV2UsageCostRow, error)
|
||||||
|
// ListLayingUsageCostRowsByProductFlags meng-anchor atribusi ke kandang recording
|
||||||
|
// (recordings.project_flock_kandangs_id), bukan ke recording_stocks.project_flock_kandang_id.
|
||||||
|
// Diperlukan karena pakan/OVK kandang LAYING yang dikonsumsi dari gudang tipe LOKASI
|
||||||
|
// punya recording_stocks.project_flock_kandang_id = NULL — kasus ini harus tetap diatribusikan
|
||||||
|
// ke kandang laying sebagai production_cost (bukan jatuh ke RECORDING_STOCK_ROUTE / pullet_cost).
|
||||||
|
ListLayingUsageCostRowsByProductFlags(ctx context.Context, layingProjectFlockKandangID uint, flagNames []string, date *time.Time) ([]HppV2UsageCostRow, error)
|
||||||
ListAdjustmentCostRowsByProductFlags(ctx context.Context, projectFlockKandangIDs []uint, flagNames []string, date *time.Time) ([]HppV2AdjustmentCostRow, error)
|
ListAdjustmentCostRowsByProductFlags(ctx context.Context, projectFlockKandangIDs []uint, flagNames []string, date *time.Time) ([]HppV2AdjustmentCostRow, error)
|
||||||
ListExpenseRealizationRowsByProjectFlockKandangIDs(ctx context.Context, projectFlockKandangIDs []uint, date *time.Time, ekspedisi bool) ([]HppV2ExpenseCostRow, error)
|
ListExpenseRealizationRowsByProjectFlockKandangIDs(ctx context.Context, projectFlockKandangIDs []uint, date *time.Time, ekspedisi bool) ([]HppV2ExpenseCostRow, error)
|
||||||
ListExpenseRealizationRowsByProjectFlockID(ctx context.Context, projectFlockID uint, date *time.Time, ekspedisi bool) ([]HppV2ExpenseCostRow, error)
|
ListExpenseRealizationRowsByProjectFlockID(ctx context.Context, projectFlockID uint, date *time.Time, ekspedisi bool) ([]HppV2ExpenseCostRow, error)
|
||||||
@@ -367,18 +373,19 @@ func (r *HppV2RepositoryImpl) GetRecordingStockRoutingAdjustmentCostByProjectFlo
|
|||||||
Joins("JOIN purchase_items AS pi ON pi.id = sa.stockable_id").
|
Joins("JOIN purchase_items AS pi ON pi.id = sa.stockable_id").
|
||||||
Where("pfk_rec.project_flock_id = ?", projectFlockID).
|
Where("pfk_rec.project_flock_id = ?", projectFlockID).
|
||||||
Where("DATE(r.record_datetime) <= DATE(?)", periodDate).
|
Where("DATE(r.record_datetime) <= DATE(?)", periodDate).
|
||||||
|
// Hanya routing cross-kandang ASLI: stok yang dicatat di recording kandang X tetapi
|
||||||
|
// recording_stocks.project_flock_kandang_id menunjuk kandang lain (Y) saat ada transfer.
|
||||||
|
// Cabang lama "NOT(transferExists) AND rs.pfk IS NULL" DIHAPUS — kasus pakan/OVK laying
|
||||||
|
// dari gudang LOKASI (pfk NULL) kini diatribusikan sebagai production_cost via
|
||||||
|
// ListLayingUsageCostRowsByProductFlags, sehingga kedua jalur jadi disjoint (tanpa dobel).
|
||||||
Where(
|
Where(
|
||||||
fmt.Sprintf(
|
fmt.Sprintf(
|
||||||
"((%s) AND rs.project_flock_kandang_id IS NOT NULL AND rs.project_flock_kandang_id <> r.project_flock_kandangs_id) OR (NOT (%s) AND rs.project_flock_kandang_id IS NULL)",
|
"(%s) AND rs.project_flock_kandang_id IS NOT NULL AND rs.project_flock_kandang_id <> r.project_flock_kandangs_id",
|
||||||
transferExistsCondition,
|
|
||||||
transferExistsCondition,
|
transferExistsCondition,
|
||||||
),
|
),
|
||||||
periodDate,
|
periodDate,
|
||||||
string(utils.ApprovalWorkflowTransferToLaying),
|
string(utils.ApprovalWorkflowTransferToLaying),
|
||||||
entity.ApprovalActionApproved,
|
entity.ApprovalActionApproved,
|
||||||
periodDate,
|
|
||||||
string(utils.ApprovalWorkflowTransferToLaying),
|
|
||||||
entity.ApprovalActionApproved,
|
|
||||||
).
|
).
|
||||||
Where("EXISTS (SELECT 1 FROM flags f WHERE f.flagable_id = pw.product_id AND f.flagable_type = ? AND f.name IN ?)", entity.FlagableTypeProduct, flags).
|
Where("EXISTS (SELECT 1 FROM flags f WHERE f.flagable_id = pw.product_id AND f.flagable_type = ? AND f.name IN ?)", entity.FlagableTypeProduct, flags).
|
||||||
Scan(&total).Error
|
Scan(&total).Error
|
||||||
@@ -585,6 +592,91 @@ func (r *HppV2RepositoryImpl) ListUsageCostRowsByProductFlags(
|
|||||||
return rows, nil
|
return rows, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ListLayingUsageCostRowsByProductFlags identik dengan ListUsageCostRowsByProductFlags,
|
||||||
|
// tetapi atribusi baris ditentukan oleh kandang RECORDING (r.project_flock_kandangs_id),
|
||||||
|
// dengan recording_stocks.project_flock_kandang_id boleh NULL (gudang LOKASI) atau sama
|
||||||
|
// dengan kandang laying. Baris yang routed ke kandang lain (rs.pfk <> kandang recording)
|
||||||
|
// SENGAJA TIDAK diikutkan di sini — itu ranah RECORDING_STOCK_ROUTE.
|
||||||
|
func (r *HppV2RepositoryImpl) ListLayingUsageCostRowsByProductFlags(
|
||||||
|
ctx context.Context,
|
||||||
|
layingProjectFlockKandangID uint,
|
||||||
|
flagNames []string,
|
||||||
|
date *time.Time,
|
||||||
|
) ([]HppV2UsageCostRow, error) {
|
||||||
|
if layingProjectFlockKandangID == 0 || len(flagNames) == 0 {
|
||||||
|
return []HppV2UsageCostRow{}, nil
|
||||||
|
}
|
||||||
|
if date == nil {
|
||||||
|
now := time.Now()
|
||||||
|
date = &now
|
||||||
|
}
|
||||||
|
|
||||||
|
stockablePurchase := fifo.StockableKeyPurchaseItems.String()
|
||||||
|
stockableAdjustment := fifo.StockableKeyAdjustmentIn.String()
|
||||||
|
usableRecordingStock := fifo.UsableKeyRecordingStock.String()
|
||||||
|
|
||||||
|
rows := make([]HppV2UsageCostRow, 0)
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Table("recordings AS r").
|
||||||
|
Select(`
|
||||||
|
sa.stockable_type AS stockable_type,
|
||||||
|
sa.stockable_id AS stockable_id,
|
||||||
|
COALESCE(pi.product_id, ast_pw.product_id, 0) AS source_product_id,
|
||||||
|
COALESCE(pi_prod.name, ast_prod.name, '') AS source_product_name,
|
||||||
|
COALESCE(SUM(sa.qty), 0) AS qty,
|
||||||
|
COALESCE(MAX(CASE
|
||||||
|
WHEN sa.stockable_type = ? THEN COALESCE(pi.price, 0)
|
||||||
|
WHEN sa.stockable_type = ? THEN COALESCE(ast.price, 0)
|
||||||
|
ELSE 0
|
||||||
|
END), 0) AS unit_price,
|
||||||
|
COALESCE(SUM(sa.qty * CASE
|
||||||
|
WHEN sa.stockable_type = ? THEN COALESCE(pi.price, 0)
|
||||||
|
WHEN sa.stockable_type = ? THEN COALESCE(ast.price, 0)
|
||||||
|
ELSE 0
|
||||||
|
END), 0) AS total_cost,
|
||||||
|
MIN(r.record_datetime) AS first_used_at,
|
||||||
|
MAX(r.record_datetime) AS last_used_at
|
||||||
|
`,
|
||||||
|
stockablePurchase,
|
||||||
|
stockableAdjustment,
|
||||||
|
stockablePurchase,
|
||||||
|
stockableAdjustment,
|
||||||
|
).
|
||||||
|
Joins("JOIN recording_stocks AS rs ON rs.recording_id = r.id").
|
||||||
|
Joins("JOIN product_warehouses AS pw ON pw.id = rs.product_warehouse_id").
|
||||||
|
Joins(
|
||||||
|
"JOIN stock_allocations AS sa ON sa.usable_type = ? AND sa.usable_id = rs.id AND (sa.stockable_type = ? OR sa.stockable_type = ?) AND sa.status = ? AND sa.allocation_purpose = ?",
|
||||||
|
usableRecordingStock,
|
||||||
|
stockablePurchase,
|
||||||
|
stockableAdjustment,
|
||||||
|
entity.StockAllocationStatusActive,
|
||||||
|
entity.StockAllocationPurposeConsume,
|
||||||
|
).
|
||||||
|
Joins("LEFT JOIN purchase_items AS pi ON pi.id = sa.stockable_id AND sa.stockable_type = ?", stockablePurchase).
|
||||||
|
Joins("LEFT JOIN products AS pi_prod ON pi_prod.id = pi.product_id").
|
||||||
|
Joins("LEFT JOIN adjustment_stocks AS ast ON ast.id = sa.stockable_id AND sa.stockable_type = ?", stockableAdjustment).
|
||||||
|
Joins("LEFT JOIN product_warehouses AS ast_pw ON ast_pw.id = ast.product_warehouse_id").
|
||||||
|
Joins("LEFT JOIN products AS ast_prod ON ast_prod.id = ast_pw.product_id").
|
||||||
|
Where("r.project_flock_kandangs_id = ?", layingProjectFlockKandangID).
|
||||||
|
Where("(rs.project_flock_kandang_id IS NULL OR rs.project_flock_kandang_id = ?)", layingProjectFlockKandangID).
|
||||||
|
Where("r.deleted_at IS NULL").
|
||||||
|
Where("r.record_datetime <= ?", *date).
|
||||||
|
Where("EXISTS (SELECT 1 FROM flags f WHERE f.flagable_id = pw.product_id AND f.flagable_type = ? AND f.name IN ?)", entity.FlagableTypeProduct, flagNames).
|
||||||
|
Group(`
|
||||||
|
sa.stockable_type,
|
||||||
|
sa.stockable_id,
|
||||||
|
COALESCE(pi.product_id, ast_pw.product_id, 0),
|
||||||
|
COALESCE(pi_prod.name, ast_prod.name, '')
|
||||||
|
`).
|
||||||
|
Order("MIN(r.record_datetime) ASC, sa.stockable_type ASC, sa.stockable_id ASC").
|
||||||
|
Scan(&rows).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return rows, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (r *HppV2RepositoryImpl) ListAdjustmentCostRowsByProductFlags(
|
func (r *HppV2RepositoryImpl) ListAdjustmentCostRowsByProductFlags(
|
||||||
ctx context.Context,
|
ctx context.Context,
|
||||||
projectFlockKandangIDs []uint,
|
projectFlockKandangIDs []uint,
|
||||||
|
|||||||
@@ -192,19 +192,27 @@ func TestHppV2RepositoryGetRecordingStockRoutingAdjustmentCostByProjectFlockID(t
|
|||||||
|
|
||||||
repo := &HppV2RepositoryImpl{db: db}
|
repo := &HppV2RepositoryImpl{db: db}
|
||||||
|
|
||||||
|
// Route sekarang HANYA menangkap routing cross-kandang asli
|
||||||
|
// (transferExists AND rs.pfk IS NOT NULL AND rs.pfk <> r.project_flock_kandangs_id).
|
||||||
|
// Baris pfk NULL (gudang LOKASI) tidak lagi masuk route — kini jadi production_cost
|
||||||
|
// laying-usage via ListLayingUsageCostRowsByProductFlags.
|
||||||
|
// Pada 2026-04-30 hanya rs 102 yang lolos: recording pfk 101 (transfer 1001 approved &
|
||||||
|
// executed, effective 04-05 <= 04-30), rs.pfk 201 <> 101 → 1 × 110 = 110.
|
||||||
periodDate := mustJakartaTime(t, "2026-04-30 00:00:00")
|
periodDate := mustJakartaTime(t, "2026-04-30 00:00:00")
|
||||||
total, err := repo.GetRecordingStockRoutingAdjustmentCostByProjectFlockID(context.Background(), 1, periodDate)
|
total, err := repo.GetRecordingStockRoutingAdjustmentCostByProjectFlockID(context.Background(), 1, periodDate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("expected no error, got %v", err)
|
t.Fatalf("expected no error, got %v", err)
|
||||||
}
|
}
|
||||||
assertFloatEquals(t, total, 750)
|
assertFloatEquals(t, total, 110)
|
||||||
|
|
||||||
|
// Pada 2026-04-10 hanya recording pfk 101 & 102 yang masuk rentang tanggal; tetap hanya
|
||||||
|
// rs 102 (cross-kandang) yang lolos → 110.
|
||||||
earlyPeriod := mustJakartaTime(t, "2026-04-10 23:59:59")
|
earlyPeriod := mustJakartaTime(t, "2026-04-10 23:59:59")
|
||||||
earlyTotal, err := repo.GetRecordingStockRoutingAdjustmentCostByProjectFlockID(context.Background(), 1, earlyPeriod)
|
earlyTotal, err := repo.GetRecordingStockRoutingAdjustmentCostByProjectFlockID(context.Background(), 1, earlyPeriod)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("expected no error, got %v", err)
|
t.Fatalf("expected no error, got %v", err)
|
||||||
}
|
}
|
||||||
assertFloatEquals(t, earlyTotal, 240)
|
assertFloatEquals(t, earlyTotal, 110)
|
||||||
}
|
}
|
||||||
|
|
||||||
func setupHppV2RepositoryTestDB(t *testing.T) *gorm.DB {
|
func setupHppV2RepositoryTestDB(t *testing.T) *gorm.DB {
|
||||||
|
|||||||
@@ -55,6 +55,10 @@ type HppV2Service interface {
|
|||||||
GetDirectPulletPurchaseBreakdown(projectFlockKandangId uint, endDate *time.Time) (*HppV2Component, error)
|
GetDirectPulletPurchaseBreakdown(projectFlockKandangId uint, endDate *time.Time) (*HppV2Component, error)
|
||||||
GetBopRegularBreakdown(projectFlockKandangId uint, endDate *time.Time) (*HppV2Component, error)
|
GetBopRegularBreakdown(projectFlockKandangId uint, endDate *time.Time) (*HppV2Component, error)
|
||||||
GetBopEkspedisiBreakdown(projectFlockKandangId uint, endDate *time.Time) (*HppV2Component, error)
|
GetBopEkspedisiBreakdown(projectFlockKandangId uint, endDate *time.Time) (*HppV2Component, error)
|
||||||
|
// GetBopRegularProductionScopeRange / GetBopEkspedisiProductionScopeRange mengembalikan BOP
|
||||||
|
// production_cost untuk rentang [startDate, endDate] secara range-correct (tidak pernah negatif).
|
||||||
|
GetBopRegularProductionScopeRange(projectFlockKandangId uint, startDate, endDate *time.Time) (float64, error)
|
||||||
|
GetBopEkspedisiProductionScopeRange(projectFlockKandangId uint, startDate, endDate *time.Time) (float64, error)
|
||||||
GetHppEstimationDanRealisasi(totalProductionCost float64, projectFlockKandangId uint, startDate *time.Time, endDate *time.Time) (*HppCostResponse, error)
|
GetHppEstimationDanRealisasi(totalProductionCost float64, projectFlockKandangId uint, startDate *time.Time, endDate *time.Time) (*HppCostResponse, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -453,7 +457,7 @@ func (s *hppV2Service) getStockUsageComponent(projectFlockKandangId uint, endDat
|
|||||||
total += growingCutoverPart.Total
|
total += growingCutoverPart.Total
|
||||||
}
|
}
|
||||||
|
|
||||||
layingNormalPart, err := s.buildLayingUsagePart(projectFlockKandangId, endDate, config, false)
|
layingNormalPart, err := s.buildLayingUsagePart(projectFlockKandangId, contextRow, endDate, config, false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -462,7 +466,7 @@ func (s *hppV2Service) getStockUsageComponent(projectFlockKandangId uint, endDat
|
|||||||
total += layingNormalPart.Total
|
total += layingNormalPart.Total
|
||||||
}
|
}
|
||||||
|
|
||||||
layingCutoverPart, err := s.buildLayingUsagePart(projectFlockKandangId, endDate, config, true)
|
layingCutoverPart, err := s.buildLayingUsagePart(projectFlockKandangId, contextRow, endDate, config, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -737,6 +741,7 @@ func (s *hppV2Service) buildGrowingUsagePart(
|
|||||||
|
|
||||||
func (s *hppV2Service) buildLayingUsagePart(
|
func (s *hppV2Service) buildLayingUsagePart(
|
||||||
projectFlockKandangId uint,
|
projectFlockKandangId uint,
|
||||||
|
contextRow *commonRepo.HppV2ProjectFlockKandangContext,
|
||||||
endDate *time.Time,
|
endDate *time.Time,
|
||||||
config hppV2StockComponentConfig,
|
config hppV2StockComponentConfig,
|
||||||
cutover bool,
|
cutover bool,
|
||||||
@@ -778,7 +783,16 @@ func (s *hppV2Service) buildLayingUsagePart(
|
|||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
rows, err := s.hppRepo.ListUsageCostRowsByProductFlags(context.Background(), []uint{projectFlockKandangId}, config.NormalFlags, endDate)
|
// Untuk kandang LAYING, atribusi pakan/OVK berbasis kandang recording (termasuk konsumsi
|
||||||
|
// dari gudang LOKASI yang punya recording_stocks.project_flock_kandang_id = NULL). Untuk
|
||||||
|
// kandang non-laying, pertahankan semantik lama (strict rs.project_flock_kandang_id IN [pfk]).
|
||||||
|
var rows []commonRepo.HppV2UsageCostRow
|
||||||
|
var err error
|
||||||
|
if contextRow != nil && contextRow.ProjectFlockCategory == string(utils.ProjectFlockCategoryLaying) {
|
||||||
|
rows, err = s.hppRepo.ListLayingUsageCostRowsByProductFlags(context.Background(), projectFlockKandangId, config.NormalFlags, endDate)
|
||||||
|
} else {
|
||||||
|
rows, err = s.hppRepo.ListUsageCostRowsByProductFlags(context.Background(), []uint{projectFlockKandangId}, config.NormalFlags, endDate)
|
||||||
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -931,17 +945,48 @@ func (s *hppV2Service) buildLayingExpenseFarmPart(
|
|||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
farmPFKIDs, err := s.hppRepo.GetProjectFlockKandangIDs(context.Background(), contextRow.ProjectFlockID)
|
ratio, proration, err := s.layingFarmExpenseRatio(projectFlockKandangId, contextRow, endDate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
if ratio <= 0 {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return buildExpensePartFromRows(
|
||||||
|
rows,
|
||||||
|
hppV2PartLayingFarm,
|
||||||
|
"Laying Farm",
|
||||||
|
[]string{hppV2ScopeProductionCost},
|
||||||
|
proration,
|
||||||
|
ratio,
|
||||||
|
), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// layingFarmExpenseRatio menghitung porsi (share) kandang laying terhadap seluruh farm pada
|
||||||
|
// endDate berdasarkan bobot telur KUMULATIF (fallback ke jumlah butir bila bobot 0). Return
|
||||||
|
// ratio 0 bila tak terhitung. Diekstrak agar dipakai bersama oleh buildLayingExpenseFarmPart
|
||||||
|
// dan GetExpenseProductionScopeRange (perhitungan BOP range-correct).
|
||||||
|
func (s *hppV2Service) layingFarmExpenseRatio(
|
||||||
|
projectFlockKandangId uint,
|
||||||
|
contextRow *commonRepo.HppV2ProjectFlockKandangContext,
|
||||||
|
endDate *time.Time,
|
||||||
|
) (float64, *HppV2Proration, error) {
|
||||||
|
if contextRow == nil {
|
||||||
|
return 0, nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
farmPFKIDs, err := s.hppRepo.GetProjectFlockKandangIDs(context.Background(), contextRow.ProjectFlockID)
|
||||||
|
if err != nil {
|
||||||
|
return 0, nil, err
|
||||||
|
}
|
||||||
targetPieces, targetWeight, err := s.hppRepo.GetEggProduksiPiecesAndWeightKgByProjectFlockKandangIds(context.Background(), []uint{projectFlockKandangId}, endDate)
|
targetPieces, targetWeight, err := s.hppRepo.GetEggProduksiPiecesAndWeightKgByProjectFlockKandangIds(context.Background(), []uint{projectFlockKandangId}, endDate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return 0, nil, err
|
||||||
}
|
}
|
||||||
farmPieces, farmWeight, err := s.hppRepo.GetEggProduksiPiecesAndWeightKgByProjectFlockKandangIds(context.Background(), farmPFKIDs, endDate)
|
farmPieces, farmWeight, err := s.hppRepo.GetEggProduksiPiecesAndWeightKgByProjectFlockKandangIds(context.Background(), farmPFKIDs, endDate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return 0, nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
basis := hppV2ProrationEggWeight
|
basis := hppV2ProrationEggWeight
|
||||||
@@ -953,27 +998,120 @@ func (s *hppV2Service) buildLayingExpenseFarmPart(
|
|||||||
denominator = farmPieces
|
denominator = farmPieces
|
||||||
}
|
}
|
||||||
if denominator <= 0 {
|
if denominator <= 0 {
|
||||||
return nil, nil
|
return 0, nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
ratio := numerator / denominator
|
ratio := numerator / denominator
|
||||||
if ratio <= 0 {
|
if ratio <= 0 {
|
||||||
return nil, nil
|
return 0, nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return buildExpensePartFromRows(
|
return ratio, &HppV2Proration{
|
||||||
rows,
|
Basis: basis,
|
||||||
hppV2PartLayingFarm,
|
Numerator: numerator,
|
||||||
"Laying Farm",
|
Denominator: denominator,
|
||||||
[]string{hppV2ScopeProductionCost},
|
Ratio: ratio,
|
||||||
&HppV2Proration{
|
}, nil
|
||||||
Basis: basis,
|
}
|
||||||
Numerator: numerator,
|
|
||||||
Denominator: denominator,
|
// GetExpenseProductionScopeRange menghitung BOP production_cost satu komponen expense untuk rentang
|
||||||
Ratio: ratio,
|
// [startDate, endDate] secara range-correct (tidak pernah negatif untuk expense non-negatif).
|
||||||
},
|
// - laying-direct (ratio 1, monoton): selisih kumulatif end - start.
|
||||||
ratio,
|
// - laying-farm (prorated): (expenseCum(end) - expenseCum(start)) × ratio(end).
|
||||||
), nil
|
//
|
||||||
|
// Ini mengganti pola lama di report yang men-differensiasi dua angka yang sudah diprorata dengan
|
||||||
|
// ratio berbeda (ratio(end) vs ratio(start)) — sumber bug BOP negatif saat share antar kandang bergeser.
|
||||||
|
func (s *hppV2Service) GetExpenseProductionScopeRange(projectFlockKandangId uint, startDate, endDate *time.Time, config hppV2ExpenseComponentConfig) (float64, error) {
|
||||||
|
if s.hppRepo == nil {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
contextRow, err := s.hppRepo.GetProjectFlockKandangContext(context.Background(), projectFlockKandangId)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Samakan semantik tanggal dengan CalculateHppBreakdown: kumulatif dihitung sampai AKHIR hari
|
||||||
|
// (endOfDay). Penting karena ratio egg-weight memakai r.record_datetime (granular jam).
|
||||||
|
_, endOfEndDay, err := hppV2DayWindow(endDate)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
_, endOfStartDay, err := hppV2DayWindow(startDate)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// laying-direct: delta kumulatif (monoton, >= 0).
|
||||||
|
directEnd, err := s.buildLayingExpenseDirectPart(projectFlockKandangId, &endOfEndDay, config)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
directStart, err := s.buildLayingExpenseDirectPart(projectFlockKandangId, &endOfStartDay, config)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
directDelta := hppV2PartTotal(directEnd) - hppV2PartTotal(directStart)
|
||||||
|
if directDelta < 0 {
|
||||||
|
directDelta = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// laying-farm: delta expense kumulatif × ratio(end).
|
||||||
|
farmRowsEnd, err := s.hppRepo.ListExpenseRealizationRowsByProjectFlockID(context.Background(), contextRow.ProjectFlockID, &endOfEndDay, config.Ekspedisi)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
farmRowsStart, err := s.hppRepo.ListExpenseRealizationRowsByProjectFlockID(context.Background(), contextRow.ProjectFlockID, &endOfStartDay, config.Ekspedisi)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
farmExpenseDelta := hppV2SumExpenseRows(farmRowsEnd) - hppV2SumExpenseRows(farmRowsStart)
|
||||||
|
if farmExpenseDelta < 0 {
|
||||||
|
farmExpenseDelta = 0
|
||||||
|
}
|
||||||
|
farmDelta := 0.0
|
||||||
|
if farmExpenseDelta > 0 {
|
||||||
|
ratio, _, err := s.layingFarmExpenseRatio(projectFlockKandangId, contextRow, &endOfEndDay)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
farmDelta = farmExpenseDelta * ratio
|
||||||
|
}
|
||||||
|
|
||||||
|
return directDelta + farmDelta, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetBopRegularProductionScopeRange / GetBopEkspedisiProductionScopeRange — wrapper range-correct
|
||||||
|
// untuk dua komponen BOP, memakai config yang sama dengan GetBopRegularBreakdown/GetBopEkspedisiBreakdown.
|
||||||
|
func (s *hppV2Service) GetBopRegularProductionScopeRange(projectFlockKandangId uint, startDate, endDate *time.Time) (float64, error) {
|
||||||
|
return s.GetExpenseProductionScopeRange(projectFlockKandangId, startDate, endDate, hppV2ExpenseComponentConfig{
|
||||||
|
Code: hppV2ComponentBopRegular,
|
||||||
|
Title: "BOP Regular",
|
||||||
|
Ekspedisi: false,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *hppV2Service) GetBopEkspedisiProductionScopeRange(projectFlockKandangId uint, startDate, endDate *time.Time) (float64, error) {
|
||||||
|
return s.GetExpenseProductionScopeRange(projectFlockKandangId, startDate, endDate, hppV2ExpenseComponentConfig{
|
||||||
|
Code: hppV2ComponentBopEksp,
|
||||||
|
Title: "BOP Ekspedisi",
|
||||||
|
Ekspedisi: true,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func hppV2PartTotal(part *HppV2ComponentPart) float64 {
|
||||||
|
if part == nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return part.Total
|
||||||
|
}
|
||||||
|
|
||||||
|
func hppV2SumExpenseRows(rows []commonRepo.HppV2ExpenseCostRow) float64 {
|
||||||
|
total := 0.0
|
||||||
|
for _, row := range rows {
|
||||||
|
total += row.TotalCost
|
||||||
|
}
|
||||||
|
return total
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *hppV2Service) getManualPulletCostComponent(
|
func (s *hppV2Service) getManualPulletCostComponent(
|
||||||
|
|||||||
@@ -25,9 +25,13 @@ type hppV2RepoStub struct {
|
|||||||
chickinRowsByKey map[string][]commonRepo.HppV2ChickinCostRow
|
chickinRowsByKey map[string][]commonRepo.HppV2ChickinCostRow
|
||||||
expenseRowsByPFKKey map[string][]commonRepo.HppV2ExpenseCostRow
|
expenseRowsByPFKKey map[string][]commonRepo.HppV2ExpenseCostRow
|
||||||
expenseRowsByFarmKey map[string][]commonRepo.HppV2ExpenseCostRow
|
expenseRowsByFarmKey map[string][]commonRepo.HppV2ExpenseCostRow
|
||||||
routeCostByProject map[uint]float64
|
// expenseRowsByFarmDateKey (opsional) membuat ListExpenseRealizationRowsByProjectFlockID
|
||||||
totalPopulationByKey map[string]float64
|
// date-aware untuk menguji perhitungan range BOP. Bila non-nil, dipakai menggantikan
|
||||||
transferSummaryByPFK map[uint]struct {
|
// expenseRowsByFarmKey; key = "<flock>|<ekspedisi>|<YYYY-MM-DD>".
|
||||||
|
expenseRowsByFarmDateKey map[string][]commonRepo.HppV2ExpenseCostRow
|
||||||
|
routeCostByProject map[uint]float64
|
||||||
|
totalPopulationByKey map[string]float64
|
||||||
|
transferSummaryByPFK map[uint]struct {
|
||||||
projectFlockID uint
|
projectFlockID uint
|
||||||
totalQty float64
|
totalQty float64
|
||||||
}
|
}
|
||||||
@@ -118,6 +122,10 @@ func (s *hppV2RepoStub) ListUsageCostRowsByProductFlags(_ context.Context, proje
|
|||||||
return append([]commonRepo.HppV2UsageCostRow{}, s.usageRowsByKey[stubKey(projectFlockKandangIDs, flagNames)]...), nil
|
return append([]commonRepo.HppV2UsageCostRow{}, s.usageRowsByKey[stubKey(projectFlockKandangIDs, flagNames)]...), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *hppV2RepoStub) ListLayingUsageCostRowsByProductFlags(_ context.Context, layingProjectFlockKandangID uint, flagNames []string, _ *time.Time) ([]commonRepo.HppV2UsageCostRow, error) {
|
||||||
|
return append([]commonRepo.HppV2UsageCostRow{}, s.usageRowsByKey[stubKey([]uint{layingProjectFlockKandangID}, flagNames)]...), nil
|
||||||
|
}
|
||||||
|
|
||||||
func (s *hppV2RepoStub) ListAdjustmentCostRowsByProductFlags(_ context.Context, projectFlockKandangIDs []uint, flagNames []string, _ *time.Time) ([]commonRepo.HppV2AdjustmentCostRow, error) {
|
func (s *hppV2RepoStub) ListAdjustmentCostRowsByProductFlags(_ context.Context, projectFlockKandangIDs []uint, flagNames []string, _ *time.Time) ([]commonRepo.HppV2AdjustmentCostRow, error) {
|
||||||
return append([]commonRepo.HppV2AdjustmentCostRow{}, s.adjustRowsByKey[stubKey(projectFlockKandangIDs, flagNames)]...), nil
|
return append([]commonRepo.HppV2AdjustmentCostRow{}, s.adjustRowsByKey[stubKey(projectFlockKandangIDs, flagNames)]...), nil
|
||||||
}
|
}
|
||||||
@@ -126,7 +134,10 @@ func (s *hppV2RepoStub) ListExpenseRealizationRowsByProjectFlockKandangIDs(_ con
|
|||||||
return append([]commonRepo.HppV2ExpenseCostRow{}, s.expenseRowsByPFKKey[expenseStubKey(projectFlockKandangIDs, ekspedisi)]...), nil
|
return append([]commonRepo.HppV2ExpenseCostRow{}, s.expenseRowsByPFKKey[expenseStubKey(projectFlockKandangIDs, ekspedisi)]...), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *hppV2RepoStub) ListExpenseRealizationRowsByProjectFlockID(_ context.Context, projectFlockID uint, _ *time.Time, ekspedisi bool) ([]commonRepo.HppV2ExpenseCostRow, error) {
|
func (s *hppV2RepoStub) ListExpenseRealizationRowsByProjectFlockID(_ context.Context, projectFlockID uint, date *time.Time, ekspedisi bool) ([]commonRepo.HppV2ExpenseCostRow, error) {
|
||||||
|
if s.expenseRowsByFarmDateKey != nil && date != nil {
|
||||||
|
return append([]commonRepo.HppV2ExpenseCostRow{}, s.expenseRowsByFarmDateKey[expenseFarmDateKey(projectFlockID, ekspedisi, *date)]...), nil
|
||||||
|
}
|
||||||
return append([]commonRepo.HppV2ExpenseCostRow{}, s.expenseRowsByFarmKey[expenseFarmKey(projectFlockID, ekspedisi)]...), nil
|
return append([]commonRepo.HppV2ExpenseCostRow{}, s.expenseRowsByFarmKey[expenseFarmKey(projectFlockID, ekspedisi)]...), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -904,6 +915,108 @@ func expenseFarmKey(projectFlockID uint, ekspedisi bool) string {
|
|||||||
return fmt.Sprintf("farm=%d|ekspedisi=%t", projectFlockID, ekspedisi)
|
return fmt.Sprintf("farm=%d|ekspedisi=%t", projectFlockID, ekspedisi)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func expenseFarmDateKey(projectFlockID uint, ekspedisi bool, date time.Time) string {
|
||||||
|
return fmt.Sprintf("%d|%t|%s", projectFlockID, ekspedisi, date.Format("2006-01-02"))
|
||||||
|
}
|
||||||
|
|
||||||
func chickinStubKey(ids []uint, flags []string, excludeTransferToLaying bool) string {
|
func chickinStubKey(ids []uint, flags []string, excludeTransferToLaying bool) string {
|
||||||
return stubKey(ids, append(append([]string{}, flags...), fmt.Sprintf("exclude_transfer_to_laying=%t", excludeTransferToLaying)))
|
return stubKey(ids, append(append([]string{}, flags...), fmt.Sprintf("exclude_transfer_to_laying=%t", excludeTransferToLaying)))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TestHppV2PakanBreakdown_LayingAttributesLokasiFeedAsProductionCost membuktikan Fix 1:
|
||||||
|
// untuk kandang LAYING, pemakaian pakan (termasuk dari gudang LOKASI dengan pfk NULL) diatribusikan
|
||||||
|
// sebagai production_cost via ListLayingUsageCostRowsByProductFlags — BUKAN pullet_cost.
|
||||||
|
// Stub memetakan ListLayingUsageCostRowsByProductFlags(50,...) ke usageRowsByKey[[50]+PAKAN].
|
||||||
|
func TestHppV2PakanBreakdown_LayingAttributesLokasiFeedAsProductionCost(t *testing.T) {
|
||||||
|
repo := &hppV2RepoStub{
|
||||||
|
contextByPFK: map[uint]*commonRepo.HppV2ProjectFlockKandangContext{
|
||||||
|
50: {
|
||||||
|
ProjectFlockKandangID: 50,
|
||||||
|
ProjectFlockID: 20,
|
||||||
|
ProjectFlockCategory: string(utils.ProjectFlockCategoryLaying),
|
||||||
|
KandangID: 1,
|
||||||
|
LocationID: 14,
|
||||||
|
HouseType: "close_house",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
usageRowsByKey: map[string][]commonRepo.HppV2UsageCostRow{
|
||||||
|
stubKey([]uint{50}, []string{"PAKAN"}): {
|
||||||
|
{StockableType: "purchase_items", StockableID: 9001, SourceProductID: 9, SourceProductName: "Pakan Laying", Qty: 310, UnitPrice: 1, TotalCost: 310},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
// Tanpa transferSummaryByPFK[50] -> growing part nil; tanpa adjustRowsByKey -> laying cutover nil.
|
||||||
|
}
|
||||||
|
|
||||||
|
svc := NewHppV2Service(repo)
|
||||||
|
component, err := svc.GetPakanBreakdown(50, mustDate(t, "2026-05-31"))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("expected no error, got %v", err)
|
||||||
|
}
|
||||||
|
if component == nil {
|
||||||
|
t.Fatal("expected PAKAN component")
|
||||||
|
}
|
||||||
|
if component.Total != 310 {
|
||||||
|
t.Fatalf("expected component total 310, got %v", component.Total)
|
||||||
|
}
|
||||||
|
if len(component.Parts) != 1 || component.Parts[0].Code != hppV2PartLayingNormal {
|
||||||
|
t.Fatalf("expected single laying_normal part, got %+v", component.Parts)
|
||||||
|
}
|
||||||
|
if got := componentScopeTotal(component, hppV2ScopeProductionCost); got != 310 {
|
||||||
|
t.Fatalf("expected production_cost 310, got %v", got)
|
||||||
|
}
|
||||||
|
if got := componentScopeTotal(component, hppV2ScopePulletCost); got != 0 {
|
||||||
|
t.Fatalf("expected pullet_cost 0 (feed laying bukan pullet), got %v", got)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestHppV2BopProductionScopeRange_NonNegativeAndProrated membuktikan Fix 2: BOP farm-level dihitung
|
||||||
|
// sebagai (expenseCum(end) - expenseCum(start)) × ratio(end) — range-correct & tidak pernah negatif.
|
||||||
|
// Range [2026-04-30, 2026-05-31] -> engine memakai endOfDay: start=2026-05-01, end=2026-06-01.
|
||||||
|
// Share kandang 50 = 30/(30+70) = 0.3.
|
||||||
|
// - REGULAR: expense farm tumbuh 1000 -> 1300 (delta 300) => 300 × 0.3 = 90.
|
||||||
|
// - EKSPEDISI: expense farm "turun" 500 -> 200 (delta -300, kasus uji clamp) => di-clamp ke 0.
|
||||||
|
func TestHppV2BopProductionScopeRange_NonNegativeAndProrated(t *testing.T) {
|
||||||
|
repo := &hppV2RepoStub{
|
||||||
|
contextByPFK: map[uint]*commonRepo.HppV2ProjectFlockKandangContext{
|
||||||
|
50: {ProjectFlockKandangID: 50, ProjectFlockID: 20, ProjectFlockCategory: string(utils.ProjectFlockCategoryLaying)},
|
||||||
|
},
|
||||||
|
pfkIDsByProject: map[uint][]uint{
|
||||||
|
20: {50, 51},
|
||||||
|
},
|
||||||
|
eggProductionByPFK: map[uint]struct {
|
||||||
|
pieces float64
|
||||||
|
kg float64
|
||||||
|
}{
|
||||||
|
50: {pieces: 300, kg: 30},
|
||||||
|
51: {pieces: 700, kg: 70},
|
||||||
|
},
|
||||||
|
expenseRowsByFarmDateKey: map[string][]commonRepo.HppV2ExpenseCostRow{
|
||||||
|
// REGULAR (ekspedisi=false): kumulatif 1000 (start) -> 1300 (end)
|
||||||
|
expenseFarmDateKey(20, false, mustTime(t, "2026-05-01")): {{TotalCost: 1000}},
|
||||||
|
expenseFarmDateKey(20, false, mustTime(t, "2026-06-01")): {{TotalCost: 800}, {TotalCost: 500}},
|
||||||
|
// EKSPEDISI (ekspedisi=true): kumulatif 500 (start) -> 200 (end) => delta negatif, harus di-clamp
|
||||||
|
expenseFarmDateKey(20, true, mustTime(t, "2026-05-01")): {{TotalCost: 500}},
|
||||||
|
expenseFarmDateKey(20, true, mustTime(t, "2026-06-01")): {{TotalCost: 200}},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
svc := NewHppV2Service(repo)
|
||||||
|
start := mustDate(t, "2026-04-30")
|
||||||
|
end := mustDate(t, "2026-05-31")
|
||||||
|
|
||||||
|
reg, err := svc.GetBopRegularProductionScopeRange(50, start, end)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("expected no error, got %v", err)
|
||||||
|
}
|
||||||
|
if reg != 90 {
|
||||||
|
t.Fatalf("expected BOP regular range 90 (300 × 0.3), got %v", reg)
|
||||||
|
}
|
||||||
|
|
||||||
|
eksp, err := svc.GetBopEkspedisiProductionScopeRange(50, start, end)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("expected no error, got %v", err)
|
||||||
|
}
|
||||||
|
if eksp != 0 {
|
||||||
|
t.Fatalf("expected BOP ekspedisi range clamped to 0 (delta negatif), got %v", eksp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -200,9 +200,12 @@ func ToMarketingListDTO(marketing *entity.Marketing, deliveryProducts []entity.M
|
|||||||
salesOrderProducts[i] = ToDeliveryMarketingProductDTO(product, marketing.MarketingType)
|
salesOrderProducts[i] = ToDeliveryMarketingProductDTO(product, marketing.MarketingType)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
var grandTotalSO float64
|
var grandTotalSO, grandTotalDO float64
|
||||||
for _, p := range marketing.Products {
|
for _, p := range marketing.Products {
|
||||||
grandTotalSO += p.TotalPrice
|
grandTotalSO += p.TotalPrice
|
||||||
|
if p.DeliveryProduct != nil && p.DeliveryProduct.DeliveryDate != nil {
|
||||||
|
grandTotalDO += p.DeliveryProduct.TotalPrice
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return MarketingListDTO{
|
return MarketingListDTO{
|
||||||
@@ -211,7 +214,7 @@ func ToMarketingListDTO(marketing *entity.Marketing, deliveryProducts []entity.M
|
|||||||
SalesPerson: salesPerson,
|
SalesPerson: salesPerson,
|
||||||
SoDocs: marketing.SoDocs,
|
SoDocs: marketing.SoDocs,
|
||||||
GrandTotalSO: grandTotalSO,
|
GrandTotalSO: grandTotalSO,
|
||||||
GrandTotalDO: marketing.GrandTotal,
|
GrandTotalDO: grandTotalDO,
|
||||||
SalesOrder: salesOrderProducts,
|
SalesOrder: salesOrderProducts,
|
||||||
DeliveryOrder: extractDeliveryGroupsFromProducts(marketing),
|
DeliveryOrder: extractDeliveryGroupsFromProducts(marketing),
|
||||||
CreatedUser: createdUser,
|
CreatedUser: createdUser,
|
||||||
|
|||||||
@@ -152,6 +152,29 @@ func (c *RepportController) GetExpenseDepreciation(ctx *fiber.Ctx) error {
|
|||||||
return ctx.Status(fiber.StatusOK).JSON(resp)
|
return ctx.Status(fiber.StatusOK).JSON(resp)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *RepportController) GetExpenseDepreciationV2(ctx *fiber.Ctx) error {
|
||||||
|
rows, meta, err := c.RepportService.GetExpenseDepreciationV2(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := struct {
|
||||||
|
Code int `json:"code"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Meta dto.ExpenseDepreciationV2MetaDTO `json:"meta"`
|
||||||
|
Data []dto.ExpenseDepreciationV2RowDTO `json:"data"`
|
||||||
|
}{
|
||||||
|
Code: fiber.StatusOK,
|
||||||
|
Status: "success",
|
||||||
|
Message: "Get expense depreciation report v2 successfully",
|
||||||
|
Meta: *meta,
|
||||||
|
Data: rows,
|
||||||
|
}
|
||||||
|
|
||||||
|
return ctx.Status(fiber.StatusOK).JSON(resp)
|
||||||
|
}
|
||||||
|
|
||||||
func (c *RepportController) GetExpenseDepreciationManualInputs(ctx *fiber.Ctx) error {
|
func (c *RepportController) GetExpenseDepreciationManualInputs(ctx *fiber.Ctx) error {
|
||||||
rows, meta, err := c.RepportService.GetExpenseDepreciationManualInputs(ctx)
|
rows, meta, err := c.RepportService.GetExpenseDepreciationManualInputs(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -457,6 +480,29 @@ func (c *RepportController) GetHppPerKandang(ctx *fiber.Ctx) error {
|
|||||||
return ctx.Status(fiber.StatusOK).JSON(resp)
|
return ctx.Status(fiber.StatusOK).JSON(resp)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *RepportController) GetHppPerFarm(ctx *fiber.Ctx) error {
|
||||||
|
data, meta, err := c.RepportService.GetHppPerFarm(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := struct {
|
||||||
|
Code int `json:"code"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Meta dto.HppPerFarmMetaDTO `json:"meta"`
|
||||||
|
Data dto.HppPerFarmResponseData `json:"data"`
|
||||||
|
}{
|
||||||
|
Code: fiber.StatusOK,
|
||||||
|
Status: "success",
|
||||||
|
Message: "Get HPP per farm successfully",
|
||||||
|
Meta: *meta,
|
||||||
|
Data: *data,
|
||||||
|
}
|
||||||
|
|
||||||
|
return ctx.Status(fiber.StatusOK).JSON(resp)
|
||||||
|
}
|
||||||
|
|
||||||
func (c *RepportController) GetCustomerPayment(ctx *fiber.Ctx) error {
|
func (c *RepportController) GetCustomerPayment(ctx *fiber.Ctx) error {
|
||||||
var customerIDs []uint
|
var customerIDs []uint
|
||||||
if customerIDsStr := ctx.Query("customer_ids"); customerIDsStr != "" {
|
if customerIDsStr := ctx.Query("customer_ids"); customerIDsStr != "" {
|
||||||
|
|||||||
@@ -40,6 +40,29 @@ type ExpenseDepreciationManualInputRowDTO struct {
|
|||||||
Note *string `json:"note"`
|
Note *string `json:"note"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ExpenseDepreciationV2MetaDTO struct {
|
||||||
|
ProjectFlockID int64 `json:"project_flock_id"`
|
||||||
|
FarmName string `json:"farm_name"`
|
||||||
|
LocationID int64 `json:"location_id"`
|
||||||
|
Period string `json:"period"`
|
||||||
|
Limit int `json:"limit"`
|
||||||
|
TotalDays int `json:"total_days"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ExpenseDepreciationV2RowDTO struct {
|
||||||
|
Date string `json:"date"`
|
||||||
|
DepreciationPercentEffective float64 `json:"depreciation_percent_effective"`
|
||||||
|
DepreciationValue float64 `json:"depreciation_value"`
|
||||||
|
PulletCostDayNTotal float64 `json:"pullet_cost_day_n_total"`
|
||||||
|
MultiplicationPercentage float64 `json:"multiplication_percentage"`
|
||||||
|
DayN int `json:"day_n"`
|
||||||
|
ChickinDate string `json:"chickin_date"`
|
||||||
|
TotalValuePulletAfterDepreciation float64 `json:"total_value_pullet_after_depreciation"`
|
||||||
|
StandardEffectiveDate string `json:"standard_effective_date,omitempty"`
|
||||||
|
TotalPopulation float64 `json:"total_population"`
|
||||||
|
Components any `json:"components"`
|
||||||
|
}
|
||||||
|
|
||||||
func NewExpenseDepreciationFiltersDTO(area, location, projectFlockID, period string) ExpenseDepreciationFiltersDTO {
|
func NewExpenseDepreciationFiltersDTO(area, location, projectFlockID, period string) ExpenseDepreciationFiltersDTO {
|
||||||
return ExpenseDepreciationFiltersDTO{
|
return ExpenseDepreciationFiltersDTO{
|
||||||
AreaID: area,
|
AreaID: area,
|
||||||
|
|||||||
@@ -0,0 +1,79 @@
|
|||||||
|
package dto
|
||||||
|
|
||||||
|
type HppPerFarmFiltersDTO struct {
|
||||||
|
AreaID string `json:"area_id"`
|
||||||
|
LocationID string `json:"location_id"`
|
||||||
|
StartDate string `json:"start_date"`
|
||||||
|
EndDate string `json:"end_date"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type HppPerFarmMetaDTO struct {
|
||||||
|
Page int `json:"page"`
|
||||||
|
Limit int `json:"limit"`
|
||||||
|
TotalPages int64 `json:"total_pages"`
|
||||||
|
TotalResults int64 `json:"total_results"`
|
||||||
|
Filters HppPerFarmFiltersDTO `json:"filters"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type HppPerFarmResponseData struct {
|
||||||
|
StartDate string `json:"start_date"`
|
||||||
|
EndDate string `json:"end_date"`
|
||||||
|
Rows []HppPerFarmRowDTO `json:"rows"`
|
||||||
|
Summary HppPerFarmSummaryDTO `json:"summary"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// HppPerFarmRowDTO is one farm (location) row, aggregating all LAYING project
|
||||||
|
// flocks within the same location over the selected date range.
|
||||||
|
type HppPerFarmRowDTO struct {
|
||||||
|
Location HppPerKandangLocationDTO `json:"location"`
|
||||||
|
// total_cost_rp = depreciation + pakan + ovk + bop (+ other production cost).
|
||||||
|
// DOC/pullet is NOT included here (it is expensed through depreciation);
|
||||||
|
// average_doc_price_rp is provided for information only.
|
||||||
|
TotalCostRp float64 `json:"total_cost_rp"`
|
||||||
|
FeedCostRp float64 `json:"feed_cost_rp"`
|
||||||
|
OvkCostRp float64 `json:"ovk_cost_rp"`
|
||||||
|
BopCostRp float64 `json:"bop_cost_rp"`
|
||||||
|
DepreciationRp float64 `json:"depreciation_rp"`
|
||||||
|
OtherCostRp float64 `json:"other_cost_rp"`
|
||||||
|
EggWeightRecordingKg float64 `json:"egg_weight_recording_kg"`
|
||||||
|
EggWeightDoKg float64 `json:"egg_weight_do_kg"`
|
||||||
|
HppPerKgProduction float64 `json:"hpp_per_kg_production"`
|
||||||
|
HppPerKgSales float64 `json:"hpp_per_kg_sales"`
|
||||||
|
AverageDocPriceRp int64 `json:"average_doc_price_rp"`
|
||||||
|
|
||||||
|
Flocks []HppPerFarmFlockDTO `json:"flocks"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// HppPerFarmFlockDTO is the per-project-flock breakdown inside a farm row.
|
||||||
|
type HppPerFarmFlockDTO struct {
|
||||||
|
ProjectFlockID int64 `json:"project_flock_id"`
|
||||||
|
FlockName string `json:"flock_name"`
|
||||||
|
TotalCostRp float64 `json:"total_cost_rp"`
|
||||||
|
FeedCostRp float64 `json:"feed_cost_rp"`
|
||||||
|
OvkCostRp float64 `json:"ovk_cost_rp"`
|
||||||
|
BopCostRp float64 `json:"bop_cost_rp"`
|
||||||
|
DepreciationRp float64 `json:"depreciation_rp"`
|
||||||
|
OtherCostRp float64 `json:"other_cost_rp"`
|
||||||
|
EggWeightRecordingKg float64 `json:"egg_weight_recording_kg"`
|
||||||
|
EggWeightDoKg float64 `json:"egg_weight_do_kg"`
|
||||||
|
HppPerKgProduction float64 `json:"hpp_per_kg_production"`
|
||||||
|
HppPerKgSales float64 `json:"hpp_per_kg_sales"`
|
||||||
|
AverageDocPriceRp int64 `json:"average_doc_price_rp"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type HppPerFarmSummaryDTO struct {
|
||||||
|
TotalCostRp float64 `json:"total_cost_rp"`
|
||||||
|
TotalEggWeightRecordingKg float64 `json:"total_egg_weight_recording_kg"`
|
||||||
|
TotalEggWeightDoKg float64 `json:"total_egg_weight_do_kg"`
|
||||||
|
AverageHppPerKgProduction float64 `json:"average_hpp_per_kg_production"`
|
||||||
|
AverageHppPerKgSales float64 `json:"average_hpp_per_kg_sales"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewHppPerFarmFiltersDTO(area, location, startDate, endDate string) HppPerFarmFiltersDTO {
|
||||||
|
return HppPerFarmFiltersDTO{
|
||||||
|
AreaID: area,
|
||||||
|
LocationID: location,
|
||||||
|
StartDate: startDate,
|
||||||
|
EndDate: endDate,
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -37,6 +37,7 @@ func (RepportModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate *
|
|||||||
purchaseSupplierRepository := repportRepo.NewPurchaseSupplierRepository(db)
|
purchaseSupplierRepository := repportRepo.NewPurchaseSupplierRepository(db)
|
||||||
debtSupplierRepository := repportRepo.NewDebtSupplierRepository(db)
|
debtSupplierRepository := repportRepo.NewDebtSupplierRepository(db)
|
||||||
hppPerKandangRepository := repportRepo.NewHppPerKandangRepository(db)
|
hppPerKandangRepository := repportRepo.NewHppPerKandangRepository(db)
|
||||||
|
hppPerFarmRepository := repportRepo.NewHppPerFarmRepository(db)
|
||||||
expenseDepreciationRepository := repportRepo.NewExpenseDepreciationRepository(db)
|
expenseDepreciationRepository := repportRepo.NewExpenseDepreciationRepository(db)
|
||||||
productionResultRepository := repportRepo.NewProductionResultRepository(db)
|
productionResultRepository := repportRepo.NewProductionResultRepository(db)
|
||||||
customerPaymentRepository := repportRepo.NewCustomerPaymentRepository(db)
|
customerPaymentRepository := repportRepo.NewCustomerPaymentRepository(db)
|
||||||
@@ -65,6 +66,7 @@ func (RepportModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate *
|
|||||||
purchaseSupplierRepository,
|
purchaseSupplierRepository,
|
||||||
debtSupplierRepository,
|
debtSupplierRepository,
|
||||||
hppPerKandangRepository,
|
hppPerKandangRepository,
|
||||||
|
hppPerFarmRepository,
|
||||||
productionResultRepository,
|
productionResultRepository,
|
||||||
customerPaymentRepository,
|
customerPaymentRepository,
|
||||||
balanceMonitoringRepository,
|
balanceMonitoringRepository,
|
||||||
|
|||||||
@@ -0,0 +1,233 @@
|
|||||||
|
package repositories
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
commonRepo "gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
||||||
|
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||||
|
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
||||||
|
"gitlab.com/mbugroup/lti-api.git/internal/utils/fifo"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
// HppPerFarmFlockMetaRow describes a LAYING project flock and the farm
|
||||||
|
// (location) it belongs to. Farm identity is project_flocks.location_id.
|
||||||
|
type HppPerFarmFlockMetaRow struct {
|
||||||
|
ProjectFlockID uint
|
||||||
|
FlockName string
|
||||||
|
LocationID uint
|
||||||
|
LocationName string
|
||||||
|
AreaID uint
|
||||||
|
}
|
||||||
|
|
||||||
|
// HppPerFarmDocRow holds the DOC/pullet acquisition cost trace per flock.
|
||||||
|
// Used only as an informational field (average_doc_price_rp); it is NOT part
|
||||||
|
// of total_cost because the pullet cost is expensed through depreciation.
|
||||||
|
type HppPerFarmDocRow struct {
|
||||||
|
ProjectFlockID uint
|
||||||
|
DocCost float64
|
||||||
|
DocQty float64
|
||||||
|
}
|
||||||
|
|
||||||
|
type HppPerFarmRepository interface {
|
||||||
|
GetCandidateFlocks(ctx context.Context, start time.Time, areaIDs, locationIDs []int64) ([]HppPerFarmFlockMetaRow, error)
|
||||||
|
SumRecordingEggWeightByFlock(ctx context.Context, start, endExclusive time.Time, projectFlockIDs []uint) (map[uint]float64, error)
|
||||||
|
SumMarketingDoTelurWeightByFlock(ctx context.Context, start, endExclusive time.Time, projectFlockIDs []uint) (map[uint]float64, error)
|
||||||
|
GetDocCostByFlock(ctx context.Context, projectFlockIDs []uint) (map[uint]HppPerFarmDocRow, error)
|
||||||
|
DB() *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
type hppPerFarmRepository struct {
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewHppPerFarmRepository(db *gorm.DB) HppPerFarmRepository {
|
||||||
|
return &hppPerFarmRepository{db: db}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *hppPerFarmRepository) DB() *gorm.DB {
|
||||||
|
return r.db
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetCandidateFlocks returns the LAYING project flocks (with their farm/location
|
||||||
|
// metadata) that are still active on or after the range start, scoped by area
|
||||||
|
// and location. Mirrors ExpenseDepreciationRepository.GetCandidateFarms but adds
|
||||||
|
// location info so flocks can be grouped per farm.
|
||||||
|
func (r *hppPerFarmRepository) GetCandidateFlocks(ctx context.Context, start time.Time, areaIDs, locationIDs []int64) ([]HppPerFarmFlockMetaRow, error) {
|
||||||
|
rows := make([]HppPerFarmFlockMetaRow, 0)
|
||||||
|
|
||||||
|
query := r.db.WithContext(ctx).
|
||||||
|
Table("project_flocks AS pf").
|
||||||
|
Select(`
|
||||||
|
DISTINCT pf.id AS project_flock_id,
|
||||||
|
pf.flock_name AS flock_name,
|
||||||
|
pf.location_id AS location_id,
|
||||||
|
loc.name AS location_name,
|
||||||
|
pf.area_id AS area_id`).
|
||||||
|
Joins("JOIN project_flock_kandangs AS pfk ON pfk.project_flock_id = pf.id").
|
||||||
|
Joins("JOIN locations AS loc ON loc.id = pf.location_id").
|
||||||
|
Where("pf.deleted_at IS NULL").
|
||||||
|
Where("pf.category = ?", utils.ProjectFlockCategoryLaying).
|
||||||
|
Where("(pfk.closed_at IS NULL OR DATE(pfk.closed_at) >= DATE(?))", start)
|
||||||
|
|
||||||
|
if len(areaIDs) > 0 {
|
||||||
|
query = query.Where("pf.area_id IN ?", areaIDs)
|
||||||
|
}
|
||||||
|
if len(locationIDs) > 0 {
|
||||||
|
query = query.Where("pf.location_id IN ?", locationIDs)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := query.Order("pf.location_id ASC, pf.id ASC").Scan(&rows).Error; err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return rows, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SumRecordingEggWeightByFlock sums recording_eggs.weight (kg) per project flock
|
||||||
|
// for non-rejected recordings whose record_datetime falls inside [start, endExclusive).
|
||||||
|
func (r *hppPerFarmRepository) SumRecordingEggWeightByFlock(ctx context.Context, start, endExclusive time.Time, projectFlockIDs []uint) (map[uint]float64, error) {
|
||||||
|
result := make(map[uint]float64)
|
||||||
|
if len(projectFlockIDs) == 0 {
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
latestApproval := r.db.WithContext(ctx).
|
||||||
|
Table("approvals AS a").
|
||||||
|
Select("a.approvable_id, a.action").
|
||||||
|
Joins(`
|
||||||
|
JOIN (
|
||||||
|
SELECT approvable_id, MAX(action_at) AS latest_action_at
|
||||||
|
FROM approvals
|
||||||
|
WHERE approvable_type = ?
|
||||||
|
GROUP BY approvable_id
|
||||||
|
) AS la ON la.approvable_id = a.approvable_id AND la.latest_action_at = a.action_at`,
|
||||||
|
string(utils.ApprovalWorkflowRecording),
|
||||||
|
)
|
||||||
|
|
||||||
|
type eggRow struct {
|
||||||
|
ProjectFlockID uint
|
||||||
|
Weight float64
|
||||||
|
}
|
||||||
|
rows := make([]eggRow, 0)
|
||||||
|
|
||||||
|
query := r.db.WithContext(ctx).
|
||||||
|
Table("recordings AS r").
|
||||||
|
Select(`
|
||||||
|
pfk.project_flock_id AS project_flock_id,
|
||||||
|
COALESCE(SUM(re.weight), 0) AS weight`).
|
||||||
|
Joins("JOIN project_flock_kandangs AS pfk ON pfk.id = r.project_flock_kandangs_id").
|
||||||
|
Joins("LEFT JOIN (?) AS la ON la.approvable_id = r.id", latestApproval).
|
||||||
|
Joins("JOIN recording_eggs AS re ON re.recording_id = r.id").
|
||||||
|
Where("pfk.project_flock_id IN ?", projectFlockIDs).
|
||||||
|
Where("r.record_datetime >= ? AND r.record_datetime < ?", start, endExclusive).
|
||||||
|
Where("r.deleted_at IS NULL").
|
||||||
|
Where("(la.action IS NULL OR la.action != ?)", string(entity.ApprovalActionRejected)).
|
||||||
|
Group("pfk.project_flock_id")
|
||||||
|
|
||||||
|
if err := query.Scan(&rows).Error; err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, row := range rows {
|
||||||
|
result[row.ProjectFlockID] = row.Weight
|
||||||
|
}
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SumMarketingDoTelurWeightByFlock sums delivered TELUR weight (marketing_delivery_products.total_weight)
|
||||||
|
// per project flock, for delivery_date inside [start, endExclusive). A delivery product that is
|
||||||
|
// attributed to multiple flocks is prorated by each flock's allocated qty share, so that
|
||||||
|
// the farm total equals the sum of its flocks.
|
||||||
|
func (r *hppPerFarmRepository) SumMarketingDoTelurWeightByFlock(ctx context.Context, start, endExclusive time.Time, projectFlockIDs []uint) (map[uint]float64, error) {
|
||||||
|
result := make(map[uint]float64)
|
||||||
|
if len(projectFlockIDs) == 0 {
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
telurFlags := []string{
|
||||||
|
string(utils.FlagTelur),
|
||||||
|
string(utils.FlagTelurUtuh),
|
||||||
|
string(utils.FlagTelurPecah),
|
||||||
|
string(utils.FlagTelurPutih),
|
||||||
|
string(utils.FlagTelurRetak),
|
||||||
|
}
|
||||||
|
|
||||||
|
// allocated qty per (marketing_delivery_product, project_flock)
|
||||||
|
attrByFlock := r.db.WithContext(ctx).
|
||||||
|
Table("(?) AS mda", commonRepo.MarketingDeliveryAttributionRowsQuery(r.db.WithContext(ctx))).
|
||||||
|
Select(`
|
||||||
|
mda.marketing_delivery_product_id AS mdp_id,
|
||||||
|
mda.project_flock_id AS project_flock_id,
|
||||||
|
SUM(mda.allocated_qty) AS flock_qty`).
|
||||||
|
Group("mda.marketing_delivery_product_id, mda.project_flock_id")
|
||||||
|
|
||||||
|
// prorate each delivery product's total_weight across its attributed flocks.
|
||||||
|
// Use EXISTS for the TELUR flag filter (not a JOIN) so a product carrying
|
||||||
|
// multiple egg flags does not fan out and double-count the weight share.
|
||||||
|
shareQuery := r.db.WithContext(ctx).
|
||||||
|
Table("(?) AS a", attrByFlock).
|
||||||
|
Select(`
|
||||||
|
a.project_flock_id AS project_flock_id,
|
||||||
|
mdp.total_weight * a.flock_qty / NULLIF(SUM(a.flock_qty) OVER (PARTITION BY a.mdp_id), 0) AS weight_share`).
|
||||||
|
Joins("JOIN marketing_delivery_products AS mdp ON mdp.id = a.mdp_id").
|
||||||
|
Joins("JOIN marketing_products AS mp ON mp.id = mdp.marketing_product_id").
|
||||||
|
Joins("JOIN product_warehouses AS pw ON pw.id = mp.product_warehouse_id").
|
||||||
|
Where("EXISTS (SELECT 1 FROM flags f WHERE f.flagable_id = pw.product_id AND f.flagable_type = ? AND f.name IN ?)", entity.FlagableTypeProduct, telurFlags).
|
||||||
|
Where("mdp.delivery_date >= ? AND mdp.delivery_date < ?", start, endExclusive)
|
||||||
|
|
||||||
|
type doRow struct {
|
||||||
|
ProjectFlockID uint
|
||||||
|
Weight float64
|
||||||
|
}
|
||||||
|
rows := make([]doRow, 0)
|
||||||
|
|
||||||
|
query := r.db.WithContext(ctx).
|
||||||
|
Table("(?) AS s", shareQuery).
|
||||||
|
Select(`
|
||||||
|
s.project_flock_id AS project_flock_id,
|
||||||
|
COALESCE(SUM(s.weight_share), 0) AS weight`).
|
||||||
|
Where("s.project_flock_id IN ?", projectFlockIDs).
|
||||||
|
Group("s.project_flock_id")
|
||||||
|
|
||||||
|
if err := query.Scan(&rows).Error; err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, row := range rows {
|
||||||
|
result[row.ProjectFlockID] = row.Weight
|
||||||
|
}
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetDocCostByFlock returns the DOC acquisition cost (qty * purchase price) and qty
|
||||||
|
// traced to chick-in per project flock. Informational only.
|
||||||
|
func (r *hppPerFarmRepository) GetDocCostByFlock(ctx context.Context, projectFlockIDs []uint) (map[uint]HppPerFarmDocRow, error) {
|
||||||
|
result := make(map[uint]HppPerFarmDocRow)
|
||||||
|
if len(projectFlockIDs) == 0 {
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
rows := make([]HppPerFarmDocRow, 0)
|
||||||
|
query := r.db.WithContext(ctx).
|
||||||
|
Table("project_chickins AS pc").
|
||||||
|
Select(`
|
||||||
|
pfk.project_flock_id AS project_flock_id,
|
||||||
|
COALESCE(SUM(sa.qty * COALESCE(pi.price, 0)), 0) AS doc_cost,
|
||||||
|
COALESCE(SUM(sa.qty), 0) AS doc_qty`).
|
||||||
|
Joins("JOIN project_flock_kandangs AS pfk ON pfk.id = pc.project_flock_kandang_id").
|
||||||
|
Joins("LEFT JOIN stock_allocations AS sa ON sa.usable_type = ? AND sa.usable_id = pc.id AND sa.stockable_type = ? AND sa.status = ? AND sa.allocation_purpose = ?", fifo.UsableKeyProjectChickin.String(), fifo.StockableKeyPurchaseItems.String(), entity.StockAllocationStatusActive, entity.StockAllocationPurposeTraceChickin).
|
||||||
|
Joins("LEFT JOIN purchase_items AS pi ON pi.id = sa.stockable_id").
|
||||||
|
Where("pfk.project_flock_id IN ?", projectFlockIDs).
|
||||||
|
Group("pfk.project_flock_id")
|
||||||
|
|
||||||
|
if err := query.Scan(&rows).Error; err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, row := range rows {
|
||||||
|
result[row.ProjectFlockID] = row
|
||||||
|
}
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
@@ -17,12 +17,14 @@ func RepportRoutes(v1 fiber.Router, u user.UserService, s repport.RepportService
|
|||||||
|
|
||||||
route.Get("/expense", m.RequirePermissions(m.P_ReportExpenseGetAll), ctrl.GetExpense)
|
route.Get("/expense", m.RequirePermissions(m.P_ReportExpenseGetAll), ctrl.GetExpense)
|
||||||
route.Get("/expense/depreciation", ctrl.GetExpenseDepreciation)
|
route.Get("/expense/depreciation", ctrl.GetExpenseDepreciation)
|
||||||
|
route.Get("/expense/v2/depreciation", ctrl.GetExpenseDepreciationV2)
|
||||||
route.Get("/expense/depreciation/manual-inputs", ctrl.GetExpenseDepreciationManualInputs)
|
route.Get("/expense/depreciation/manual-inputs", ctrl.GetExpenseDepreciationManualInputs)
|
||||||
route.Put("/expense/depreciation/manual-inputs", ctrl.UpsertExpenseDepreciationManualInput)
|
route.Put("/expense/depreciation/manual-inputs", ctrl.UpsertExpenseDepreciationManualInput)
|
||||||
route.Get("/marketing", m.RequirePermissions(m.P_ReportDeliveryGetAll), ctrl.GetMarketing)
|
route.Get("/marketing", m.RequirePermissions(m.P_ReportDeliveryGetAll), ctrl.GetMarketing)
|
||||||
route.Get("/purchase-supplier", m.RequirePermissions(m.P_ReportPurchaseSupplierGetAll), ctrl.GetPurchaseSupplier)
|
route.Get("/purchase-supplier", m.RequirePermissions(m.P_ReportPurchaseSupplierGetAll), ctrl.GetPurchaseSupplier)
|
||||||
route.Get("/debt-supplier", m.RequirePermissions(m.P_ReportDebtSupplierGetAll), ctrl.GetDebtSupplier)
|
route.Get("/debt-supplier", m.RequirePermissions(m.P_ReportDebtSupplierGetAll), ctrl.GetDebtSupplier)
|
||||||
route.Get("/hpp-per-kandang", m.RequirePermissions(m.P_ReportHppPerKandangGetAll), ctrl.GetHppPerKandang)
|
route.Get("/hpp-per-kandang", m.RequirePermissions(m.P_ReportHppPerKandangGetAll), ctrl.GetHppPerKandang)
|
||||||
|
route.Get("/hpp-per-farm", m.RequirePermissions(m.P_ReportHppPerKandangGetAll), ctrl.GetHppPerFarm)
|
||||||
route.Get("/hpp-v2-breakdown", m.RequirePermissions(m.P_ReportHppPerKandangGetAll), ctrl.GetHppV2Breakdown)
|
route.Get("/hpp-v2-breakdown", m.RequirePermissions(m.P_ReportHppPerKandangGetAll), ctrl.GetHppV2Breakdown)
|
||||||
route.Get("/production-result/:idProjectFlockKandang", m.RequirePermissions(m.P_ReportProductionResultGetAll), ctrl.GetProductionResult)
|
route.Get("/production-result/:idProjectFlockKandang", m.RequirePermissions(m.P_ReportProductionResultGetAll), ctrl.GetProductionResult)
|
||||||
route.Get("/customer-payment", m.RequirePermissions(m.P_ReportCustomerPaymentGetAll), ctrl.GetCustomerPayment)
|
route.Get("/customer-payment", m.RequirePermissions(m.P_ReportCustomerPaymentGetAll), ctrl.GetCustomerPayment)
|
||||||
|
|||||||
@@ -0,0 +1,93 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
approvalService "gitlab.com/mbugroup/lti-api.git/internal/common/service"
|
||||||
|
)
|
||||||
|
|
||||||
|
// production-scope total should sum only parts tagged production_cost (a part
|
||||||
|
// tagged with both scopes still counts once).
|
||||||
|
func TestHppPerFarmProductionScopeTotalPartLevelScopes(t *testing.T) {
|
||||||
|
comp := &approvalService.HppV2Component{
|
||||||
|
Code: "PAKAN",
|
||||||
|
Parts: []approvalService.HppV2ComponentPart{
|
||||||
|
{Total: 100, Scopes: []string{"production_cost"}},
|
||||||
|
{Total: 50, Scopes: []string{"pullet_cost"}},
|
||||||
|
{Total: 25, Scopes: []string{"production_cost", "pullet_cost"}},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
if got := hppPerFarmProductionScopeTotal(comp); got != 125 {
|
||||||
|
t.Fatalf("expected 125, got %v", got)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// when parts carry no scopes, fall back to the component-level scope.
|
||||||
|
func TestHppPerFarmProductionScopeTotalComponentLevelFallback(t *testing.T) {
|
||||||
|
prod := &approvalService.HppV2Component{
|
||||||
|
Code: "DIRECT_PULLET_PURCHASE",
|
||||||
|
Scopes: []string{"production_cost"},
|
||||||
|
Total: 300,
|
||||||
|
Parts: []approvalService.HppV2ComponentPart{{Total: 300}},
|
||||||
|
}
|
||||||
|
if got := hppPerFarmProductionScopeTotal(prod); got != 300 {
|
||||||
|
t.Fatalf("expected 300 component fallback, got %v", got)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DOC/pullet is pullet-scope only -> contributes 0 to production cost,
|
||||||
|
// which is exactly why it must not be added to total_cost (depreciation
|
||||||
|
// already expenses the pullet).
|
||||||
|
pulletOnly := &approvalService.HppV2Component{
|
||||||
|
Code: "DOC_CHICKIN",
|
||||||
|
Scopes: []string{"pullet_cost"},
|
||||||
|
Total: 999,
|
||||||
|
Parts: []approvalService.HppV2ComponentPart{{Total: 999}},
|
||||||
|
}
|
||||||
|
if got := hppPerFarmProductionScopeTotal(pulletOnly); got != 0 {
|
||||||
|
t.Fatalf("expected 0 for pullet-only component, got %v", got)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestHppPerFarmProductionScopeTotalsByCode(t *testing.T) {
|
||||||
|
b := &approvalService.HppV2Breakdown{
|
||||||
|
Components: []approvalService.HppV2Component{
|
||||||
|
{Code: "PAKAN", Parts: []approvalService.HppV2ComponentPart{{Total: 100, Scopes: []string{"production_cost"}}}},
|
||||||
|
{Code: "OVK", Parts: []approvalService.HppV2ComponentPart{{Total: 40, Scopes: []string{"production_cost"}}}},
|
||||||
|
{Code: "DOC_CHICKIN", Scopes: []string{"pullet_cost"}, Total: 500, Parts: []approvalService.HppV2ComponentPart{{Total: 500}}},
|
||||||
|
{Code: "DEPRECIATION", Scopes: []string{"production_cost"}, Total: 30, Parts: []approvalService.HppV2ComponentPart{{Total: 30, Scopes: []string{"production_cost"}}}},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
got := hppPerFarmProductionScopeTotalsByCode(b)
|
||||||
|
if got["PAKAN"] != 100 {
|
||||||
|
t.Fatalf("expected PAKAN 100, got %v", got["PAKAN"])
|
||||||
|
}
|
||||||
|
if got["OVK"] != 40 {
|
||||||
|
t.Fatalf("expected OVK 40, got %v", got["OVK"])
|
||||||
|
}
|
||||||
|
if got["DOC_CHICKIN"] != 0 {
|
||||||
|
t.Fatalf("expected DOC_CHICKIN production scope 0, got %v", got["DOC_CHICKIN"])
|
||||||
|
}
|
||||||
|
if got["DEPRECIATION"] != 30 {
|
||||||
|
t.Fatalf("expected DEPRECIATION 30, got %v", got["DEPRECIATION"])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestHppPerFarmSafeDiv(t *testing.T) {
|
||||||
|
cases := []struct {
|
||||||
|
num, den, want float64
|
||||||
|
}{
|
||||||
|
{100, 4, 25},
|
||||||
|
{100, 0, 0},
|
||||||
|
{100, -5, 0},
|
||||||
|
{0, 0, 0},
|
||||||
|
}
|
||||||
|
for _, c := range cases {
|
||||||
|
if got := hppPerFarmSafeDiv(c.num, c.den); got != c.want {
|
||||||
|
t.Fatalf("safeDiv(%v,%v)=%v want %v", c.num, c.den, got, c.want)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if got := hppPerFarmSafeDiv(math.Inf(1), 1); got != 0 {
|
||||||
|
t.Fatalf("expected 0 for inf numerator, got %v", got)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -43,12 +43,14 @@ import (
|
|||||||
type RepportService interface {
|
type RepportService interface {
|
||||||
GetExpense(ctx *fiber.Ctx, params *validation.ExpenseQuery) ([]dto.RepportExpenseListDTO, int64, error)
|
GetExpense(ctx *fiber.Ctx, params *validation.ExpenseQuery) ([]dto.RepportExpenseListDTO, int64, error)
|
||||||
GetExpenseDepreciation(ctx *fiber.Ctx) ([]dto.ExpenseDepreciationRowDTO, *dto.ExpenseDepreciationMetaDTO, error)
|
GetExpenseDepreciation(ctx *fiber.Ctx) ([]dto.ExpenseDepreciationRowDTO, *dto.ExpenseDepreciationMetaDTO, error)
|
||||||
|
GetExpenseDepreciationV2(ctx *fiber.Ctx) ([]dto.ExpenseDepreciationV2RowDTO, *dto.ExpenseDepreciationV2MetaDTO, error)
|
||||||
GetExpenseDepreciationManualInputs(ctx *fiber.Ctx) ([]dto.ExpenseDepreciationManualInputRowDTO, *dto.ExpenseDepreciationMetaDTO, error)
|
GetExpenseDepreciationManualInputs(ctx *fiber.Ctx) ([]dto.ExpenseDepreciationManualInputRowDTO, *dto.ExpenseDepreciationMetaDTO, error)
|
||||||
UpsertExpenseDepreciationManualInput(ctx *fiber.Ctx, req *validation.ExpenseDepreciationManualInputUpsert) (*dto.ExpenseDepreciationManualInputRowDTO, error)
|
UpsertExpenseDepreciationManualInput(ctx *fiber.Ctx, req *validation.ExpenseDepreciationManualInputUpsert) (*dto.ExpenseDepreciationManualInputRowDTO, error)
|
||||||
GetMarketing(ctx *fiber.Ctx, params *validation.MarketingQuery) ([]dto.RepportMarketingItemDTO, int64, error)
|
GetMarketing(ctx *fiber.Ctx, params *validation.MarketingQuery) ([]dto.RepportMarketingItemDTO, int64, error)
|
||||||
GetPurchaseSupplier(ctx *fiber.Ctx, params *validation.PurchaseSupplierQuery) ([]dto.PurchaseSupplierDTO, int64, error)
|
GetPurchaseSupplier(ctx *fiber.Ctx, params *validation.PurchaseSupplierQuery) ([]dto.PurchaseSupplierDTO, int64, error)
|
||||||
GetDebtSupplier(ctx *fiber.Ctx, params *validation.DebtSupplierQuery) ([]dto.DebtSupplierDTO, int64, error)
|
GetDebtSupplier(ctx *fiber.Ctx, params *validation.DebtSupplierQuery) ([]dto.DebtSupplierDTO, int64, error)
|
||||||
GetHppPerKandang(ctx *fiber.Ctx) (*dto.HppPerKandangResponseData, *dto.HppPerKandangMetaDTO, 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)
|
GetHppV2Breakdown(ctx *fiber.Ctx, params *validation.HppV2BreakdownQuery) (*approvalService.HppV2Breakdown, error)
|
||||||
GetProductionResult(ctx *fiber.Ctx, params *validation.ProductionResultQuery) ([]dto.ProductionResultDTO, int64, error)
|
GetProductionResult(ctx *fiber.Ctx, params *validation.ProductionResultQuery) ([]dto.ProductionResultDTO, int64, error)
|
||||||
GetCustomerPayment(ctx *fiber.Ctx, params *validation.CustomerPaymentQuery) ([]dto.CustomerPaymentReportItem, int64, error)
|
GetCustomerPayment(ctx *fiber.Ctx, params *validation.CustomerPaymentQuery) ([]dto.CustomerPaymentReportItem, int64, error)
|
||||||
@@ -73,6 +75,7 @@ type repportService struct {
|
|||||||
PurchaseSupplierRepo repportRepo.PurchaseSupplierRepository
|
PurchaseSupplierRepo repportRepo.PurchaseSupplierRepository
|
||||||
DebtSupplierRepo repportRepo.DebtSupplierRepository
|
DebtSupplierRepo repportRepo.DebtSupplierRepository
|
||||||
HppPerKandangRepo repportRepo.HppPerKandangRepository
|
HppPerKandangRepo repportRepo.HppPerKandangRepository
|
||||||
|
HppPerFarmRepo repportRepo.HppPerFarmRepository
|
||||||
ProductionResultRepo repportRepo.ProductionResultRepository
|
ProductionResultRepo repportRepo.ProductionResultRepository
|
||||||
CustomerPaymentRepo repportRepo.CustomerPaymentRepository
|
CustomerPaymentRepo repportRepo.CustomerPaymentRepository
|
||||||
BalanceMonitoringRepo repportRepo.BalanceMonitoringRepository
|
BalanceMonitoringRepo repportRepo.BalanceMonitoringRepository
|
||||||
@@ -106,6 +109,7 @@ func NewRepportService(
|
|||||||
purchaseSupplierRepo repportRepo.PurchaseSupplierRepository,
|
purchaseSupplierRepo repportRepo.PurchaseSupplierRepository,
|
||||||
debtSupplierRepo repportRepo.DebtSupplierRepository,
|
debtSupplierRepo repportRepo.DebtSupplierRepository,
|
||||||
hppPerKandangRepo repportRepo.HppPerKandangRepository,
|
hppPerKandangRepo repportRepo.HppPerKandangRepository,
|
||||||
|
hppPerFarmRepo repportRepo.HppPerFarmRepository,
|
||||||
productionResultRepo repportRepo.ProductionResultRepository,
|
productionResultRepo repportRepo.ProductionResultRepository,
|
||||||
customerPaymentRepo repportRepo.CustomerPaymentRepository,
|
customerPaymentRepo repportRepo.CustomerPaymentRepository,
|
||||||
balanceMonitoringRepo repportRepo.BalanceMonitoringRepository,
|
balanceMonitoringRepo repportRepo.BalanceMonitoringRepository,
|
||||||
@@ -130,6 +134,7 @@ func NewRepportService(
|
|||||||
PurchaseSupplierRepo: purchaseSupplierRepo,
|
PurchaseSupplierRepo: purchaseSupplierRepo,
|
||||||
DebtSupplierRepo: debtSupplierRepo,
|
DebtSupplierRepo: debtSupplierRepo,
|
||||||
HppPerKandangRepo: hppPerKandangRepo,
|
HppPerKandangRepo: hppPerKandangRepo,
|
||||||
|
HppPerFarmRepo: hppPerFarmRepo,
|
||||||
ProductionResultRepo: productionResultRepo,
|
ProductionResultRepo: productionResultRepo,
|
||||||
CustomerPaymentRepo: customerPaymentRepo,
|
CustomerPaymentRepo: customerPaymentRepo,
|
||||||
BalanceMonitoringRepo: balanceMonitoringRepo,
|
BalanceMonitoringRepo: balanceMonitoringRepo,
|
||||||
@@ -355,6 +360,182 @@ func (s *repportService) GetExpenseDepreciation(ctx *fiber.Ctx) ([]dto.ExpenseDe
|
|||||||
return rows[offset:end], meta, nil
|
return rows[offset:end], meta, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *repportService) GetExpenseDepreciationV2(ctx *fiber.Ctx) ([]dto.ExpenseDepreciationV2RowDTO, *dto.ExpenseDepreciationV2MetaDTO, error) {
|
||||||
|
params, err := s.parseExpenseDepreciationV2Query(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.ExpenseDepreciationRepo == nil {
|
||||||
|
return nil, nil, fiber.NewError(fiber.StatusInternalServerError, "expense depreciation repository is not configured")
|
||||||
|
}
|
||||||
|
if s.HppCostRepo == nil {
|
||||||
|
return nil, nil, fiber.NewError(fiber.StatusInternalServerError, "hpp cost repository is not configured")
|
||||||
|
}
|
||||||
|
if s.HppV2Svc == nil {
|
||||||
|
return nil, nil, fiber.NewError(fiber.StatusInternalServerError, "hpp v2 service is not configured")
|
||||||
|
}
|
||||||
|
|
||||||
|
location, err := time.LoadLocation("Asia/Jakarta")
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, fiber.NewError(fiber.StatusInternalServerError, "failed to load timezone configuration")
|
||||||
|
}
|
||||||
|
periodDate, err := time.ParseInLocation("2006-01-02", params.Period, location)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, fiber.NewError(fiber.StatusBadRequest, "period must follow format YYYY-MM-DD")
|
||||||
|
}
|
||||||
|
|
||||||
|
limit := params.Limit
|
||||||
|
if limit <= 0 {
|
||||||
|
limit = 10
|
||||||
|
}
|
||||||
|
|
||||||
|
farmID := uint(params.ProjectFlockID)
|
||||||
|
kandangIDs, err := s.HppCostRepo.GetProjectFlockKandangIDs(ctx.Context(), farmID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
if len(kandangIDs) == 0 {
|
||||||
|
return nil, nil, fiber.NewError(fiber.StatusNotFound, "project flock has no kandangs")
|
||||||
|
}
|
||||||
|
|
||||||
|
var farmName string
|
||||||
|
if err := s.db.WithContext(ctx.Context()).
|
||||||
|
Table("project_flocks").
|
||||||
|
Select("flock_name").
|
||||||
|
Where("id = ? AND deleted_at IS NULL", farmID).
|
||||||
|
Scan(&farmName).Error; err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
if farmName == "" {
|
||||||
|
return nil, nil, fiber.NewError(fiber.StatusNotFound, "project flock not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
rows := make([]dto.ExpenseDepreciationV2RowDTO, 0, limit)
|
||||||
|
actualDays := 0
|
||||||
|
|
||||||
|
for i := 0; i < limit; i++ {
|
||||||
|
dayDate := periodDate.AddDate(0, 0, i)
|
||||||
|
dayStr := dayDate.Format("2006-01-02")
|
||||||
|
|
||||||
|
var totalDepreciationValue float64
|
||||||
|
var totalPulletCostDayN float64
|
||||||
|
var totalPopulation float64
|
||||||
|
var allKandangComponents []depreciationKandangComponent
|
||||||
|
|
||||||
|
for _, kandangID := range kandangIDs {
|
||||||
|
breakdown, err := s.HppV2Svc.CalculateHppBreakdown(kandangID, &dayDate)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
if breakdown == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
depreciationComponent := hppV2FindDepreciationComponent(breakdown)
|
||||||
|
if depreciationComponent == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, part := range depreciationComponent.Parts {
|
||||||
|
if part.Total <= 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
houseType := approvalService.NormalizeDepreciationHouseType(breakdown.HouseType)
|
||||||
|
component := depreciationKandangComponent{
|
||||||
|
ProjectFlockKandangID: breakdown.ProjectFlockKandangID,
|
||||||
|
KandangID: breakdown.KandangID,
|
||||||
|
KandangName: breakdown.KandangName,
|
||||||
|
SourceProjectFlockID: hppV2DetailUint(part.Details, "source_project_flock_id"),
|
||||||
|
HouseType: houseType,
|
||||||
|
DayN: hppV2DetailInt(part.Details, "schedule_day"),
|
||||||
|
DepreciationPercent: hppV2DetailFloat(part.Details, "depreciation_percent"),
|
||||||
|
MultiplicationPercentage: hppV2DetailFloat(part.Details, "multiplication_percentage"),
|
||||||
|
PulletCostDayN: hppV2DetailFloat(part.Details, "pullet_cost_day_n"),
|
||||||
|
DepreciationValue: part.Total,
|
||||||
|
TotalValuePulletAfterDepreciation: hppV2DetailFloat(part.Details, "total_value_pullet_after_depreciation"),
|
||||||
|
DepreciationSource: part.Code,
|
||||||
|
OriginDate: hppV2DetailString(part.Details, "origin_date"),
|
||||||
|
ChickinDate: hppV2DetailString(part.Details, "origin_date"),
|
||||||
|
StandardEffectiveDate: hppV2DetailString(part.Details, "standard_effective_date"),
|
||||||
|
Population: hppV2DetailFloat(part.Details, "kandang_population"),
|
||||||
|
}
|
||||||
|
|
||||||
|
if component.HouseType == "" {
|
||||||
|
component.HouseType = approvalService.NormalizeDepreciationHouseType(hppV2DetailString(part.Details, "house_type"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if ref := hppV2FindReference(part.References, "laying_transfer"); ref != nil {
|
||||||
|
component.TransferID = ref.ID
|
||||||
|
component.TransferDate = ref.Date
|
||||||
|
component.TransferQty = ref.Qty
|
||||||
|
}
|
||||||
|
|
||||||
|
if part.Code == "manual_cutover" {
|
||||||
|
if startDay := hppV2DetailInt(part.Details, "start_schedule_day"); startDay > 0 {
|
||||||
|
component.StartScheduleDay = &startDay
|
||||||
|
}
|
||||||
|
component.CutoverDate = hppV2DetailString(part.Details, "cutover_date")
|
||||||
|
if manualID := hppV2DetailUint(part.Details, "manual_input_id"); manualID > 0 {
|
||||||
|
component.ManualInputID = &manualID
|
||||||
|
}
|
||||||
|
if component.ManualInputID == nil {
|
||||||
|
if ref := hppV2FindReference(part.References, "farm_depreciation_manual_input"); ref != nil && ref.ID > 0 {
|
||||||
|
manualID := ref.ID
|
||||||
|
component.ManualInputID = &manualID
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
totalPulletCostDayN += component.PulletCostDayN
|
||||||
|
totalDepreciationValue += component.DepreciationValue
|
||||||
|
totalPopulation += component.Population
|
||||||
|
allKandangComponents = append(allKandangComponents, component)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
effectivePercent := approvalService.CalculateEffectiveDepreciationPercent(totalDepreciationValue, totalPulletCostDayN)
|
||||||
|
|
||||||
|
components := depreciationFarmComponents{
|
||||||
|
KandangCount: len(allKandangComponents),
|
||||||
|
TotalPopulation: totalPopulation,
|
||||||
|
Kandang: allKandangComponents,
|
||||||
|
}
|
||||||
|
componentsJSON, _ := json.Marshal(components)
|
||||||
|
|
||||||
|
multiplicationPercentage, dayN, chickinDate, standardEffectiveDate := depreciationSnapshotInfo(parseSnapshotComponents(componentsJSON))
|
||||||
|
|
||||||
|
rows = append(rows, dto.ExpenseDepreciationV2RowDTO{
|
||||||
|
Date: dayStr,
|
||||||
|
DepreciationPercentEffective: effectivePercent,
|
||||||
|
DepreciationValue: totalDepreciationValue,
|
||||||
|
PulletCostDayNTotal: totalPulletCostDayN,
|
||||||
|
MultiplicationPercentage: multiplicationPercentage,
|
||||||
|
DayN: dayN,
|
||||||
|
ChickinDate: chickinDate,
|
||||||
|
TotalValuePulletAfterDepreciation: totalPulletCostDayN - totalDepreciationValue,
|
||||||
|
StandardEffectiveDate: standardEffectiveDate,
|
||||||
|
TotalPopulation: totalPopulation,
|
||||||
|
Components: parseSnapshotComponents(componentsJSON),
|
||||||
|
})
|
||||||
|
actualDays++
|
||||||
|
}
|
||||||
|
|
||||||
|
meta := &dto.ExpenseDepreciationV2MetaDTO{
|
||||||
|
ProjectFlockID: params.ProjectFlockID,
|
||||||
|
FarmName: farmName,
|
||||||
|
LocationID: params.LocationID,
|
||||||
|
Period: params.Period,
|
||||||
|
Limit: limit,
|
||||||
|
TotalDays: actualDays,
|
||||||
|
}
|
||||||
|
|
||||||
|
return rows, meta, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (s *repportService) GetExpenseDepreciationManualInputs(ctx *fiber.Ctx) ([]dto.ExpenseDepreciationManualInputRowDTO, *dto.ExpenseDepreciationMetaDTO, error) {
|
func (s *repportService) GetExpenseDepreciationManualInputs(ctx *fiber.Ctx) ([]dto.ExpenseDepreciationManualInputRowDTO, *dto.ExpenseDepreciationMetaDTO, error) {
|
||||||
params, filters, err := s.parseExpenseDepreciationQuery(ctx)
|
params, filters, err := s.parseExpenseDepreciationQuery(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -2945,6 +3126,534 @@ func (s *repportService) parseHppPerKandangQuery(ctx *fiber.Ctx) (*validation.Hp
|
|||||||
return params, filters, nil
|
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 dihitung range-correct via engine (hindari differential rasio egg-weight yang bisa
|
||||||
|
// negatif saat share antar kandang bergeser). Keluarkan kode BOP dari codeTotals agar tidak
|
||||||
|
// ikut terjumlah dua kali di akumulasi 'nonDepreciation'/'other'.
|
||||||
|
delete(codeTotals, hppPerFarmComponentBopRegular)
|
||||||
|
delete(codeTotals, hppPerFarmComponentBopEkspedisi)
|
||||||
|
|
||||||
|
bop, err := s.hppPerFarmFlockBopRange(ctx.Context(), flockID, startBreakdownDate, endBreakdownDate)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
nonDepreciation := bop
|
||||||
|
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
|
||||||
|
}
|
||||||
|
|
||||||
|
// hppPerFarmFlockBopRange menjumlah BOP production_cost range-correct (BOP_REGULAR + BOP_EKSPEDISI)
|
||||||
|
// untuk seluruh PFK dalam flock, memakai GetBop*ProductionScopeRange di engine. Pendekatan ini
|
||||||
|
// menghitung delta expense kumulatif lalu memproratanya dengan rasio akhir-range — bukan
|
||||||
|
// men-differensiasi dua angka yang sudah diprorata berbeda — sehingga tidak pernah negatif.
|
||||||
|
func (s *repportService) hppPerFarmFlockBopRange(ctx context.Context, projectFlockID uint, startBreakdownDate, endBreakdownDate time.Time) (float64, error) {
|
||||||
|
if s.HppCostRepo == nil {
|
||||||
|
return 0, errors.New("hpp cost repository is not configured")
|
||||||
|
}
|
||||||
|
if s.HppV2Svc == nil {
|
||||||
|
return 0, errors.New("hpp v2 service is not configured")
|
||||||
|
}
|
||||||
|
|
||||||
|
pfkIDs, err := s.HppCostRepo.GetProjectFlockKandangIDs(ctx, projectFlockID)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
total := 0.0
|
||||||
|
for _, pfkID := range pfkIDs {
|
||||||
|
reg, err := s.HppV2Svc.GetBopRegularProductionScopeRange(pfkID, &startBreakdownDate, &endBreakdownDate)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
eksp, err := s.HppV2Svc.GetBopEkspedisiProductionScopeRange(pfkID, &startBreakdownDate, &endBreakdownDate)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
total += reg + eksp
|
||||||
|
}
|
||||||
|
|
||||||
|
return total, 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", "")
|
||||||
|
|
||||||
|
if strings.TrimSpace(startDate) == "" {
|
||||||
|
return nil, dto.HppPerFarmFiltersDTO{}, fiber.NewError(fiber.StatusBadRequest, "start_date is required")
|
||||||
|
}
|
||||||
|
if strings.TrimSpace(endDate) == "" {
|
||||||
|
return nil, dto.HppPerFarmFiltersDTO{}, fiber.NewError(fiber.StatusBadRequest, "end_date is required")
|
||||||
|
}
|
||||||
|
if strings.TrimSpace(rawLocation) == "" {
|
||||||
|
return nil, dto.HppPerFarmFiltersDTO{}, fiber.NewError(fiber.StatusBadRequest, "location_id is required")
|
||||||
|
}
|
||||||
|
|
||||||
|
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) {
|
func (s *repportService) parseExpenseDepreciationQuery(ctx *fiber.Ctx) (*validation.ExpenseDepreciationQuery, dto.ExpenseDepreciationFiltersDTO, error) {
|
||||||
page := ctx.QueryInt("page", 1)
|
page := ctx.QueryInt("page", 1)
|
||||||
if page < 1 {
|
if page < 1 {
|
||||||
@@ -3025,6 +3734,45 @@ func (s *repportService) parseExpenseDepreciationQuery(ctx *fiber.Ctx) (*validat
|
|||||||
return params, filters, nil
|
return params, filters, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *repportService) parseExpenseDepreciationV2Query(ctx *fiber.Ctx) (*validation.ExpenseDepreciationV2Query, error) {
|
||||||
|
limit := ctx.QueryInt("limit", 10)
|
||||||
|
if limit < 1 {
|
||||||
|
limit = 10
|
||||||
|
}
|
||||||
|
period := strings.TrimSpace(ctx.Query("period", ""))
|
||||||
|
locationID := ctx.QueryInt("location_id", 0)
|
||||||
|
projectFlockID := ctx.QueryInt("project_flock_id", 0)
|
||||||
|
|
||||||
|
locationScope, err := m.ResolveLocationScope(ctx, s.ExpenseRealizationRepo.DB())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if locationScope.Restrict && locationID > 0 {
|
||||||
|
allowed := toInt64Slice(locationScope.IDs)
|
||||||
|
if len(allowed) == 0 {
|
||||||
|
return nil, fiber.NewError(fiber.StatusForbidden, "no location access")
|
||||||
|
}
|
||||||
|
found := false
|
||||||
|
for _, id := range allowed {
|
||||||
|
if id == int64(locationID) {
|
||||||
|
found = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !found {
|
||||||
|
return nil, fiber.NewError(fiber.StatusForbidden, "location not in scope")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &validation.ExpenseDepreciationV2Query{
|
||||||
|
Limit: limit,
|
||||||
|
Period: period,
|
||||||
|
LocationID: int64(locationID),
|
||||||
|
ProjectFlockID: int64(projectFlockID),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
func parseCommaSeparatedInt64s(raw string) ([]int64, error) {
|
func parseCommaSeparatedInt64s(raw string) ([]int64, error) {
|
||||||
raw = strings.TrimSpace(raw)
|
raw = strings.TrimSpace(raw)
|
||||||
if raw == "" {
|
if raw == "" {
|
||||||
|
|||||||
@@ -78,6 +78,15 @@ type HppPerKandangQuery struct {
|
|||||||
WeightMax *float64 `query:"-"`
|
WeightMax *float64 `query:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type HppPerFarmQuery struct {
|
||||||
|
Page int `query:"page" validate:"omitempty,min=1,gt=0"`
|
||||||
|
Limit int `query:"limit" validate:"omitempty,min=1,gt=0"`
|
||||||
|
StartDate string `query:"start_date" validate:"required,datetime=2006-01-02"`
|
||||||
|
EndDate string `query:"end_date" validate:"required,datetime=2006-01-02"`
|
||||||
|
AreaIDs []int64 `query:"-"`
|
||||||
|
LocationIDs []int64 `query:"-"`
|
||||||
|
}
|
||||||
|
|
||||||
type HppV2BreakdownQuery struct {
|
type HppV2BreakdownQuery struct {
|
||||||
ProjectFlockKandangID uint `query:"project_flock_kandang_id" validate:"required,gt=0"`
|
ProjectFlockKandangID uint `query:"project_flock_kandang_id" validate:"required,gt=0"`
|
||||||
Period string `query:"period" validate:"required,datetime=2006-01-02"`
|
Period string `query:"period" validate:"required,datetime=2006-01-02"`
|
||||||
@@ -93,6 +102,13 @@ type ExpenseDepreciationQuery struct {
|
|||||||
LocationIDs []int64 `query:"-"`
|
LocationIDs []int64 `query:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ExpenseDepreciationV2Query struct {
|
||||||
|
Limit int `query:"limit" validate:"omitempty,min=1,max=90"`
|
||||||
|
Period string `query:"period" validate:"required,datetime=2006-01-02"`
|
||||||
|
LocationID int64 `query:"location_id" validate:"omitempty,gt=0"`
|
||||||
|
ProjectFlockID int64 `query:"project_flock_id" validate:"required,gt=0"`
|
||||||
|
}
|
||||||
|
|
||||||
type ExpenseDepreciationManualInputUpsert struct {
|
type ExpenseDepreciationManualInputUpsert struct {
|
||||||
ProjectFlockID uint `json:"project_flock_id" validate:"required,gt=0"`
|
ProjectFlockID uint `json:"project_flock_id" validate:"required,gt=0"`
|
||||||
TotalCost float64 `json:"total_cost" validate:"required,gte=0"`
|
TotalCost float64 `json:"total_cost" validate:"required,gte=0"`
|
||||||
|
|||||||
Reference in New Issue
Block a user