Compare commits

...

41 Commits

Author SHA1 Message Date
giovanni 59d72f20b4 non active phase daily checklist 2026-06-08 21:01:20 +07:00
Giovanni Gabriel Septriadi 540434e33b Merge branch 'development' into 'production'
Development

See merge request mbugroup/lti-api!612
2026-06-08 05:51:32 +00:00
Giovanni Gabriel Septriadi 0ebad48348 Merge branch 'fix/depretiatio-response' into 'development'
adjust total bobot laporan keuangan

See merge request mbugroup/lti-api!611
2026-06-08 05:39:09 +00:00
giovanni 9ab4e1a6ef adjust total bobot laporan keuangan 2026-06-08 12:37:39 +07:00
Giovanni Gabriel Septriadi 0a900986e7 Merge branch 'fix/depretiatio-response' into 'development'
adjust response depretitation v2

See merge request mbugroup/lti-api!610
2026-06-08 05:32:28 +00:00
giovanni 217f35b250 adjust response depretitation v2 2026-06-08 12:30:51 +07:00
Giovanni Gabriel Septriadi b3887b8d08 Merge branch 'hotfix/manual-inputs' into 'production'
fix data manual input; remove update manual input from crud recording

See merge request mbugroup/lti-api!608
2026-06-07 15:20:01 +00:00
Giovanni Gabriel Septriadi 2ddfa57aed Merge branch 'hotfix/manual-inputs' into 'development'
Hotfix/manual inputs

See merge request mbugroup/lti-api!609
2026-06-07 15:01:22 +00:00
giovanni 085d2f9bfe fix data manual input; remove update manual input from crud recording 2026-06-07 21:59:23 +07:00
Giovanni Gabriel Septriadi 61d375a59a Merge branch 'development' into 'production'
Development

See merge request mbugroup/lti-api!607
2026-06-07 12:18:00 +00:00
Giovanni Gabriel Septriadi 09242a6998 Merge branch 'fix/hpp-per-farm' into 'development'
adjust hpp per farm query to take feed and ovk

See merge request mbugroup/lti-api!606
2026-06-07 11:58:01 +00:00
Giovanni Gabriel Septriadi 7639e30326 Merge branch 'fix/recording-population' into 'development'
Fix/recording population

See merge request mbugroup/lti-api!605
2026-06-07 11:56:47 +00:00
giovanni 2216f572c2 fix recording standar prod laying 2026-06-07 18:55:24 +07:00
giovanni edfd6ac95c add command for normalize data recording population not match; adjust closing overhead and keuangan 2026-06-07 16:34:22 +07:00
giovanni aa3e655a67 adjust hpp per farm query to take feed and ovk 2026-06-06 10:29:33 +07:00
Giovanni Gabriel Septriadi 98bfdac3c5 Merge branch 'development' into 'production'
Development

See merge request mbugroup/lti-api!603
2026-06-06 01:47:16 +00:00
Giovanni Gabriel Septriadi a98d026ccb Merge branch 'feat/hpp-per-farm' into 'development'
adjust list marketing

See merge request mbugroup/lti-api!604
2026-06-06 01:40:01 +00:00
giovanni c3eab60f49 adjust list marketing 2026-06-06 08:38:38 +07:00
Giovanni Gabriel Septriadi e455889dae Merge branch 'feat/hpp-per-farm' into 'development'
Feat/hpp per farm

See merge request mbugroup/lti-api!602
2026-06-06 01:20:25 +00:00
Giovanni Gabriel Septriadi be8b99e7e8 Merge branch 'feat/depresiasi-v2' into 'development'
adjust

See merge request mbugroup/lti-api!601
2026-06-06 01:12:45 +00:00
giovanni 5760bb6de8 adjust 2026-06-06 08:12:06 +07:00
Giovanni Gabriel Septriadi 1e8651b8f2 Merge branch 'fix/migration-do' into 'development'
fix over consume by code, revert migration overconsume sell

See merge request mbugroup/lti-api!600
2026-06-06 00:53:25 +00:00
Giovanni Gabriel Septriadi efe9f0ce3c Merge branch 'feat/depresiasi-v2' into 'development'
Feat/depresiasi v2

See merge request mbugroup/lti-api!599
2026-06-05 06:51:58 +00:00
giovanni 1ef32407f1 create api get depresiasi v2 2026-06-05 13:51:09 +07:00
Giovanni Gabriel Septriadi 1cd72e5598 Merge branch 'fix/daily-checklist-fk' into 'development'
Fix/daily checklist fk

See merge request mbugroup/lti-api!596
2026-06-05 06:00:54 +00:00
Giovanni Gabriel Septriadi 7f701511d3 Merge branch 'feat/cut-over-depresiasi' into 'development'
Feat/cut over depresiasi

See merge request mbugroup/lti-api!589
2026-06-04 07:14:49 +00:00
Giovanni Gabriel Septriadi 9405c9d64b Merge branch 'feat/patch-chickindate' into 'development'
add api PATCH for edit chickin date

See merge request mbugroup/lti-api!588
2026-06-03 04:57:50 +00:00
Giovanni Gabriel Septriadi b179ed2bc9 Merge branch 'feat/overselling-telur' into 'development'
add validasi overselling telur

See merge request mbugroup/lti-api!587
2026-06-03 03:28:17 +00:00
giovanni 255e6a16d3 add validate query param 2026-06-03 09:43:34 +07:00
giovanni 93ed89b4ef ini api per farm 2026-06-03 00:30:41 +07:00
Rivaldi A N S b9201c2a4f Merge branch 'feat/marketing-filter-range-date' into 'development'
[FEAT][BE] Marketing Filter Range Date

See merge request mbugroup/lti-api!586
2026-06-02 09:50:02 +00:00
Rivaldi A N S f443686505 Merge branch 'feat/marketing-filter-range-date' into 'development'
[FEAT][BE] Marketing Filter Range Date

See merge request mbugroup/lti-api!585
2026-06-02 06:27:57 +00:00
Giovanni Gabriel Septriadi 9d8d54bd3c Merge branch 'fix/recording-chickin' into 'development'
Fix/recording chickin

See merge request mbugroup/lti-api!584
2026-06-02 03:40:59 +00:00
Giovanni Gabriel Septriadi 791c5880fd Merge branch 'fix/reconcile-fifo' into 'development'
Fix/reconcile fifo

See merge request mbugroup/lti-api!580
2026-06-01 14:03:06 +00:00
Giovanni Gabriel Septriadi 0581bf4a17 Merge branch 'feat/export-marketing' into 'development'
filter warehouse id to marketing; export recording add detail eggs; adjust format export marketing; adjust resposne list marketing

See merge request mbugroup/lti-api!579
2026-06-01 13:53:01 +00:00
Giovanni Gabriel Septriadi 1a5dfbb162 Merge branch 'fix/week-recording' into 'development'
Fix/week recording

See merge request mbugroup/lti-api!576
2026-05-30 03:05:08 +00:00
Giovanni Gabriel Septriadi b28ffdf9c6 Merge branch 'feat/trf-dep' into 'development'
Feat/trf dep

See merge request mbugroup/lti-api!574
2026-05-29 14:50:10 +00:00
Giovanni Gabriel Septriadi 90a921ff46 Merge branch 'feat/db' into 'development'
add command for cleanup relesed stock allocations

See merge request mbugroup/lti-api!569
2026-05-29 10:41:33 +00:00
Giovanni Gabriel Septriadi 2c9ae1d5ab Merge branch 'fix/nomor-po' into 'development'
Fix/nomor po

See merge request mbugroup/lti-api!568
2026-05-29 10:23:22 +00:00
Giovanni Gabriel Septriadi bf93770798 Merge branch 'feat/fifo-ar' into 'development'
init ar fifo

See merge request mbugroup/lti-api!566
2026-05-28 19:10:33 +00:00
Giovanni Gabriel Septriadi 98d031cc18 Merge branch 'fix/jamali' into 'development'
Fix/jamali

See merge request mbugroup/lti-api!564
2026-05-28 15:59:40 +00:00
34 changed files with 2742 additions and 239 deletions
@@ -0,0 +1,266 @@
// Command normalize-recording-cutover-depletion
//
// Data-only normalization of recording population metrics for a cut-over flock
// where pre-cutover mortality (culling + dead) was booked via stock adjustments
// (which do NOT feed the recording population). It applies an "opening depletion"
// offset to the CUMULATIVE depletion of every recording in a project_flock_kandang,
// recomputing the population-dependent metric columns DIRECTLY on the `recordings`
// table.
//
// It does NOT touch recording_depletions, stock_allocations, product_warehouses,
// project_flock_populations, or adjustment_stocks — so inventory/FIFO stay intact
// (the existing adjustments keep owning the stock movement).
//
// Recomputed columns (per recording, ordered by record_datetime,id):
//
// cumDepByDate = running SUM(recording_depletions.qty) up to that recording (INVARIANT)
// new_tcq = initialChickin - cumDepByDate - opening
// cum_depletion_rate = (cumDepByDate + opening) / initialChickin * 100
// feed_intake = feed_intake_old * (old_total_chick_qty / new_tcq) [null->null]
// fcr_value = fcr_value_old * (old_total_chick_qty / new_tcq) [null->null]
//
// cum_intake and egg-based metrics are left untouched (see plan).
//
// Idempotent: only rows where total_chick_qty IS DISTINCT FROM new_tcq are updated.
// Self-check: run with -opening=0; consistent rows are no-ops, any row that changes
// was already inconsistent (stale) and gets reconciled to follow the depletion data.
//
// Usage:
//
// DB_HOST=localhost DB_PORT=5542 go run ./cmd/normalize-recording-cutover-depletion/ -pfk=91 -opening=0 # self-check dry-run
// DB_HOST=localhost DB_PORT=5542 go run ./cmd/normalize-recording-cutover-depletion/ -pfk=91 -opening=3126 # dry-run
// DB_HOST=localhost DB_PORT=5542 go run ./cmd/normalize-recording-cutover-depletion/ -pfk=91 -opening=3126 -apply # apply
package main
import (
"flag"
"fmt"
"log"
"math"
"os"
"text/tabwriter"
"gitlab.com/mbugroup/lti-api.git/internal/config"
"gitlab.com/mbugroup/lti-api.git/internal/database"
"gorm.io/gorm"
)
type recRow struct {
ID uint `gorm:"column:id"`
Day *int `gorm:"column:day"`
RecordDate string `gorm:"column:record_date"`
TotalChickQty *float64 `gorm:"column:total_chick_qty"`
CumDepletionRate *float64 `gorm:"column:cum_depletion_rate"`
FeedIntake *float64 `gorm:"column:feed_intake"`
FcrValue *float64 `gorm:"column:fcr_value"`
CumDepByDate float64 `gorm:"column:cum_dep_by_date"`
}
const eps = 1e-6
func main() {
var (
pfk uint
opening float64
apply bool
chickinOverride float64
)
flag.UintVar(&pfk, "pfk", 0, "project_flock_kandangs_id (required)")
flag.Float64Var(&opening, "opening", 0, "opening depletion qty added to cumulative depletion of every recording")
flag.BoolVar(&apply, "apply", false, "apply changes (default: dry-run)")
flag.Float64Var(&chickinOverride, "chickin", 0, "override initial chickin base (0 = auto SUM project_chickins.usage_qty)")
flag.Parse()
if pfk == 0 {
log.Fatal("-pfk is required")
}
db := database.Connect(config.DBHost, config.DBName)
// 1) initial chickin base
var initialChickin float64
if chickinOverride > 0 {
initialChickin = chickinOverride
} else {
if err := db.Raw(
`SELECT COALESCE(SUM(usage_qty),0) FROM project_chickins WHERE project_flock_kandang_id = ?`, pfk,
).Scan(&initialChickin).Error; err != nil {
log.Fatalf("query initial chickin: %v", err)
}
}
if initialChickin <= 0 {
log.Fatalf("initial chickin <= 0 for pfk %d (got %.3f)", pfk, initialChickin)
}
// 2) sanity: duplicate record_datetime would make cumulative-by-date ambiguous
var dupDatetimes int64
if err := db.Raw(
`SELECT COUNT(*) FROM (
SELECT record_datetime FROM recordings
WHERE project_flock_kandangs_id = ? AND deleted_at IS NULL
GROUP BY record_datetime HAVING COUNT(*) > 1
) t`, pfk,
).Scan(&dupDatetimes).Error; err != nil {
log.Fatalf("check duplicate datetimes: %v", err)
}
if dupDatetimes > 0 {
fmt.Printf("WARNING: %d duplicate record_datetime group(s) for pfk %d — cumulative-by-date ordering may be ambiguous; review carefully.\n\n", dupDatetimes, pfk)
}
// 3) load recordings + running cumulative depletion (by record_datetime, id)
var rows []recRow
q := `
WITH dep AS (
SELECT r.id, r.day, r.record_datetime,
r.total_chick_qty, r.cum_depletion_rate, r.feed_intake, r.fcr_value,
COALESCE((SELECT SUM(rd.qty) FROM recording_depletions rd WHERE rd.recording_id = r.id), 0) AS daily_dep
FROM recordings r
WHERE r.project_flock_kandangs_id = ? AND r.deleted_at IS NULL
)
SELECT id, day,
to_char(record_datetime, 'YYYY-MM-DD') AS record_date,
total_chick_qty, cum_depletion_rate, feed_intake, fcr_value,
SUM(daily_dep) OVER (ORDER BY record_datetime, id) AS cum_dep_by_date
FROM dep
ORDER BY record_datetime, id`
if err := db.Raw(q, pfk).Scan(&rows).Error; err != nil {
log.Fatalf("query recordings: %v", err)
}
if len(rows) == 0 {
log.Fatalf("no recordings found for pfk %d", pfk)
}
mode := "DRY-RUN"
if apply {
mode = "APPLY"
}
fmt.Printf("=== normalize-recording-cutover-depletion ===\n")
fmt.Printf("Mode: %s | pfk=%d | initialChickin=%.3f | opening=%.3f | recordings=%d\n\n", mode, pfk, initialChickin, opening, len(rows))
tw := tabwriter.NewWriter(os.Stdout, 0, 2, 2, ' ', 0)
fmt.Fprintln(tw, "id\tday\tdate\ttcq_old->new\tcumRate_old->new\tfeed_old->new\tfcr_old->new\tstatus")
var willChange, anomalies, skipped int
var negTcq int
for _, r := range rows {
newTcq := initialChickin - r.CumDepByDate - opening
newRate := (r.CumDepByDate + opening) / initialChickin * 100
status := ""
// detect pre-existing inconsistency (stale row): old tcq != invariant base (opening=0 expectation)
expectedBase := initialChickin - r.CumDepByDate
if r.TotalChickQty == nil || math.Abs(*r.TotalChickQty-expectedBase) > 1e-3 {
status = "ANOMALY"
anomalies++
}
if newTcq < -eps {
status = "NEG_TCQ!"
negTcq++
}
// idempotent guard
if r.TotalChickQty != nil && math.Abs(*r.TotalChickQty-newTcq) < 1e-6 {
if status == "" {
status = "noop"
}
skipped++
} else {
willChange++
}
var newFeed, newFcr *float64
if r.FeedIntake != nil && r.TotalChickQty != nil && math.Abs(newTcq) > eps {
v := *r.FeedIntake * (*r.TotalChickQty / newTcq)
newFeed = &v
} else {
newFeed = r.FeedIntake
}
if r.FcrValue != nil && r.TotalChickQty != nil && math.Abs(newTcq) > eps {
v := *r.FcrValue * (*r.TotalChickQty / newTcq)
newFcr = &v
} else {
newFcr = r.FcrValue
}
fmt.Fprintf(tw, "%d\t%s\t%s\t%s -> %.3f\t%s -> %.3f\t%s -> %s\t%s -> %s\t%s\n",
r.ID, iptr(r.Day), r.RecordDate,
fptr(r.TotalChickQty), newTcq,
fptr(r.CumDepletionRate), newRate,
fptr(r.FeedIntake), fptrV(newFeed),
fptr(r.FcrValue), fptrV(newFcr),
status,
)
}
tw.Flush()
fmt.Printf("\nSummary: will_change=%d skipped(noop)=%d anomalies=%d neg_tcq=%d\n", willChange, skipped, anomalies, negTcq)
if negTcq > 0 {
log.Fatalf("ABORT: %d recording(s) would get negative total_chick_qty — opening too large or data issue", negTcq)
}
if !apply {
fmt.Println("\nDry-run only. Re-run with -apply to persist.")
return
}
// 4) APPLY — single set-based UPDATE in a transaction (RHS uses pre-update column values)
err := db.Transaction(func(tx *gorm.DB) error {
res := tx.Exec(`
WITH dep AS (
SELECT r.id, r.record_datetime,
COALESCE((SELECT SUM(rd.qty) FROM recording_depletions rd WHERE rd.recording_id = r.id), 0) AS daily_dep
FROM recordings r
WHERE r.project_flock_kandangs_id = ? AND r.deleted_at IS NULL
),
calc AS (
SELECT id,
(? - cum_dep - ?) AS new_tcq,
((cum_dep + ?) / ? * 100) AS new_rate
FROM (
SELECT id, SUM(daily_dep) OVER (ORDER BY record_datetime, id) AS cum_dep
FROM dep
) s
)
UPDATE recordings r SET
total_chick_qty = c.new_tcq,
cum_depletion_rate = c.new_rate,
feed_intake = CASE WHEN r.feed_intake IS NULL OR r.total_chick_qty IS NULL OR c.new_tcq = 0
THEN r.feed_intake ELSE r.feed_intake * (r.total_chick_qty / c.new_tcq) END,
fcr_value = CASE WHEN r.fcr_value IS NULL OR r.total_chick_qty IS NULL OR c.new_tcq = 0
THEN r.fcr_value ELSE r.fcr_value * (r.total_chick_qty / c.new_tcq) END,
updated_at = NOW()
FROM calc c
WHERE r.id = c.id
AND r.total_chick_qty IS DISTINCT FROM c.new_tcq`,
pfk,
initialChickin, opening,
opening, initialChickin,
)
if res.Error != nil {
return res.Error
}
fmt.Printf("\nAPPLIED: %d recording row(s) updated.\n", res.RowsAffected)
return nil
})
if err != nil {
log.Fatalf("apply failed: %v", err)
}
fmt.Println("Done. Verify with the queries in tmp/pfk91-cutover-fix.md.")
}
func fptr(p *float64) string {
if p == nil {
return "null"
}
return fmt.Sprintf("%.3f", *p)
}
func fptrV(p *float64) string { return fptr(p) }
func iptr(p *int) string {
if p == nil {
return "-"
}
return fmt.Sprintf("%d", *p)
}
@@ -96,6 +96,7 @@ type HppV2FarmDepreciationSnapshotRow struct {
DepreciationPercentEffective float64 DepreciationPercentEffective float64
DepreciationValue float64 DepreciationValue float64
PulletCostDayNTotal float64 PulletCostDayNTotal float64
Components []byte
} }
type HppV2CostRepository interface { type HppV2CostRepository interface {
@@ -114,6 +115,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 +374,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
@@ -397,7 +405,7 @@ func (r *HppV2RepositoryImpl) GetFarmDepreciationSnapshotByProjectFlockIDAndPeri
var row HppV2FarmDepreciationSnapshotRow var row HppV2FarmDepreciationSnapshotRow
err := r.db.WithContext(ctx). err := r.db.WithContext(ctx).
Table("farm_depreciation_snapshots"). Table("farm_depreciation_snapshots").
Select("id, project_flock_id, period_date, depreciation_percent_effective, depreciation_value, pullet_cost_day_n_total"). Select("id, project_flock_id, period_date, depreciation_percent_effective, depreciation_value, pullet_cost_day_n_total, components").
Where("project_flock_id = ?", projectFlockID). Where("project_flock_id = ?", projectFlockID).
Where("period_date = DATE(?)", periodDate). Where("period_date = DATE(?)", periodDate).
Limit(1). Limit(1).
@@ -585,6 +593,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 {
+247 -25
View File
@@ -2,6 +2,7 @@ package service
import ( import (
"context" "context"
"encoding/json"
"time" "time"
commonRepo "gitlab.com/mbugroup/lti-api.git/internal/common/repository" commonRepo "gitlab.com/mbugroup/lti-api.git/internal/common/repository"
@@ -55,6 +56,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 +458,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 +467,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 +742,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 +784,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 +946,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 +999,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,
hppV2PartLayingFarm,
"Laying Farm",
[]string{hppV2ScopeProductionCost},
&HppV2Proration{
Basis: basis, Basis: basis,
Numerator: numerator, Numerator: numerator,
Denominator: denominator, Denominator: denominator,
Ratio: ratio, Ratio: ratio,
}, }, nil
ratio, }
), nil
// GetExpenseProductionScopeRange menghitung BOP production_cost satu komponen expense untuk rentang
// [startDate, endDate] secara range-correct (tidak pernah negatif untuk expense non-negatif).
// - laying-direct (ratio 1, monoton): selisih kumulatif end - start.
// - laying-farm (prorated): (expenseCum(end) - expenseCum(start)) × ratio(end).
//
// 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(
@@ -1334,6 +1473,18 @@ func (s *hppV2Service) buildFarmSnapshotDepreciationPart(
depreciationPercent = (appliedDepreciation / appliedPulletCostDayN) * 100 depreciationPercent = (appliedDepreciation / appliedPulletCostDayN) * 100
} }
details := map[string]any{
"basis_total": snapshot.DepreciationValue,
"pullet_cost_day_n": appliedPulletCostDayN,
"depreciation_percent": depreciationPercent,
"snapshot_id": snapshot.ID,
"snapshot_period_date": formatDateOnly(snapshot.PeriodDate),
"snapshot_project_flock": snapshot.ProjectFlockID,
}
for key, value := range farmDepreciationSnapshotMetadata(snapshot.Components, projectFlockKandangId) {
details[key] = value
}
return &HppV2ComponentPart{ return &HppV2ComponentPart{
Code: hppV2PartDepreciationFarmSnapshot, Code: hppV2PartDepreciationFarmSnapshot,
Title: "Farm Snapshot", Title: "Farm Snapshot",
@@ -1345,14 +1496,7 @@ func (s *hppV2Service) buildFarmSnapshotDepreciationPart(
Denominator: denominator, Denominator: denominator,
Ratio: ratio, Ratio: ratio,
}, },
Details: map[string]any{ Details: details,
"basis_total": snapshot.DepreciationValue,
"pullet_cost_day_n": appliedPulletCostDayN,
"depreciation_percent": depreciationPercent,
"snapshot_id": snapshot.ID,
"snapshot_period_date": formatDateOnly(snapshot.PeriodDate),
"snapshot_project_flock": snapshot.ProjectFlockID,
},
References: []HppV2Reference{ References: []HppV2Reference{
{ {
Type: "farm_depreciation_snapshot", Type: "farm_depreciation_snapshot",
@@ -1366,6 +1510,84 @@ func (s *hppV2Service) buildFarmSnapshotDepreciationPart(
}, nil }, nil
} }
type farmDepreciationSnapshotComponents struct {
Kandang []farmDepreciationSnapshotKandangComponent `json:"kandang"`
}
type farmDepreciationSnapshotKandangComponent struct {
ProjectFlockKandangID uint `json:"project_flock_kandang_id"`
DayN int `json:"day_n"`
MultiplicationPercent float64 `json:"multiplication_percentage"`
ChickinDate string `json:"chickin_date"`
OriginDate string `json:"origin_date"`
StandardEffectiveDate string `json:"standard_effective_date"`
Population float64 `json:"population"`
}
func farmDepreciationSnapshotMetadata(raw []byte, projectFlockKandangID uint) map[string]any {
result := make(map[string]any)
if len(raw) == 0 {
return result
}
var components farmDepreciationSnapshotComponents
if err := json.Unmarshal(raw, &components); err != nil {
return result
}
var fallback *farmDepreciationSnapshotKandangComponent
for i := range components.Kandang {
component := &components.Kandang[i]
if !component.hasDepreciationMetadata() {
continue
}
if component.ProjectFlockKandangID == projectFlockKandangID {
return component.snapshotDetails()
}
if fallback == nil {
fallback = component
}
}
if fallback != nil {
return fallback.snapshotDetails()
}
return result
}
func (c farmDepreciationSnapshotKandangComponent) hasDepreciationMetadata() bool {
return c.DayN > 0 ||
c.MultiplicationPercent > 0 ||
c.ChickinDate != "" ||
c.OriginDate != "" ||
c.StandardEffectiveDate != "" ||
c.Population > 0
}
func (c farmDepreciationSnapshotKandangComponent) snapshotDetails() map[string]any {
chickinDate := c.ChickinDate
if chickinDate == "" {
chickinDate = c.OriginDate
}
details := map[string]any{
"schedule_day": c.DayN,
"multiplication_percentage": c.MultiplicationPercent,
}
if chickinDate != "" {
details["origin_date"] = chickinDate
details["chickin_date"] = chickinDate
}
if c.StandardEffectiveDate != "" {
details["standard_effective_date"] = c.StandardEffectiveDate
}
if c.Population > 0 {
details["kandang_population"] = c.Population
}
return details
}
func (s *hppV2Service) buildNormalTransferDepreciationPart( func (s *hppV2Service) buildNormalTransferDepreciationPart(
contextRow *commonRepo.HppV2ProjectFlockKandangContext, contextRow *commonRepo.HppV2ProjectFlockKandangContext,
transferInput *commonRepo.HppV2LatestTransferInputRow, transferInput *commonRepo.HppV2LatestTransferInputRow,
@@ -25,6 +25,10 @@ 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
// expenseRowsByFarmDateKey (opsional) membuat ListExpenseRealizationRowsByProjectFlockID
// date-aware untuk menguji perhitungan range BOP. Bila non-nil, dipakai menggantikan
// expenseRowsByFarmKey; key = "<flock>|<ekspedisi>|<YYYY-MM-DD>".
expenseRowsByFarmDateKey map[string][]commonRepo.HppV2ExpenseCostRow
routeCostByProject map[uint]float64 routeCostByProject map[uint]float64
totalPopulationByKey map[string]float64 totalPopulationByKey map[string]float64
transferSummaryByPFK map[uint]struct { transferSummaryByPFK map[uint]struct {
@@ -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
} }
@@ -814,6 +825,28 @@ func TestHppV2CalculateHppBreakdown_UsesFarmSnapshotDepreciationProratedByEggPro
DepreciationPercentEffective: 10, DepreciationPercentEffective: 10,
DepreciationValue: 1000, DepreciationValue: 1000,
PulletCostDayNTotal: 10000, PulletCostDayNTotal: 10000,
Components: []byte(`{
"kandang_count": 2,
"total_population": 1000,
"kandang": [
{
"project_flock_kandang_id": 71,
"day_n": 5,
"multiplication_percentage": 0.95,
"chickin_date": "2026-01-02",
"standard_effective_date": "2026-06-01",
"population": 800
},
{
"project_flock_kandang_id": 70,
"day_n": 7,
"multiplication_percentage": 0.93,
"chickin_date": "2026-01-01",
"standard_effective_date": "2026-06-02",
"population": 200
}
]
}`),
}, },
}, },
eggProductionByPFK: map[uint]struct { eggProductionByPFK: map[uint]struct {
@@ -862,6 +895,21 @@ func TestHppV2CalculateHppBreakdown_UsesFarmSnapshotDepreciationProratedByEggPro
if depreciation.Parts[0].Details["snapshot_id"] != uint(901) { if depreciation.Parts[0].Details["snapshot_id"] != uint(901) {
t.Fatalf("expected snapshot id 901, got %+v", depreciation.Parts[0].Details) t.Fatalf("expected snapshot id 901, got %+v", depreciation.Parts[0].Details)
} }
if depreciation.Parts[0].Details["schedule_day"] != 7 {
t.Fatalf("expected snapshot schedule_day 7, got %+v", depreciation.Parts[0].Details)
}
if depreciation.Parts[0].Details["multiplication_percentage"] != 0.93 {
t.Fatalf("expected snapshot multiplication_percentage 0.93, got %+v", depreciation.Parts[0].Details)
}
if depreciation.Parts[0].Details["chickin_date"] != "2026-01-01" {
t.Fatalf("expected snapshot chickin_date 2026-01-01, got %+v", depreciation.Parts[0].Details)
}
if depreciation.Parts[0].Details["standard_effective_date"] != "2026-06-02" {
t.Fatalf("expected snapshot standard_effective_date 2026-06-02, got %+v", depreciation.Parts[0].Details)
}
if depreciation.Parts[0].Details["kandang_population"] != float64(200) {
t.Fatalf("expected snapshot kandang_population 200, got %+v", depreciation.Parts[0].Details)
}
} }
func stubKey(ids []uint, flags []string) string { func stubKey(ids []uint, flags []string) string {
@@ -904,6 +952,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)
}
}
@@ -0,0 +1,14 @@
-- Reverse UPSERT: hapus baris PFK 47 & 48 yang kemungkinan baru diinsert oleh up migration ini.
-- Jika sebelumnya sudah ada (ON CONFLICT DO UPDATE), baris ini akan terhapus —
-- restore manual dari backup jika diperlukan.
DELETE FROM farm_depreciation_manual_inputs
WHERE project_flock_id IN (47, 48);
-- UPDATE rows untuk PFK 427 tidak bisa di-reverse secara presisi:
-- nilai total_cost sebelum migration ini tidak tersimpan di migration history
-- (data awal di-load via cmd/import-farm-depreciation-manual-inputs dari Excel).
-- PFK 10 dan 11 tidak berubah (nilai sama dengan state dari migration 20260529144559).
-- Jika perlu rollback penuh: restore dari database backup atau re-import Excel lama.
-- Recompute snapshots setelah rollback
TRUNCATE TABLE farm_depreciation_snapshots;
@@ -0,0 +1,105 @@
UPDATE farm_depreciation_manual_inputs
SET total_cost = 1900157533.55,
cutover_date = DATE '2026-02-28',
updated_at = NOW()
WHERE project_flock_id = 10;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 146658321.066,
cutover_date = DATE '2026-02-28',
updated_at = NOW()
WHERE project_flock_id = 13;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 51824694.138,
cutover_date = DATE '2026-02-28',
updated_at = NOW()
WHERE project_flock_id = 17;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 15491774.796,
cutover_date = DATE '2026-02-28',
updated_at = NOW()
WHERE project_flock_id = 8;
-- Cutover 2026-02-28 (lanjutan)
UPDATE farm_depreciation_manual_inputs
SET total_cost = 575074391.36, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 4;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 578360642.51, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 5;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 880983605.92, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 6;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 391669576.153, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 9;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 2521797832.14, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 11;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 139227054.164, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 12;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 380083106.836, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 14;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 705136853.847, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 15;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 209816474.000, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 18;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 557606867.000, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 19;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 239330456.11, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 20;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 4724203916.72, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 26;
-- Cutover 2026-05-15
UPDATE farm_depreciation_manual_inputs
SET total_cost = 5449963647.43, cutover_date = DATE '2026-05-15', updated_at = NOW()
WHERE project_flock_id = 27;
-- Cutover 2026-06-08 (upsert — row mungkin belum ada)
INSERT INTO farm_depreciation_manual_inputs (project_flock_id, total_cost, cutover_date, created_at, updated_at)
VALUES (47, 5395429899.42, DATE '2026-06-08', NOW(), NOW())
ON CONFLICT (project_flock_id) DO UPDATE
SET total_cost = EXCLUDED.total_cost,
cutover_date = EXCLUDED.cutover_date,
updated_at = NOW();
-- Cutover 2026-06-16 (upsert — row mungkin belum ada)
INSERT INTO farm_depreciation_manual_inputs (project_flock_id, total_cost, cutover_date, created_at, updated_at)
VALUES (48, 5514616442.08, DATE '2026-06-16', NOW(), NOW())
ON CONFLICT (project_flock_id) DO UPDATE
SET total_cost = EXCLUDED.total_cost,
cutover_date = EXCLUDED.cutover_date,
updated_at = NOW();
-- Pengaman: pastikan snapshot di-recompute dengan total_cost baru
-- saat user request /api/reports/expense/depreciation
TRUNCATE TABLE farm_depreciation_snapshots;
@@ -0,0 +1,14 @@
-- Reverse UPSERT: hapus baris PFK 47 & 48 yang kemungkinan baru diinsert oleh up migration ini.
-- Jika sebelumnya sudah ada (ON CONFLICT DO UPDATE), baris ini akan terhapus —
-- restore manual dari backup jika diperlukan.
DELETE FROM farm_depreciation_manual_inputs
WHERE project_flock_id IN (47, 48);
-- UPDATE rows untuk PFK 427 tidak bisa di-reverse secara presisi:
-- nilai total_cost sebelum migration ini tidak tersimpan di migration history
-- (data awal di-load via cmd/import-farm-depreciation-manual-inputs dari Excel).
-- PFK 10 dan 11 tidak berubah (nilai sama dengan state dari migration 20260529144559).
-- Jika perlu rollback penuh: restore dari database backup atau re-import Excel lama.
-- Recompute snapshots setelah rollback
TRUNCATE TABLE farm_depreciation_snapshots;
@@ -0,0 +1,105 @@
UPDATE farm_depreciation_manual_inputs
SET total_cost = 1900157533.55,
cutover_date = DATE '2026-02-28',
updated_at = NOW()
WHERE project_flock_id = 10;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 146658321.066,
cutover_date = DATE '2026-02-28',
updated_at = NOW()
WHERE project_flock_id = 13;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 51824694.138,
cutover_date = DATE '2026-02-28',
updated_at = NOW()
WHERE project_flock_id = 17;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 15491774.796,
cutover_date = DATE '2026-02-28',
updated_at = NOW()
WHERE project_flock_id = 8;
-- Cutover 2026-02-28 (lanjutan)
UPDATE farm_depreciation_manual_inputs
SET total_cost = 575074391.36, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 4;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 578360642.51, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 5;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 880983605.92, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 6;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 391669576.153, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 9;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 2521797832.14, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 11;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 139227054.164, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 12;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 380083106.836, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 14;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 705136853.847, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 15;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 209816474.000, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 18;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 557606867.000, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 19;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 239330456.11, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 20;
UPDATE farm_depreciation_manual_inputs
SET total_cost = 4724203916.72, cutover_date = DATE '2026-02-28', updated_at = NOW()
WHERE project_flock_id = 26;
-- Cutover 2026-05-15
UPDATE farm_depreciation_manual_inputs
SET total_cost = 5449963647.43, cutover_date = DATE '2026-05-15', updated_at = NOW()
WHERE project_flock_id = 27;
-- Cutover 2026-06-08 (upsert — row mungkin belum ada)
INSERT INTO farm_depreciation_manual_inputs (project_flock_id, total_cost, cutover_date, created_at, updated_at)
VALUES (47, 5395429899.42, DATE '2026-06-08', NOW(), NOW())
ON CONFLICT (project_flock_id) DO UPDATE
SET total_cost = EXCLUDED.total_cost,
cutover_date = EXCLUDED.cutover_date,
updated_at = NOW();
-- Cutover 2026-06-16 (upsert — row mungkin belum ada)
INSERT INTO farm_depreciation_manual_inputs (project_flock_id, total_cost, cutover_date, created_at, updated_at)
VALUES (48, 5514616442.08, DATE '2026-06-16', NOW(), NOW())
ON CONFLICT (project_flock_id) DO UPDATE
SET total_cost = EXCLUDED.total_cost,
cutover_date = EXCLUDED.cutover_date,
updated_at = NOW();
-- Pengaman: pastikan snapshot di-recompute dengan total_cost baru
-- saat user request /api/reports/expense/depreciation
TRUNCATE TABLE farm_depreciation_snapshots;
@@ -0,0 +1,2 @@
UPDATE phases SET is_active = true
WHERE id IN (2, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26);
@@ -0,0 +1,2 @@
UPDATE phases SET is_active = false
WHERE id IN (2, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26);
@@ -789,11 +789,56 @@ func (s closingService) GetOverhead(c *fiber.Ctx, projectFlockID uint, projectFl
totalActualPopulation := totalChickinQty - totalDepletion totalActualPopulation := totalChickinQty - totalDepletion
// Prefer recording-based population (recordings.total_chick_qty) so closing stays
// consistent with normalized cut-over flocks. For normal flocks this equals
// chickin - depletion (no-op); it only differs when the recording population was
// normalized separately from recording_depletions. Falls back if any kandang in
// scope lacks a recording.
scopeKandangs := projectFlockKandangs
if projectFlockKandangID != nil {
scopeKandangs = nil
for _, k := range projectFlockKandangs {
if k.Id == *projectFlockKandangID {
scopeKandangs = append(scopeKandangs, k)
break
}
}
}
if recPop, ok := s.actualPopulationFromRecordings(c.Context(), scopeKandangs); ok {
totalActualPopulation = recPop
}
result := dto.ToOverheadListDTOs(budgets, realizations, totalChickinQty, totalActualPopulation, projectFlockKandangID != nil, totalKandangCount) result := dto.ToOverheadListDTOs(budgets, realizations, totalChickinQty, totalActualPopulation, projectFlockKandangID != nil, totalKandangCount)
return &result, nil return &result, nil
} }
// actualPopulationFromRecordings sums the latest recordings.total_chick_qty across the
// given kandangs (the production population source of truth). Returns ok=false if any
// kandang lacks a recording, so the caller falls back to chickin-minus-depletion.
// For normal flocks this equals chickin - depletion; it only differs for cut-over flocks
// whose recording population was normalized separately from recording_depletions.
func (s closingService) actualPopulationFromRecordings(ctx context.Context, kandangs []entity.ProjectFlockKandang) (float64, bool) {
if s.RecordingRepo == nil || len(kandangs) == 0 {
return 0, false
}
total := 0.0
for _, k := range kandangs {
latest, err := s.RecordingRepo.GetLatestByProjectFlockKandangID(ctx, k.Id)
if err != nil {
s.Log.Warnf("actualPopulationFromRecordings: latest recording pfk=%d: %v", k.Id, err)
return 0, false
}
if latest == nil || latest.TotalChickQty == nil {
return 0, false
}
if *latest.TotalChickQty > 0 {
total += *latest.TotalChickQty
}
}
return total, true
}
type activeKandangMetricRow struct { type activeKandangMetricRow struct {
ProjectFlockKandangID uint `gorm:"column:project_flock_kandang_id"` ProjectFlockKandangID uint `gorm:"column:project_flock_kandang_id"`
ProjectFlockID uint `gorm:"column:project_flock_id"` ProjectFlockID uint `gorm:"column:project_flock_id"`
@@ -156,7 +156,7 @@ func (s closingKeuanganService) calculateClosingKeuangan(c *fiber.Ctx, projectFl
hppSection := s.buildHPPSection(c, projectFlock, projectFlockKandangs, costs, productionData) hppSection := s.buildHPPSection(c, projectFlock, projectFlockKandangs, costs, productionData)
profitLossSection := s.buildProfitLossSection(projectFlock, costs, productionData) profitLossSection := s.buildProfitLossSection(c, projectFlock, projectFlockKandangs, costs, productionData)
data := dto.ToClosingKeuanganData(hppSection, profitLossSection) data := dto.ToClosingKeuanganData(hppSection, profitLossSection)
return &data, nil return &data, nil
@@ -386,7 +386,7 @@ func (s closingKeuanganService) buildHPPSection(c *fiber.Ctx, projectFlock *enti
return dto.ToHPPSection(hppItems, hppSummary) return dto.ToHPPSection(hppItems, hppSummary)
} }
func (s closingKeuanganService) buildProfitLossSection(projectFlock *entity.ProjectFlock, costs *CostData, production *ProductionData) dto.ProfitLossSection { func (s closingKeuanganService) buildProfitLossSection(c *fiber.Ctx, projectFlock *entity.ProjectFlock, projectFlockKandangs []entity.ProjectFlockKandang, costs *CostData, production *ProductionData) dto.ProfitLossSection {
totalWeightProduced := production.TotalWeightProduced totalWeightProduced := production.TotalWeightProduced
totalEggWeightKg := production.TotalEggWeightKg totalEggWeightKg := production.TotalEggWeightKg
@@ -394,6 +394,11 @@ func (s closingKeuanganService) buildProfitLossSection(projectFlock *entity.Proj
totalWeightSold := production.TotalWeightSold totalWeightSold := production.TotalWeightSold
totalBirdSold := production.TotalBirdSold totalBirdSold := production.TotalBirdSold
actualPopulation := production.TotalPopulationIn - production.TotalDepletion actualPopulation := production.TotalPopulationIn - production.TotalDepletion
// Prefer recording-based population (consistent with buildHPPSection) so per-ekor
// P&L matches the normalized recording population for cut-over flocks.
if lastPopulation, ok := s.getLastPopulationFromRecordings(c, projectFlockKandangs); ok {
actualPopulation = lastPopulation
}
isLaying := projectFlock.Category == string(utils.ProjectFlockCategoryLaying) isLaying := projectFlock.Category == string(utils.ProjectFlockCategoryLaying)
@@ -1215,7 +1215,9 @@ func (s dailyChecklistService) AssignPhases(c *fiber.Ctx, id uint, req *validati
} }
if len(phaseIDs) > 0 { if len(phaseIDs) > 0 {
phases, err := s.PhaseRepo.GetByIDs(c.Context(), phaseIDs, nil) phases, err := s.PhaseRepo.GetByIDs(c.Context(), phaseIDs, func(db *gorm.DB) *gorm.DB {
return db.Where("is_active = true")
})
if err != nil { if err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) { if errors.Is(err, gorm.ErrRecordNotFound) {
return fiber.NewError(fiber.StatusBadRequest, "Phase not found") return fiber.NewError(fiber.StatusBadRequest, "Phase not found")
@@ -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,
@@ -50,6 +50,7 @@ func (s phasesService) GetAll(c *fiber.Ctx, params *validation.Query) ([]entity.
phasess, total, err := s.Repository.GetAll(c.Context(), offset, params.Limit, func(db *gorm.DB) *gorm.DB { phasess, total, err := s.Repository.GetAll(c.Context(), offset, params.Limit, func(db *gorm.DB) *gorm.DB {
db = s.withRelations(db) db = s.withRelations(db)
db = db.Where("is_active = true")
if params.Search != "" { if params.Search != "" {
return db.Where("name ILIKE ?", "%"+params.Search+"%") return db.Where("name ILIKE ?", "%"+params.Search+"%")
} }
@@ -387,35 +387,87 @@ func (s productionStandardService) EnsureWeekAvailable(ctx context.Context, stan
return nil return nil
} }
week := ((day - 1) / 7) + 1 requestedWeek := ((day - 1) / 7) + 1
if week <= 0 { if requestedWeek <= 0 {
return nil return nil
} }
upperCategory := strings.ToUpper(category) upperCategory := strings.ToUpper(category)
if upperCategory == string(utils.ProjectFlockCategoryLaying) { if upperCategory == string(utils.ProjectFlockCategoryLaying) {
detail, err := s.ProductionStandardDetailRepo.GetByStandardIDAndWeek(ctx, standardID, week) effectiveWeek := requestedWeek
firstCommonWeek, ok, err := s.layingFirstCommonStandardWeek(ctx, standardID)
if err != nil { if err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) {
return fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Standart production tidak tersedia untuk week %d", week))
}
return err return err
} }
if detail == nil { if ok && requestedWeek < firstCommonWeek {
return fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Standart production tidak tersedia untuk week %d", week)) effectiveWeek = firstCommonWeek
}
detail, err := s.ProductionStandardDetailRepo.GetByStandardIDAndWeek(ctx, standardID, effectiveWeek)
if err != nil {
if !errors.Is(err, gorm.ErrRecordNotFound) {
return err
} }
} }
growthDetail, err := s.StandardGrowthDetailRepo.GetByStandardIDAndWeek(ctx, standardID, week) growthDetail, err := s.StandardGrowthDetailRepo.GetByStandardIDAndWeek(ctx, standardID, effectiveWeek)
if err != nil {
if !errors.Is(err, gorm.ErrRecordNotFound) {
return err
}
}
if detail != nil && growthDetail != nil {
return nil
}
return fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Standart production tidak tersedia untuk week %d", requestedWeek))
}
growthDetail, err := s.StandardGrowthDetailRepo.GetByStandardIDAndWeek(ctx, standardID, requestedWeek)
if err != nil { if err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) { if errors.Is(err, gorm.ErrRecordNotFound) {
return fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Standart production tidak tersedia untuk week %d", week)) return fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Standart production tidak tersedia untuk week %d", requestedWeek))
} }
return err return err
} }
if growthDetail == nil { if growthDetail == nil {
return fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Standart production tidak tersedia untuk week %d", week)) return fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Standart production tidak tersedia untuk week %d", requestedWeek))
} }
return nil return nil
} }
func (s productionStandardService) layingFirstCommonStandardWeek(ctx context.Context, standardID uint) (int, bool, error) {
details, err := s.ProductionStandardDetailRepo.GetByProductionStandardID(ctx, standardID)
if err != nil {
return 0, false, err
}
detailWeeks := make(map[int]struct{}, len(details))
for _, detail := range details {
if detail.Week <= 0 {
continue
}
detailWeeks[detail.Week] = struct{}{}
}
growthDetails, err := s.StandardGrowthDetailRepo.GetByProductionStandardID(ctx, standardID)
if err != nil {
return 0, false, err
}
firstCommonWeek := 0
for _, detail := range growthDetails {
if detail.Week <= 0 {
continue
}
if _, ok := detailWeeks[detail.Week]; !ok {
continue
}
if firstCommonWeek == 0 || detail.Week < firstCommonWeek {
firstCommonWeek = detail.Week
}
}
return firstCommonWeek, firstCommonWeek > 0, nil
}
@@ -0,0 +1,95 @@
package service
import (
"context"
"strings"
"testing"
"github.com/glebarez/sqlite"
repositories "gitlab.com/mbugroup/lti-api.git/internal/modules/master/production-standards/repositories"
"gitlab.com/mbugroup/lti-api.git/internal/utils"
"gorm.io/gorm"
)
func TestEnsureWeekAvailableAllowsLayingBeforeFirstCommonStandardWeek(t *testing.T) {
svc := setupProductionStandardServiceTest(t)
if err := svc.EnsureWeekAvailable(context.Background(), 1, string(utils.ProjectFlockCategoryLaying), 85); err != nil {
t.Fatalf("expected pre-standard laying week to be allowed, got %v", err)
}
}
func TestEnsureWeekAvailableRejectsLayingMissingWeekAfterStandardStarts(t *testing.T) {
svc := setupProductionStandardServiceTest(t)
err := svc.EnsureWeekAvailable(context.Background(), 1, string(utils.ProjectFlockCategoryLaying), 127)
if err == nil {
t.Fatal("expected missing laying standard week to be rejected")
}
if !strings.Contains(err.Error(), "week 19") {
t.Fatalf("expected error to mention requested week 19, got %v", err)
}
}
func TestEnsureWeekAvailableKeepsGrowingWeekStrict(t *testing.T) {
svc := setupProductionStandardServiceTest(t)
err := svc.EnsureWeekAvailable(context.Background(), 2, string(utils.ProjectFlockCategoryGrowing), 8)
if err == nil {
t.Fatal("expected missing growing standard week to be rejected")
}
if !strings.Contains(err.Error(), "week 2") {
t.Fatalf("expected error to mention requested week 2, got %v", err)
}
}
func setupProductionStandardServiceTest(t *testing.T) productionStandardService {
t.Helper()
db, err := gorm.Open(sqlite.Open("file:"+t.Name()+"?mode=memory&cache=private"), &gorm.Config{})
if err != nil {
t.Fatalf("failed opening sqlite db: %v", err)
}
statements := []string{
`CREATE TABLE production_standard_details (
id INTEGER PRIMARY KEY,
production_standard_id INTEGER NOT NULL,
week INTEGER NOT NULL,
target_hen_day_production NUMERIC NULL,
target_hen_house_production NUMERIC NULL,
target_egg_weight NUMERIC NULL,
target_egg_mass NUMERIC NULL,
standard_fcr NUMERIC NULL,
created_at TIMESTAMP NULL,
updated_at TIMESTAMP NULL
)`,
`CREATE TABLE standard_growth_details (
id INTEGER PRIMARY KEY,
production_standard_id INTEGER NOT NULL,
target_mean_bw NUMERIC NULL,
max_depletion NUMERIC NULL,
min_uniformity NUMERIC NOT NULL,
week INTEGER NOT NULL,
feed_intake NUMERIC NULL,
created_at TIMESTAMP NULL,
created_by INTEGER NOT NULL
)`,
`INSERT INTO production_standard_details (id, production_standard_id, week, standard_fcr) VALUES
(1, 1, 18, 2.1)`,
`INSERT INTO standard_growth_details (id, production_standard_id, week, min_uniformity, created_by) VALUES
(1, 1, 18, 80, 1),
(2, 2, 1, 80, 1)`,
}
for _, stmt := range statements {
if err := db.Exec(stmt).Error; err != nil {
t.Fatalf("failed preparing schema: %v", err)
}
}
return productionStandardService{
ProductionStandardDetailRepo: repositories.NewProductionStandardDetailRepository(db),
StandardGrowthDetailRepo: repositories.NewStandardGrowthDetailRepository(db),
}
}
@@ -34,7 +34,6 @@ import (
"github.com/gofiber/fiber/v2" "github.com/gofiber/fiber/v2"
"github.com/sirupsen/logrus" "github.com/sirupsen/logrus"
"gorm.io/gorm" "gorm.io/gorm"
"gorm.io/gorm/clause"
) )
type RecordingService interface { type RecordingService interface {
@@ -586,10 +585,6 @@ func (s *recordingService) CreateOne(c *fiber.Ctx, req *validation.Create) (*ent
s.Log.Errorf("Failed to recalculate recordings after create: %+v", err) s.Log.Errorf("Failed to recalculate recordings after create: %+v", err)
return err return err
} }
if err := s.syncFarmDepreciationManualInputFromRecordingStocks(ctx, tx, createdRecording.ProjectFlockKandangId, createdRecording.RecordDatetime); err != nil {
s.Log.Errorf("Failed to sync farm depreciation manual input after create: %+v", err)
return err
}
action := entity.ApprovalActionCreated action := entity.ApprovalActionCreated
if err := s.createRecordingApproval(ctx, tx, createdRecording.Id, utils.RecordingStepPengajuan, action, createdRecording.CreatedBy, nil); err != nil { if err := s.createRecordingApproval(ctx, tx, createdRecording.Id, utils.RecordingStepPengajuan, action, createdRecording.CreatedBy, nil); err != nil {
@@ -892,12 +887,6 @@ func (s recordingService) UpdateOne(c *fiber.Ctx, req *validation.Update, id uin
return err return err
} }
} }
if hasStockChanges {
if err := s.syncFarmDepreciationManualInputFromRecordingStocks(ctx, tx, recordingEntity.ProjectFlockKandangId, recordingEntity.RecordDatetime); err != nil {
s.Log.Errorf("Failed to sync farm depreciation manual input after update: %+v", err)
return err
}
}
action := entity.ApprovalActionUpdated action := entity.ApprovalActionUpdated
actorID := recordingEntity.CreatedBy actorID := recordingEntity.CreatedBy
@@ -1159,10 +1148,6 @@ func (s recordingService) DeleteOne(c *fiber.Ctx, id uint) error {
s.Log.Errorf("Failed to recalculate recordings after delete: %+v", err) s.Log.Errorf("Failed to recalculate recordings after delete: %+v", err)
return err return err
} }
if err := s.syncFarmDepreciationManualInputFromRecordingStocks(ctx, tx, recording.ProjectFlockKandangId, recording.RecordDatetime); err != nil {
s.Log.Errorf("Failed to sync farm depreciation manual input after delete: %+v", err)
return err
}
s.invalidateDepreciationSnapshots(ctx, tx, recording.ProjectFlockKandangId, recording.RecordDatetime) s.invalidateDepreciationSnapshots(ctx, tx, recording.ProjectFlockKandangId, recording.RecordDatetime)
return nil return nil
@@ -1949,172 +1934,6 @@ func (s *recordingService) getEarliestChickInDateByProjectFlockKandangID(ctx con
return row.ChickInDate, nil return row.ChickInDate, nil
} }
func (s *recordingService) syncFarmDepreciationManualInputFromRecordingStocks(
ctx context.Context,
tx *gorm.DB,
projectFlockKandangID uint,
fallbackCutoverDate time.Time,
) error {
if projectFlockKandangID == 0 {
return nil
}
targetDB := s.Repository.DB()
if tx != nil {
targetDB = tx
}
projectFlockID, err := s.resolveProjectFlockIDByProjectFlockKandangID(ctx, targetDB, projectFlockKandangID)
if err != nil {
return err
}
if projectFlockID == 0 {
return nil
}
totalCost, err := s.sumNoTransferRecordingStockCostByProjectFlockID(ctx, targetDB, projectFlockID)
if err != nil {
return err
}
existing, err := s.getFarmDepreciationManualInputByProjectFlockID(ctx, targetDB, projectFlockID)
if err != nil {
return err
}
cutoverDate := normalizeDateOnlyUTC(fallbackCutoverDate)
if existing != nil && !existing.CutoverDate.IsZero() {
cutoverDate = normalizeDateOnlyUTC(existing.CutoverDate)
}
if cutoverDate.IsZero() {
earliestDate, dateErr := s.getEarliestNoTransferRecordingDateByProjectFlockID(ctx, targetDB, projectFlockID)
if dateErr != nil {
return dateErr
}
if earliestDate != nil && !earliestDate.IsZero() {
cutoverDate = normalizeDateOnlyUTC(*earliestDate)
}
}
if cutoverDate.IsZero() {
cutoverDate = normalizeDateOnlyUTC(time.Now().UTC())
}
now := time.Now().UTC()
row := entity.FarmDepreciationManualInput{
ProjectFlockId: projectFlockID,
TotalCost: totalCost,
CutoverDate: cutoverDate,
}
if existing != nil {
row.Note = existing.Note
}
return targetDB.WithContext(ctx).
Clauses(clause.OnConflict{
Columns: []clause.Column{{Name: "project_flock_id"}},
DoUpdates: clause.Assignments(map[string]any{
"total_cost": row.TotalCost,
"cutover_date": row.CutoverDate,
"updated_at": now,
}),
}).
Create(&row).Error
}
func (s *recordingService) resolveProjectFlockIDByProjectFlockKandangID(ctx context.Context, db *gorm.DB, projectFlockKandangID uint) (uint, error) {
var row struct {
ProjectFlockID uint `gorm:"column:project_flock_id"`
}
err := db.WithContext(ctx).
Table("project_flock_kandangs").
Select("project_flock_id").
Where("id = ?", projectFlockKandangID).
Take(&row).Error
if errors.Is(err, gorm.ErrRecordNotFound) {
return 0, nil
}
if err != nil {
return 0, err
}
return row.ProjectFlockID, nil
}
func (s *recordingService) sumNoTransferRecordingStockCostByProjectFlockID(ctx context.Context, db *gorm.DB, projectFlockID uint) (float64, error) {
if projectFlockID == 0 {
return 0, nil
}
var total float64
err := db.WithContext(ctx).
Table("recording_stocks AS rs").
Select("COALESCE(SUM(sa.qty * COALESCE(pi.price, 0)), 0)").
Joins("JOIN recordings AS r ON r.id = rs.recording_id AND r.deleted_at IS NULL").
Joins("JOIN project_flock_kandangs AS pfk ON pfk.id = r.project_flock_kandangs_id").
Joins(
"JOIN stock_allocations AS sa ON sa.usable_type = ? AND sa.usable_id = rs.id AND sa.stockable_type = ? AND sa.status = ? AND sa.allocation_purpose = ?",
fifo.UsableKeyRecordingStock.String(),
fifo.StockableKeyPurchaseItems.String(),
entity.StockAllocationStatusActive,
entity.StockAllocationPurposeConsume,
).
Joins("JOIN purchase_items AS pi ON pi.id = sa.stockable_id").
Where("pfk.project_flock_id = ?", projectFlockID).
Where("rs.project_flock_kandang_id IS NULL").
Scan(&total).Error
if err != nil {
return 0, err
}
return total, nil
}
func (s *recordingService) getFarmDepreciationManualInputByProjectFlockID(
ctx context.Context,
db *gorm.DB,
projectFlockID uint,
) (*entity.FarmDepreciationManualInput, error) {
if projectFlockID == 0 {
return nil, nil
}
var row entity.FarmDepreciationManualInput
err := db.WithContext(ctx).
Where("project_flock_id = ?", projectFlockID).
Take(&row).Error
if errors.Is(err, gorm.ErrRecordNotFound) {
return nil, nil
}
if err != nil {
return nil, err
}
return &row, nil
}
func (s *recordingService) getEarliestNoTransferRecordingDateByProjectFlockID(
ctx context.Context,
db *gorm.DB,
projectFlockID uint,
) (*time.Time, error) {
if projectFlockID == 0 {
return nil, nil
}
var row struct {
RecordDate *time.Time `gorm:"column:record_date"`
}
err := db.WithContext(ctx).
Table("recording_stocks AS rs").
Select("MIN(r.record_datetime) AS record_date").
Joins("JOIN recordings AS r ON r.id = rs.recording_id AND r.deleted_at IS NULL").
Joins("JOIN project_flock_kandangs AS pfk ON pfk.id = r.project_flock_kandangs_id").
Where("pfk.project_flock_id = ?", projectFlockID).
Where("rs.project_flock_kandang_id IS NULL").
Scan(&row).Error
if err != nil {
return nil, err
}
return row.RecordDate, nil
}
func (s *recordingService) resolveEggRequestsToFarmWarehouses( func (s *recordingService) resolveEggRequestsToFarmWarehouses(
ctx context.Context, ctx context.Context,
pfk *entity.ProjectFlockKandang, pfk *entity.ProjectFlockKandang,
@@ -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,28 @@ 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"`
}
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,51 @@
package dto
import (
"encoding/json"
"testing"
)
func TestExpenseDepreciationRowDTOComponentsJSONContract(t *testing.T) {
v1 := ExpenseDepreciationRowDTO{
ProjectFlockID: 1,
FarmName: "Farm A",
Period: "2026-06-05",
Components: map[string]any{"kandang_count": 1},
}
rawV1, err := json.Marshal(v1)
if err != nil {
t.Fatalf("marshal v1 dto: %v", err)
}
var decodedV1 map[string]any
if err := json.Unmarshal(rawV1, &decodedV1); err != nil {
t.Fatalf("unmarshal v1 dto: %v", err)
}
if _, ok := decodedV1["components"]; !ok {
t.Fatalf("expected v1 components to be present, got %s", string(rawV1))
}
v2 := ExpenseDepreciationV2RowDTO{
Date: "2026-06-05",
DepreciationPercentEffective: 10,
DepreciationValue: 100,
PulletCostDayNTotal: 1000,
MultiplicationPercentage: 0.9,
DayN: 2,
ChickinDate: "2026-01-01",
TotalValuePulletAfterDepreciation: 900,
TotalPopulation: 100,
}
rawV2, err := json.Marshal(v2)
if err != nil {
t.Fatalf("marshal v2 dto: %v", err)
}
var decodedV2 map[string]any
if err := json.Unmarshal(rawV2, &decodedV2); err != nil {
t.Fatalf("unmarshal v2 dto: %v", err)
}
if _, ok := decodedV2["components"]; ok {
t.Fatalf("expected v2 components to be omitted, got %s", string(rawV2))
}
}
@@ -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,
}
}
@@ -83,7 +83,7 @@ func ToMarketingReportItems(mdps []entity.MarketingDeliveryProduct, hppByDeliver
realizationDate = *mdp.DeliveryDate realizationDate = *mdp.DeliveryDate
} }
totalWeightKg := mdp.UsageQty * mdp.AvgWeight totalWeightKg := mdp.TotalWeight
salesAmount := totalWeightKg * mdp.UnitPrice salesAmount := totalWeightKg * mdp.UnitPrice
var hpp float64 var hpp float64
@@ -0,0 +1,71 @@
package dto
import (
"math"
"testing"
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
)
func TestToMarketingReportItemsUsesDeliveryProductTotalWeight(t *testing.T) {
mdps := []entity.MarketingDeliveryProduct{
{
Id: 1,
UsageQty: 10,
AvgWeight: 2.5,
TotalWeight: 17.75,
UnitPrice: 1000,
},
}
got := ToMarketingReportItems(mdps, nil, nil, nil)
if len(got) != 1 {
t.Fatalf("expected 1 marketing report item, got %d", len(got))
}
if got[0].TotalWeightKg != 17.75 {
t.Fatalf("expected total_weight_kg to use delivery product total_weight 17.75, got %.2f", got[0].TotalWeightKg)
}
if got[0].Qty != 10 {
t.Fatalf("expected qty to stay from usage_qty, got %.2f", got[0].Qty)
}
if got[0].AverageWeightKg != 2.5 {
t.Fatalf("expected average_weight_kg to stay from avg_weight, got %.2f", got[0].AverageWeightKg)
}
if got[0].SalesAmount != 17750 {
t.Fatalf("expected sales_amount to use delivery product total_weight, got %.2f", got[0].SalesAmount)
}
}
func TestMarketingSummaryUsesReportItemTotalWeight(t *testing.T) {
items := []RepportMarketingItemDTO{
{
Qty: 10,
TotalWeightKg: 17.75,
SalesAmount: 17750,
},
{
Qty: 5,
TotalWeightKg: 8.25,
SalesAmount: 8250,
},
}
got := ToSummaryFromDTOItems(items)
if got == nil {
t.Fatal("expected summary, got nil")
}
if got.TotalWeightKg != 26 {
t.Fatalf("expected summary total_weight_kg to sum item total weights, got %.2f", got.TotalWeightKg)
}
if diff := math.Abs(got.AverageWeightKg - (26.0 / 15.0)); diff > 0.000001 {
t.Fatalf("expected summary average_weight_kg to use total_weight_kg / total_qty, got %.6f", got.AverageWeightKg)
}
if got.TotalQty != 15 {
t.Fatalf("expected total qty 15, got %d", got.TotalQty)
}
if got.TotalSalesAmount != 26000 {
t.Fatalf("expected total sales amount 26000, got %d", got.TotalSalesAmount)
}
}
+2
View File
@@ -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
}
+2
View File
@@ -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)
}
}
@@ -90,6 +90,12 @@ func (m *expenseDepreciationRepoMock) DeleteSnapshotsFromDate(_ context.Context,
return nil return nil
} }
func (m *expenseDepreciationRepoMock) DeleteSnapshotsByFarmIDs(_ context.Context, farmIDs []uint) error {
m.deleteCalled = true
m.deleteFarmIDs = append([]uint{}, farmIDs...)
return nil
}
func (m *expenseDepreciationRepoMock) GetLatestManualInputsByFarms(_ context.Context, _ []int64, _ []int64, _ []int64) ([]repportRepo.FarmDepreciationManualInputRow, error) { func (m *expenseDepreciationRepoMock) GetLatestManualInputsByFarms(_ context.Context, _ []int64, _ []int64, _ []int64) ([]repportRepo.FarmDepreciationManualInputRow, error) {
return append([]repportRepo.FarmDepreciationManualInputRow{}, m.manualInputs...), nil return append([]repportRepo.FarmDepreciationManualInputRow{}, m.manualInputs...), nil
} }
@@ -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,145 @@ 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 multiplicationPercentage float64
var dayN int
var chickinDate string
var standardEffectiveDate string
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
}
partPulletCostDayN := hppV2DetailFloat(part.Details, "pullet_cost_day_n")
partPopulation := hppV2DetailFloat(part.Details, "kandang_population")
partDayN := hppV2DetailInt(part.Details, "schedule_day")
partMultiplicationPercentage := hppV2DetailFloat(part.Details, "multiplication_percentage")
partChickinDate := hppV2DetailString(part.Details, "chickin_date")
if partChickinDate == "" {
partChickinDate = hppV2DetailString(part.Details, "origin_date")
}
totalPulletCostDayN += partPulletCostDayN
totalDepreciationValue += part.Total
totalPopulation += partPopulation
if dayN == 0 && multiplicationPercentage == 0 && chickinDate == "" &&
(partDayN > 0 || partMultiplicationPercentage > 0 || partChickinDate != "") {
dayN = partDayN
multiplicationPercentage = partMultiplicationPercentage
chickinDate = partChickinDate
standardEffectiveDate = hppV2DetailString(part.Details, "standard_effective_date")
}
}
}
effectivePercent := approvalService.CalculateEffectiveDepreciationPercent(totalDepreciationValue, totalPulletCostDayN)
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,
})
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 +3089,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 +3697,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"`
@@ -205,6 +205,7 @@ func AttachProductionStandards(ctx context.Context, db *gorm.DB, warnOnly bool,
standardDetailByStd := make(map[uint]map[int]*entity.ProductionStandardDetail, len(standardIDs)) standardDetailByStd := make(map[uint]map[int]*entity.ProductionStandardDetail, len(standardIDs))
growthDetailByStd := make(map[uint]map[int]*entity.StandardGrowthDetail, len(standardIDs)) growthDetailByStd := make(map[uint]map[int]*entity.StandardGrowthDetail, len(standardIDs))
firstCommonWeekByStd := make(map[uint]int, len(standardIDs))
for standardID := range standardIDs { for standardID := range standardIDs {
details, err := standardDetailRepo.GetByProductionStandardID(ctx, standardID) details, err := standardDetailRepo.GetByProductionStandardID(ctx, standardID)
@@ -242,6 +243,10 @@ func AttachProductionStandards(ctx context.Context, db *gorm.DB, warnOnly bool,
growthMap[growth.Week] = &growth growthMap[growth.Week] = &growth
} }
growthDetailByStd[standardID] = growthMap growthDetailByStd[standardID] = growthMap
if firstCommonWeek, ok := firstCommonStandardWeek(detailMap, growthMap); ok {
firstCommonWeekByStd[standardID] = firstCommonWeek
}
} }
// Batch-load laying transfer targets → EARLIEST source PFK chick_in_date per target. // Batch-load laying transfer targets → EARLIEST source PFK chick_in_date per target.
@@ -284,6 +289,9 @@ func AttachProductionStandards(ctx context.Context, db *gorm.DB, warnOnly bool,
continue continue
} }
week := computeTransferAwareWeek(item, sourceChickInByTarget) week := computeTransferAwareWeek(item, sourceChickInByTarget)
if firstCommonWeek, ok := firstCommonWeekByStd[standardID]; ok {
week = effectiveProductionStandardWeek(item, week, firstCommonWeek)
}
item.StandardWeek = &week item.StandardWeek = &week
cacheKey := standardKey{standardID: standardID, week: week} cacheKey := standardKey{standardID: standardID, week: week}
if cached, ok := cache[cacheKey]; ok { if cached, ok := cache[cacheKey]; ok {
@@ -324,6 +332,38 @@ func applyProductionStandardValues(item *entity.Recording, values productionStan
item.StandardFcr = fcr item.StandardFcr = fcr
} }
func firstCommonStandardWeek(
detailMap map[int]*entity.ProductionStandardDetail,
growthMap map[int]*entity.StandardGrowthDetail,
) (int, bool) {
firstWeek := 0
for week := range detailMap {
if week <= 0 {
continue
}
if _, ok := growthMap[week]; !ok {
continue
}
if firstWeek == 0 || week < firstWeek {
firstWeek = week
}
}
return firstWeek, firstWeek > 0
}
func effectiveProductionStandardWeek(item *entity.Recording, actualWeek int, firstCommonWeek int) int {
if item == nil || actualWeek <= 0 || firstCommonWeek <= 0 {
return actualWeek
}
if !IsLayingRecording(*item) {
return actualWeek
}
if actualWeek < firstCommonWeek {
return firstCommonWeek
}
return actualWeek
}
// collectLayingPFKIDs mengumpulkan semua project_flock_kandang_id dari recording laying // collectLayingPFKIDs mengumpulkan semua project_flock_kandang_id dari recording laying
func collectLayingPFKIDs(items []*entity.Recording) []uint { func collectLayingPFKIDs(items []*entity.Recording) []uint {
seen := make(map[uint]struct{}) seen := make(map[uint]struct{})
@@ -1,10 +1,15 @@
package recording package recording
import ( import (
"context"
"testing" "testing"
"time"
"github.com/glebarez/sqlite"
entity "gitlab.com/mbugroup/lti-api.git/internal/entities" entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/production/recordings/validations" validation "gitlab.com/mbugroup/lti-api.git/internal/modules/production/recordings/validations"
"gitlab.com/mbugroup/lti-api.git/internal/utils"
"gorm.io/gorm"
) )
func TestMapDepletionsKeepsSourceWarehouseRoutes(t *testing.T) { func TestMapDepletionsKeepsSourceWarehouseRoutes(t *testing.T) {
@@ -45,3 +50,126 @@ func TestMapEggsSetsProjectFlockKandangID(t *testing.T) {
t.Fatalf("expected project flock kandang id 44, got %+v", got[0].ProjectFlockKandangId) t.Fatalf("expected project flock kandang id 44, got %+v", got[0].ProjectFlockKandangId)
} }
} }
func TestAttachProductionStandardsClampsLayingPreStandardWeek(t *testing.T) {
db := setupAttachProductionStandardTestDB(t)
day := 91
recordDate := time.Date(2026, 4, 2, 0, 0, 0, 0, time.UTC)
chickInDate := time.Date(2026, 1, 1, 0, 0, 0, 0, time.UTC)
recording := &entity.Recording{
Id: 501,
ProjectFlockKandangId: 103,
RecordDatetime: recordDate,
Day: &day,
ProjectFlockKandang: &entity.ProjectFlockKandang{
Id: 103,
ProjectFlock: entity.ProjectFlock{
Id: 52,
Category: string(utils.ProjectFlockCategoryLaying),
ProductionStandardId: 1,
ProductionStandard: entity.ProductionStandard{
Id: 1,
Name: "STD Laying",
},
},
},
}
actualWeek := computeTransferAwareWeek(recording, map[uint]time.Time{103: chickInDate})
if actualWeek != 13 {
t.Fatalf("expected actual transfer-aware week 13, got %d", actualWeek)
}
if err := AttachProductionStandards(context.Background(), db, false, nil, recording); err != nil {
t.Fatalf("expected attach standard to succeed, got %v", err)
}
if recording.Day == nil || *recording.Day != 91 {
t.Fatalf("expected actual recording day to remain 91, got %+v", recording.Day)
}
if recording.StandardWeek == nil || *recording.StandardWeek != 18 {
t.Fatalf("expected effective standard week 18, got %+v", recording.StandardWeek)
}
if recording.StandardFeedIntake == nil || *recording.StandardFeedIntake != 120 {
t.Fatalf("expected feed intake std from week 18, got %+v", recording.StandardFeedIntake)
}
if recording.StandardHenDay == nil || *recording.StandardHenDay != 80 {
t.Fatalf("expected hen day std from week 18, got %+v", recording.StandardHenDay)
}
if recording.StandardFcr == nil || *recording.StandardFcr != 2.1 {
t.Fatalf("expected fcr std from week 18, got %+v", recording.StandardFcr)
}
}
func setupAttachProductionStandardTestDB(t *testing.T) *gorm.DB {
t.Helper()
db, err := gorm.Open(sqlite.Open("file:"+t.Name()+"?mode=memory&cache=private"), &gorm.Config{})
if err != nil {
t.Fatalf("failed opening sqlite db: %v", err)
}
statements := []string{
`CREATE TABLE production_standard_details (
id INTEGER PRIMARY KEY,
production_standard_id INTEGER NOT NULL,
week INTEGER NOT NULL,
target_hen_day_production NUMERIC NULL,
target_hen_house_production NUMERIC NULL,
target_egg_weight NUMERIC NULL,
target_egg_mass NUMERIC NULL,
standard_fcr NUMERIC NULL,
created_at TIMESTAMP NULL,
updated_at TIMESTAMP NULL
)`,
`CREATE TABLE standard_growth_details (
id INTEGER PRIMARY KEY,
production_standard_id INTEGER NOT NULL,
target_mean_bw NUMERIC NULL,
max_depletion NUMERIC NULL,
min_uniformity NUMERIC NOT NULL,
week INTEGER NOT NULL,
feed_intake NUMERIC NULL,
created_at TIMESTAMP NULL,
created_by INTEGER NOT NULL
)`,
`CREATE TABLE laying_transfer_targets (
id INTEGER PRIMARY KEY,
laying_transfer_id INTEGER NOT NULL,
target_project_flock_kandang_id INTEGER NOT NULL,
deleted_at TIMESTAMP NULL
)`,
`CREATE TABLE laying_transfers (
id INTEGER PRIMARY KEY,
source_project_flock_kandang_id INTEGER NULL,
deleted_at TIMESTAMP NULL
)`,
`CREATE TABLE project_chickins (
id INTEGER PRIMARY KEY,
project_flock_kandang_id INTEGER NOT NULL,
chick_in_date TIMESTAMP NOT NULL,
deleted_at TIMESTAMP NULL
)`,
`INSERT INTO production_standard_details
(id, production_standard_id, week, target_hen_day_production, target_hen_house_production, target_egg_weight, target_egg_mass, standard_fcr)
VALUES (1, 1, 18, 80, 70, 55, 44, 2.1)`,
`INSERT INTO standard_growth_details
(id, production_standard_id, week, feed_intake, max_depletion, min_uniformity, created_by)
VALUES (1, 1, 18, 120, 1.5, 80, 1)`,
`INSERT INTO laying_transfers (id, source_project_flock_kandang_id, deleted_at) VALUES
(77, 83, NULL)`,
`INSERT INTO laying_transfer_targets (id, laying_transfer_id, target_project_flock_kandang_id, deleted_at) VALUES
(88, 77, 103, NULL)`,
`INSERT INTO project_chickins (id, project_flock_kandang_id, chick_in_date, deleted_at) VALUES
(99, 83, '2026-01-01 00:00:00', NULL)`,
}
for _, stmt := range statements {
if err := db.Exec(stmt).Error; err != nil {
t.Fatalf("failed preparing schema: %v", err)
}
}
return db
}